From 51d19ffeb4aaba874fadde019f29fc9bf6f30da5 Mon Sep 17 00:00:00 2001 From: Hendrik Eeckhaut Date: Tue, 25 Jun 2024 15:05:48 +0200 Subject: [PATCH] initial draft for tlsn-plugin boilerplate template --- .gitignore | 11 ++ README.md | 57 +++++++++++ assets/icon.png | Bin 0 -> 31426 bytes config.json | 39 ++++++++ esbuild.js | 85 ++++++++++++++++ examples/twitter_dm_js/README.md | 11 ++ examples/twitter_dm_js/index.d.ts | 14 +++ examples/twitter_dm_js/index.js | 127 +++++++++++++++++++++++ examples/twitter_profile_js/index.d.ts | 15 +++ examples/twitter_profile_js/index.js | 130 ++++++++++++++++++++++++ package.json | 17 ++++ src/index.d.ts | 15 +++ src/index.ts | 133 +++++++++++++++++++++++++ tsconfig.json | 14 +++ 14 files changed, 668 insertions(+) create mode 100644 .gitignore create mode 100644 README.md create mode 100644 assets/icon.png create mode 100644 config.json create mode 100644 esbuild.js create mode 100644 examples/twitter_dm_js/README.md create mode 100644 examples/twitter_dm_js/index.d.ts create mode 100644 examples/twitter_dm_js/index.js create mode 100644 examples/twitter_profile_js/index.d.ts create mode 100644 examples/twitter_profile_js/index.js create mode 100644 package.json create mode 100644 src/index.d.ts create mode 100644 src/index.ts create mode 100644 tsconfig.json diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..4429cc0 --- /dev/null +++ b/.gitignore @@ -0,0 +1,11 @@ +/target +**/*.rs.bk +Cargo.lock +bin/ +**/node_modules +**/.DS_Store +.idea +build +tlsn/ +zip +dist/ diff --git a/README.md b/README.md new file mode 100644 index 0000000..719b543 --- /dev/null +++ b/README.md @@ -0,0 +1,57 @@ +# Plugin Development for the TLSNotary Browser Extension + +This repository contains the boilerplate code for developing a plugin for the TLSNotary browser extension. Currently, the template includes a TypeScript-based plugin example that proves ownership of a Twitter profile. + +TLSNotary's plugin system uses [Extism](https://github.com/extism), which enables plugins in different programming languages. For more documentation, check [Extism's documentation](https://github.com/extism/js-pdk). + +## Installation of Extism-js + +1. **Download and Install Extism-js**: Begin by setting up `extism-js`, which enables you to compile and manage your plugins. Run these commands to download and install it: + + ```sh + curl -O https://raw.githubusercontent.com/extism/js-pdk/main/install.sh + sh install.sh + ``` + + This script installs the Extism JavaScript Plugin Development Kit from its GitHub repository, preparing your environment for plugin compilation. + +## Building the Twitter Profile Plugin + +To build the plugin, run: + +```sh +npm i +npm run build +``` + +This will output the wasm binary in `dist/index.wasm`. + +### Running the Twitter Plugin Example + +1. Build the `twitter_profile` plugin as explained above. +2. Build and install the `tlsn-extension` as documented in the [main README.md](../README.md). +3. [Run a local notary server](https://github.com/tlsnotary/tlsn/blob/main/notary-server/README.md), ensuring `TLS` is disabled in the [config file](https://github.com/tlsnotary/tlsn/blob/main/notary-server/config/config.yaml#L18). +4. Install the plugin: Click the **Add a Plugin (+)** button and select the `index.wasm` file you built in step 1. A **Twitter Profile** button should then appear below the default buttons. +5. Click the **Twitter Profile** button. This action opens the Twitter webpage along with a TLSNotary sidebar. +6. Follow the steps in the TLSNotary sidebar. +7. Access the TLSNotary results by clicking the **History** button in the TLSNotary extension. + +## Customize the Template + +You are now ready to develop your own plugins. + +Notarization requests take more time than regular requests, so we recommend testing the HTTPS request in Postman or Curl first. This allows you to figure out faster which cookies and headers are required. + +When you add or rename steps, make sure to update the `index.d.ts` file. + +### Custom Icon + +To use a custom icon, replace `icon.png` in the `./assets` folder. Make sure it is 320x320 pixels. You can use the following command: + +```sh +convert icon_source.png -resize 320x320! assets/icon.png +``` + +## More Examples + +Check the `examples` folder for more examples. \ No newline at end of file diff --git a/assets/icon.png b/assets/icon.png new file mode 100644 index 0000000000000000000000000000000000000000..ba58060a7a2c97fd3912cbfe513c2cc68a730296 GIT binary patch literal 31426 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 z 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 literal 0 HcmV?d00001 diff --git a/config.json b/config.json new file mode 100644 index 0000000..1f158b8 --- /dev/null +++ b/config.json @@ -0,0 +1,39 @@ +{ + "title": "Twitter Profile", + "description": "Notarize ownership of a twitter profile", + "steps": [ + { + "title": "Visit Twitter website", + "cta": "Go to x.com", + "action": "start" + }, + { + "title": "Collect credentials", + "description": "Login to your account if you haven't already", + "cta": "Check cookies", + "action": "two" + }, + { + "title": "Notarize twitter profile", + "cta": "Notarize", + "action": "three", + "prover": true + } + ], + "hostFunctions": [ + "redirect", + "notarize" + ], + "cookies": [ + "api.x.com" + ], + "headers": [ + "api.x.com" + ], + "requests": [ + { + "url": "https: //api.x.com/1.1/account/settings.json", + "method": "GET" + } + ] +} \ No newline at end of file diff --git a/esbuild.js b/esbuild.js new file mode 100644 index 0000000..5f290ec --- /dev/null +++ b/esbuild.js @@ -0,0 +1,85 @@ +const esbuild = require('esbuild'); +const fs = require('fs'); +const path = require('path'); +const { promisify } = require('util'); +const { name } = require('./package.json'); +const { execSync } = require('child_process'); + +// Promisify fs.readFile and fs.stat for convenience +const readFileAsync = promisify(fs.readFile); +const statAsync = promisify(fs.stat); +const mkdirAsync = promisify(fs.mkdir); + +/** + * Generates a Base64 encoded icon file. + * It checks if the output file already exists and is up-to-date before generating a new one. + */ +async function generateBase64Icon() { + const iconPath = path.join(__dirname, 'assets', "icon.png"); + const outputDir = path.join(__dirname, 'dist', 'assets'); + const outputPath = path.join(outputDir, 'icon.ts'); + + try { + // Ensure the output directory exists + await mkdirAsync(outputDir, { recursive: true }); + const [iconStat, outputStat] = await Promise.all([ + statAsync(iconPath).catch(() => null), + statAsync(outputPath).catch(() => null) + ]); + + // Check if output file exists and is newer than the icon file + if (outputStat && iconStat && outputStat.mtime > iconStat.mtime) { + console.log('Base64 icon file is up-to-date.'); + return; + } + + const fileBuffer = await readFileAsync(iconPath); + const base64Icon = `data:image/png;base64,${fileBuffer.toString('base64')}`; + + const outputContent = `// This is a generated file. Do not edit directly. +// This is a Base64 encoded version of the plugin's icon ('icon.png') used in the plugin's config. +// This file is automatically generated by esBuild.js whenever the icon is changed. +// There is no need to add it to version control. + +export const icon = "${base64Icon}";\n`; + + fs.writeFileSync(outputPath, outputContent); + console.log('Base64 icon file generated successfully.'); + } catch (error) { + console.error(`Failed to generate base64 icon: ${error.message}`); + process.exit(1); + } +} + +const outputDir = 'dist'; +const entryFile = 'src/index.ts'; +const outputFile = path.join(outputDir, 'index.js'); +const outputWasm = path.join(outputDir, `${name}.tlsn.wasm`); + +async function build() { + await generateBase64Icon(); + + try { + await esbuild.build({ + entryPoints: [entryFile], + bundle: true, + outdir: outputDir, // Use outdir for directory output + sourcemap: true, + minify: false, // might want to use true for production build + format: 'cjs', // needs to be CJS for now + target: ['es2020'], // don't go over es2020 because quickjs doesn't support it + }); + + console.log('esbuild completed successfully.'); + + // Run extism-js to generate the wasm file + const extismCommand = `extism-js ${outputFile} -i src/index.d.ts -o ${outputWasm}`; + execSync(extismCommand, { stdio: 'inherit' }); + console.log('extism-js completed successfully.'); + } catch (error) { + console.error('Build process failed:', error); + process.exit(1); + } +} + +build(); \ No newline at end of file diff --git a/examples/twitter_dm_js/README.md b/examples/twitter_dm_js/README.md new file mode 100644 index 0000000..633d978 --- /dev/null +++ b/examples/twitter_dm_js/README.md @@ -0,0 +1,11 @@ +# TLSNotary TypeScript plugin demo: Prove Twitter DM + +This is a demo demo plugin for the TLSNotary browser extension in plain Javascript. + +## Building + +Build the plugin: +``` +extism-js index.js -i index.d.ts -o index.wasm +``` +This command compiles the JavaScript code in index.js into a WebAssembly module, ready for integration with the TLSNotary extension. \ No newline at end of file diff --git a/examples/twitter_dm_js/index.d.ts b/examples/twitter_dm_js/index.d.ts new file mode 100644 index 0000000..283fe08 --- /dev/null +++ b/examples/twitter_dm_js/index.d.ts @@ -0,0 +1,14 @@ +declare module 'main' { + // Extism exports take no params and return an I32 + export function start(): I32; + export function two(): I32; + export function three(): I32; + export function config(): I32; +} + +declare module 'extism:host' { + interface user { + redirect(ptr: I64): void; + notarize(ptr: I64): I64; + } +} diff --git a/examples/twitter_dm_js/index.js b/examples/twitter_dm_js/index.js new file mode 100644 index 0000000..1ef57a2 --- /dev/null +++ b/examples/twitter_dm_js/index.js @@ -0,0 +1,127 @@ +function isCorrectUrl(urlString) { + const url = new URL(urlString); + return url.hostname === 'twitter.com' || url.hostname === 'x.com'; +} + +function extractConversationId(urlString) { + const url = new URL(urlString); + + // Validate the host and path pattern + if (url.hostname === 'x.com' && /\/messages\/[0-9]+-[0-9]+$/.test(url.pathname)) { + // Extract the Conversation ID from the path + return url.pathname.split('/messages/')[1]; + } + return null; // Return null if URL is not valid or does not match the expected format +} + +function gotoUrl() { + const { redirect } = Host.getFunctions(); + const mem = Memory.fromString('https://x.com/messages'); + redirect(mem.offset); +} + +function start() { + if (!isCorrectUrl(Config.get('tabUrl'))) { + gotoUrl(); + Host.outputString(JSON.stringify(false)); + return; + } + Host.outputString(JSON.stringify(true)); +} + +function two() { + const conversation_id = extractConversationId(Config.get('tabUrl')) + const cookies = JSON.parse(Config.get('cookies'))['x.com']; + const headers = JSON.parse(Config.get('headers'))['x.com']; + // console.log("TLSN cookies"); + // console.log(JSON.stringify(cookies)); + // console.log("TLSN headers"); + // console.log(JSON.stringify(headers)); + + if ( + !conversation_id || + !cookies.auth_token || + !cookies.ct0 || + !headers['x-csrf-token'] || + !headers['authorization'] + ) { + Host.outputString(JSON.stringify(false)); + return; + } + + Host.outputString( + JSON.stringify({ + url: `https://x.com/i/api/1.1/dm/conversation/${conversation_id}.json`, + method: 'GET', + headers: { + 'x-csrf-token': headers['x-csrf-token'], + Host: 'x.com', + authorization: headers.authorization, + Cookie: `lang=en; auth_token=${cookies.auth_token}; ct0=${cookies.ct0}`, + 'Accept-Encoding': 'identity', + Connection: 'close', + }, + secretHeaders: [ + `x-csrf-token: ${headers['x-csrf-token']}`, + `cookie: lang=en; auth_token=${cookies.auth_token}; ct0=${cookies.ct0}`, + `authorization: ${headers.authorization}`, + ], + }), + ); + +} + +function three() { + const params = JSON.parse(Host.inputString()); + const { notarize } = Host.getFunctions(); + + if (!params) { + Host.outputString(JSON.stringify(false)); + } else { + const mem = Memory.fromString(JSON.stringify(params)); + const idOffset = notarize(mem.offset); + const id = Memory.find(idOffset).readString(); + Host.outputString(JSON.stringify(id)); + } +} + +function config() { + Host.outputString( + JSON.stringify({ + title: 'Twitter Messages conversation', + description: 'Notarize a Twitter Messages conversation', + icon: '', + steps: [ + { + title: 'Visit X/Twitter Messages', + description: "Log in to your account if you haven't already", + cta: 'Go to x.com', + action: 'start', + }, + { + title: 'Open the conversation you want to notarize', + description: "Pick a short conversation (to meet current size limits)", + cta: 'Check', + action: 'two', + }, + { + title: 'Notarize conversation', + cta: 'Notarize', + action: 'three', + prover: true, + }, + ], + hostFunctions: ['redirect', 'notarize'], + cookies: ['x.com'], + headers: ['x.com'], + requests: [ + { + url: `https://x.com/i/api/1.1/dm/conversation/*.json`, + method: 'GET', + }, + ], + }), + ); +} + +module.exports = { start, config, two, three }; diff --git a/examples/twitter_profile_js/index.d.ts b/examples/twitter_profile_js/index.d.ts new file mode 100644 index 0000000..3380f1a --- /dev/null +++ b/examples/twitter_profile_js/index.d.ts @@ -0,0 +1,15 @@ +declare module 'main' { + // Extism exports take no params and return an I32 + export function start(): I32; + export function two(): I32; + export function parseTwitterResp(): I32; + export function three(): I32; + export function config(): I32; +} + +declare module 'extism:host' { + interface user { + redirect(ptr: I64): void; + notarize(ptr: I64): I64; + } +} diff --git a/examples/twitter_profile_js/index.js b/examples/twitter_profile_js/index.js new file mode 100644 index 0000000..a700905 --- /dev/null +++ b/examples/twitter_profile_js/index.js @@ -0,0 +1,130 @@ +function isValidHost(urlString) { + const url = new URL(urlString); + return url.hostname === 'twitter.com' || url.hostname === 'x.com'; +} + +function gotoTwitter() { + const { redirect } = Host.getFunctions(); + const mem = Memory.fromString('https://x.com'); + redirect(mem.offset); +} + +function start() { + if (!isValidHost(Config.get('tabUrl'))) { + gotoTwitter(); + Host.outputString(JSON.stringify(false)); + return; + } + Host.outputString(JSON.stringify(true)); +} + +function two() { + const cookies = JSON.parse(Config.get('cookies'))['api.x.com']; + const headers = JSON.parse(Config.get('headers'))['api.x.com']; + if ( + !cookies.auth_token || + !cookies.ct0 || + !headers['x-csrf-token'] || + !headers['authorization'] + ) { + Host.outputString(JSON.stringify(false)); + return; + } + + Host.outputString( + JSON.stringify({ + url: 'https://api.x.com/1.1/account/settings.json', + method: 'GET', + headers: { + 'x-twitter-client-language': 'en', + 'x-csrf-token': headers['x-csrf-token'], + Host: 'api.x.com', + authorization: headers.authorization, + Cookie: `lang=en; auth_token=${cookies.auth_token}; ct0=${cookies.ct0}`, + 'Accept-Encoding': 'identity', + Connection: 'close', + }, + secretHeaders: [ + `x-csrf-token: ${headers['x-csrf-token']}`, + `cookie: lang=en; auth_token=${cookies.auth_token}; ct0=${cookies.ct0}`, + `authorization: ${headers.authorization}`, + ], + }), + ); +} + +function parseTwitterResp() { + const bodyString = Host.inputString(); + const params = JSON.parse(bodyString); + + if (params.screen_name) { + const revealed = `"screen_name":"${params.screen_name}"`; + const selectionStart = bodyString.indexOf(revealed); + const selectionEnd = + selectionStart + revealed.length; + const secretResps = [ + bodyString.substring(0, selectionStart), + bodyString.substring(selectionEnd, bodyString.length), + ]; + Host.outputString(JSON.stringify(secretResps)); + } else { + Host.outputString(JSON.stringify(false)); + } +} + +function three() { + const params = JSON.parse(Host.inputString()); + const { notarize } = Host.getFunctions(); + + if (!params) { + Host.outputString(JSON.stringify(false)); + } else { + const mem = Memory.fromString(JSON.stringify({ + ...params, + getSecretResponse: 'parseTwitterResp', + })); + const idOffset = notarize(mem.offset); + const id = Memory.find(idOffset).readString(); + Host.outputString(JSON.stringify(id)); + } +} + +function config() { + Host.outputString( + JSON.stringify({ + title: 'Twitter Profile', + description: 'Notarize ownership of a twitter profile', + icon: '', + steps: [ + { + title: 'Visit Twitter website', + cta: 'Go to x.com', + action: 'start', + }, + { + title: 'Collect credentials', + description: "Login to your account if you haven't already", + cta: 'Check cookies', + action: 'two', + }, + { + title: 'Notarize twitter profile', + cta: 'Notarize', + action: 'three', + prover: true, + }, + ], + hostFunctions: ['redirect', 'notarize'], + cookies: ['api.x.com'], + headers: ['api.x.com'], + requests: [ + { + url: 'https://api.x.com/1.1/account/settings.json', + method: 'GET', + }, + ], + }), + ); +} + +module.exports = { start, config, two, three, parseTwitterResp }; diff --git a/package.json b/package.json new file mode 100644 index 0000000..0dd8519 --- /dev/null +++ b/package.json @@ -0,0 +1,17 @@ +{ + "name": "twitter_profile", + "version": "1.0.0", + "description": "Demo TLSNotary plugin to notarize the ownership of a twitter profile", + "main": "src/index.ts", + "scripts": { + "build": "node esbuild.js" + }, + "keywords": [], + "author": "TLSNotary", + "license": "MIT", + "devDependencies": { + "@extism/js-pdk": "^1.0.1", + "esbuild": "^0.19.6", + "typescript": "^5.3.2" + } +} \ No newline at end of file diff --git a/src/index.d.ts b/src/index.d.ts new file mode 100644 index 0000000..5f7f0b9 --- /dev/null +++ b/src/index.d.ts @@ -0,0 +1,15 @@ +declare module 'main' { + // Extism exports take no params and return an I32 + export function start(): I32; + export function two(): I32; + export function parseTwitterResp(): I32; + export function three(): I32; + export function config(): I32; +} + +declare module 'extism:host' { + interface user { + redirect(ptr: I64): void; + notarize(ptr: I64): I64; + } +} diff --git a/src/index.ts b/src/index.ts new file mode 100644 index 0000000..3412f0d --- /dev/null +++ b/src/index.ts @@ -0,0 +1,133 @@ +import { icon } from '../dist/assets/icon'; +import config_json from '../config.json'; +/** + * Plugin configuration + * This configurations defines the plugin, most importantly: + * * the different steps + * * the user data (headers, cookies) it will access + * * the web requests it will query (or notarize) + */ +export function config() { + Host.outputString( + JSON.stringify({ + ...config_json, + icon: icon + }), + ); +} + +function isValidHost(urlString: string) { + const url = new URL(urlString); + return url.hostname === 'twitter.com' || url.hostname === 'x.com'; +} + +/** + * Redirect the browser window to x.com + * This uses the `redirect` host function (see index.d.ts) + */ +function gotoTwitter() { + const { redirect } = Host.getFunctions() as any; + const mem = Memory.fromString('https://x.com'); + redirect(mem.offset); +} + +/** + * Implementation of the first (start) plugin step + */ +export function start() { + if (!isValidHost(Config.get('tabUrl'))) { + gotoTwitter(); + Host.outputString(JSON.stringify(false)); + return; + } + Host.outputString(JSON.stringify(true)); +} + +/** + * Implementation of step "two". + * This step collects and validates authentication cookies and headers for 'api.x.com'. + * If all required information, it creates the request object. + * Note that the url needs to be specified in the `config` too, otherwise the request will be refused. + */ +export function two() { + const cookies = JSON.parse(Config.get('cookies'))['api.x.com']; + const headers = JSON.parse(Config.get('headers'))['api.x.com']; + if ( + !cookies.auth_token || + !cookies.ct0 || + !headers['x-csrf-token'] || + !headers['authorization'] + ) { + Host.outputString(JSON.stringify(false)); + return; + } + + Host.outputString( + JSON.stringify({ + url: 'https://api.x.com/1.1/account/settings.json', + method: 'GET', + headers: { + 'x-twitter-client-language': 'en', + 'x-csrf-token': headers['x-csrf-token'], + Host: 'api.x.com', + authorization: headers.authorization, + Cookie: `lang=en; auth_token=${cookies.auth_token}; ct0=${cookies.ct0}`, + 'Accept-Encoding': 'identity', + Connection: 'close', + }, + secretHeaders: [ + `x-csrf-token: ${headers['x-csrf-token']}`, + `cookie: lang=en; auth_token=${cookies.auth_token}; ct0=${cookies.ct0}`, + `authorization: ${headers.authorization}`, + ], + }), + ); +} + +/** + * This method is used to parse the Twitter response and specify what information is revealed (i.e. **not** redacted) + * This method is optional in the notarization request. When it is not specified nothing is redacted. + * + * In this example it locates the `screen_name` and excludes that range from the revealed response. + */ +export function parseTwitterResp() { + const bodyString = Host.inputString(); + const params = JSON.parse(bodyString); + + // console.log("params"); + // console.log(JSON.stringify(params)); + + if (params.screen_name) { + const revealed = `"screen_name":"${params.screen_name}"`; + const selectionStart = bodyString.indexOf(revealed); + const selectionEnd = + selectionStart + revealed.length; + const secretResps = [ + bodyString.substring(0, selectionStart), + bodyString.substring(selectionEnd, bodyString.length), + ]; + Host.outputString(JSON.stringify(secretResps)); + } else { + Host.outputString(JSON.stringify(false)); + } +} + +/** + * Step 3: calls the `notarize` host function + */ +export function three() { + const params = JSON.parse(Host.inputString()); + const { notarize } = Host.getFunctions() as any; + + if (!params) { + Host.outputString(JSON.stringify(false)); + } else { + const mem = Memory.fromString(JSON.stringify({ + ...params, + getSecretResponse: 'parseTwitterResp', + })); + const idOffset = notarize(mem.offset); + const id = Memory.find(idOffset).readString(); + Host.outputString(JSON.stringify(id)); + } +} diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 0000000..a59c4c1 --- /dev/null +++ b/tsconfig.json @@ -0,0 +1,14 @@ +{ + "compilerOptions": { + "lib": [], + "types": [ + "@extism/js-pdk" + ], + "noEmit": true, + "resolveJsonModule": true, + "esModuleInterop": true, + }, + "include": [ + "src/**/*" + ] +} \ No newline at end of file