From 5d2441a57e8272659d26258deb775d992a2a974b Mon Sep 17 00:00:00 2001 From: Horea Porutiu Date: Wed, 4 Sep 2024 17:22:51 +0200 Subject: [PATCH] adding Node.js + Express.js webhooks app example (#293) adding Node.js + Express.js webhooks app example --- examples/node-webhooks/.gitignore | 24 +++++ examples/node-webhooks/.sample.env | 3 + examples/node-webhooks/README.md | 103 +++++++++++++++++++ examples/node-webhooks/app-manifest.yaml | 5 + examples/node-webhooks/jsconfig.json | 7 ++ examples/node-webhooks/package.json | 21 ++++ examples/node-webhooks/src/app.js | 64 ++++++++++++ examples/node-webhooks/src/miroMiddleware.js | 22 ++++ examples/node-webhooks/tsconfig.json | 21 ++++ examples/node-webhooks/vite.config.js | 18 ++++ 10 files changed, 288 insertions(+) create mode 100644 examples/node-webhooks/.gitignore create mode 100644 examples/node-webhooks/.sample.env create mode 100644 examples/node-webhooks/README.md create mode 100644 examples/node-webhooks/app-manifest.yaml create mode 100644 examples/node-webhooks/jsconfig.json create mode 100644 examples/node-webhooks/package.json create mode 100644 examples/node-webhooks/src/app.js create mode 100644 examples/node-webhooks/src/miroMiddleware.js create mode 100644 examples/node-webhooks/tsconfig.json create mode 100644 examples/node-webhooks/vite.config.js diff --git a/examples/node-webhooks/.gitignore b/examples/node-webhooks/.gitignore new file mode 100644 index 000000000..22367671f --- /dev/null +++ b/examples/node-webhooks/.gitignore @@ -0,0 +1,24 @@ +# See https://help.github.com/articles/ignoring-files/ for more about ignoring files. + +# dependencies +/node_modules +/.pnp +.pnp.js +.next + +# testing +/coverage + +# misc +.DS_Store +*.pem +.idea + +# debug +npm-debug.log* +yarn-debug.log* +yarn-error.log* + +# local env files +.env +dist \ No newline at end of file diff --git a/examples/node-webhooks/.sample.env b/examples/node-webhooks/.sample.env new file mode 100644 index 000000000..d099aa952 --- /dev/null +++ b/examples/node-webhooks/.sample.env @@ -0,0 +1,3 @@ +MIRO_CLIENT_ID="" +MIRO_CLIENT_SECRET="" +MIRO_REDIRECT_URL="" \ No newline at end of file diff --git a/examples/node-webhooks/README.md b/examples/node-webhooks/README.md new file mode 100644 index 000000000..1eb75f28a --- /dev/null +++ b/examples/node-webhooks/README.md @@ -0,0 +1,103 @@ +# Node Webhooks + +This app demonstrates how to receive webhook events from your Miro board using Node.js and Express.js. By following this guide, you will set up a local environment, create a webhook subscription, and test receiving events when changes are made on your Miro board. + +# 👨🏻‍💻 App Demo + +https://github.com/user-attachments/assets/1448b658-9e6f-4652-8300-6cebbf081f7a + +# 📒 Table of Contents + +- [Included Features](#features) +- [Tools and Technologies](#tools) +- [Prerequisites](#prerequisites) +- [Associated Developer Tutorial](#tutorial) +- [Run the app locally](#run) +- [Folder Structure](#folder) +- [Contributing](#contributing) +- [License](#license) + +# ⚙️ Included Features + +- [Miro Node Client Library with Express SDK](https://miroapp.github.io/api-clients/node/index.html) + - [miro.exchangeCodeForAccessToken()](https://miroapp.github.io/api-clients/node/classes/index.Miro.html#exchangeCodeForAccessToken) + - [miro.isAuthorized()](https://miroapp.github.io/api-clients/node/classes/index.Miro.html#isAuthorized) + - [miro.getAuthUrl()](https://miroapp.github.io/api-clients/node/classes/index.Miro.html#getAuthUrl) + - [miro.as()](https://miroapp.github.io/api-clients/node/classes/index.Miro.html#as) + - [api.getAllBoards()](https://miroapp.github.io/api-clients/node/classes/index.MiroApi.html#getAllBoards) + +# 🛠️ Tools and Technologies + +- [Node.js](https://nodejs.org/en) +- [Express.js](https://expressjs.com/) + +# ✅ Prerequisites + +- You have a [Miro account](https://miro.com/signup/). +- You're [signed in to Miro](https://miro.com/login/). +- Your Miro account has a [Developer team](https://developers.miro.com/docs/create-a-developer-team). +- Your development environment includes [Node.js 14.13](https://nodejs.org/en/download) or a later version, and npm. +- Your development environment includes [ngrok](https://ngrok.com/) or something similar. + +# 📖 Associated developer tutorial + +> To view a more in depth developer tutorial of this app including code explanations, see [Getting started with webhooks](https://developers.miro.com/docs/getting-started-with-webhooks) on Miro's developer portal. + +# 🏃🏽‍♂️ Run the app locally + +1. **Create a Miro app** on [developers.miro.com](https://developers.miro.com/). This will take you to the app settings page, where you will find the `MIRO_CLIENT_ID` and `MIRO_CLIENT_SECRET`. These need to be added to your `.env` file. + + - Ensure the `boards:read` scope is checked. + - Install the app on your developer team. You will get an **access token**, which is required later to authenticate your webhook subscription. + +2. In a new terminal window, run: + +``` +ngrok http 3000 +``` + +This will output something like this: + +``` +Forwarding https: -> http://localhost:3000 +``` + +The `https:` is your `MIRO_REDIRECT_URL` to be used in the `.env` file and then later when calling the API to create a webhook subscription. + +3. Rename the `.sample.env` file to `.env` and then add in your `MIRO_CLIENT_ID` and `MIRO_CLIENT_SECRET` from your [developers.miro.com](https://developers.miro.com/) app settings page. Use the `forwarding URL` from the previous step for the `MIRO_REDIRECT_URL` in the .env file. Save the file as `.env` with your new variables. + +4. Run `npm i` to install dependencies. + +5. Run `npm start` to start the dev server. + +6. Go to your developer team, and open the board you want to receive webhook events for. + +7. In a separate browser tab, open up the API Exporer for the [Create Webhook Subscription endpoint](https://developers.miro.com/reference/create-board-subscription). + +8. Provide the following information in the API Explorer: + +> **Access Token**: Once you get the access token after installing your app on a developer team (from step 4 above), you can add the access token to the Authorization section of the API reference page. +> +> **boardId:** Get the board ID of the board you want to receive notifications for. This board should be in the same team where you installed the app. You can find board ID in the URL when you go to your board: https://miro.com/app/board/. +> +> **callbackUrl:** This is the URL where you will receive events. It should be the same as `MIRO_REDIRECT_URL` in `.env`. 9. Select Try It! to run the API request right from the browser. If you get a 201 response, you are ready to receive events! + +10. Next, go to to the same board which you referenced in the request above, and create a sticky. You should now receive a webhook event! Great job! You've just learned how to get started with Miro's webhooks with Python 🎉. + +# 🗂️ Folder structure + +``` +. +├── src +│ └── app.js - main logic to receive webhooks and start the server +│ └── miroMiddleware.css <-- Middleware file to setup OAuth +├── .sample.env <-- File with sample env variables. Need to rename to .env and then add in your variables. +``` + +# 🫱🏻‍🫲🏽 Contributing + +If you want to contribute to this example, or any other Miro Open Source project, please review [Miro's contributing guide](https://github.com/miroapp/app-examples/blob/main/CONTRIBUTING.md). + +# 🪪 License + +[MIT License](https://github.com/miroapp/app-examples/blob/main/LICENSE). diff --git a/examples/node-webhooks/app-manifest.yaml b/examples/node-webhooks/app-manifest.yaml new file mode 100644 index 000000000..c7b0c121a --- /dev/null +++ b/examples/node-webhooks/app-manifest.yaml @@ -0,0 +1,5 @@ +# See https://developers.miro.com/docs/app-manifest on how to use this +appName: Node Webhooks +sdkUri: "http://localhost:3000" +scopes: + - boards:read diff --git a/examples/node-webhooks/jsconfig.json b/examples/node-webhooks/jsconfig.json new file mode 100644 index 000000000..d23878618 --- /dev/null +++ b/examples/node-webhooks/jsconfig.json @@ -0,0 +1,7 @@ +{ + "compilerOptions": { + "typeRoots": ["./node_modules/@types", "./node_modules/@mirohq"] + }, + "include": ["src"], + "exclude": ["node_modules"] +} diff --git a/examples/node-webhooks/package.json b/examples/node-webhooks/package.json new file mode 100644 index 000000000..0201106d1 --- /dev/null +++ b/examples/node-webhooks/package.json @@ -0,0 +1,21 @@ +{ + "name": "node-webhooks", + "version": "0.1.0", + "license": "MIT", + "scripts": { + "start": "vite", + "build": "vite build", + "serve": "vite preview" + }, + "dependencies": { + "express": "^4.18.1", + "@mirohq/miro-api": "^2.0.0", + "cookie-parser": "^1.4.6", + "dotenv": "^16.0.3" + }, + "devDependencies": { + "vite": "3.0.3", + "vite-plugin-node": "^2.0.0" + }, + "type": "module" +} \ No newline at end of file diff --git a/examples/node-webhooks/src/app.js b/examples/node-webhooks/src/app.js new file mode 100644 index 000000000..8d40cfa68 --- /dev/null +++ b/examples/node-webhooks/src/app.js @@ -0,0 +1,64 @@ +import { config } from "dotenv"; + +import express from "express"; +import cookieParser from "cookie-parser"; + +import miroMiddleware from "./miroMiddleware"; + +config(); + +const app = express(); + +app.use(cookieParser("")); +app.use(miroMiddleware); +app.use(express.json()); + +app.get("/auth/miro/callback", async (req, res) => { + if (typeof req.query.code !== "string") { + res.status(400); + res.send("Missing code query parameter!"); + return; + } + await req.miro.exchangeCodeForAccessToken(req.cookies.id, req.query.code); + res.redirect("/"); +}); + +app.get("/", async (req, res) => { + if (!(await req.miro.isAuthorized(req.cookies.id))) { + res.redirect(req.miro.getAuthUrl()); + return; + } + + const api = req.miro.as(req.cookies.id); + + res.header("content-type", "text/html"); + res.write("These are the boards that you have access to:
"); + + const allBoards = api.getAllBoards(); + for await (const board of allBoards) { + res.write(`${board.name}
`); + } + res.send(); +}); + +app.post("/", async (req, res) => { + if (req.body.event) { + console.log("Webhook event:"); + console.log(req.body.event); + } + + if (req.body.challenge) { + console.log("Challenge:", req.body.challenge); + res.send(req.body); + return; + } + res.send("OK"); +}); + +if (import.meta.env.PROD) { + app.listen(3000, () => + console.log("Started server on http://127.0.0.1:3000"), + ); +} + +export const viteNodeApp = app; diff --git a/examples/node-webhooks/src/miroMiddleware.js b/examples/node-webhooks/src/miroMiddleware.js new file mode 100644 index 000000000..958f90246 --- /dev/null +++ b/examples/node-webhooks/src/miroMiddleware.js @@ -0,0 +1,22 @@ +import { Miro } from "@mirohq/miro-api"; + +export default function middlware(req, res, next) { + req.miro = new Miro({ + storage: { + // eslint-disable-next-line no-unused-vars + async get(_userId) { + try { + return JSON.parse(req.cookies.state); + } catch (err) { + return undefined; + } + }, + + set(userId, state) { + res.cookie("id", userId, { path: "/", secure: true }); + res.cookie("state", JSON.stringify(state), { path: "/", secure: true }); + }, + }, + }); + next(); +} diff --git a/examples/node-webhooks/tsconfig.json b/examples/node-webhooks/tsconfig.json new file mode 100644 index 000000000..afecbf88b --- /dev/null +++ b/examples/node-webhooks/tsconfig.json @@ -0,0 +1,21 @@ +{ + "compilerOptions": { + "target": "ESNext", + "useDefineForClassFields": true, + "lib": ["DOM", "DOM.Iterable", "ESNext"], + "allowJs": false, + "skipLibCheck": false, + "esModuleInterop": false, + "allowSyntheticDefaultImports": true, + "strict": true, + "forceConsistentCasingInFileNames": true, + "module": "ESNext", + "moduleResolution": "Node", + "resolveJsonModule": true, + "isolatedModules": true, + "noEmit": true, + "jsx": "react-jsx", + "typeRoots": ["./node_modules/@types", "./node_modules/@mirohq"] + }, + "include": ["./src", "node_modules"] +} diff --git a/examples/node-webhooks/vite.config.js b/examples/node-webhooks/vite.config.js new file mode 100644 index 000000000..eb11f4b1b --- /dev/null +++ b/examples/node-webhooks/vite.config.js @@ -0,0 +1,18 @@ +import { defineConfig } from "vite"; +import { VitePluginNode } from "vite-plugin-node"; + +export default defineConfig({ + server: { + port: 3000, + }, + plugins: [ + ...VitePluginNode({ + adapter: "express", + + appPath: "./src/app.js", + + exportName: "viteNodeApp", + }), + ], + optimizeDeps: {}, +});