Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

adding python webhooks app example #294

Merged
merged 11 commits into from
Sep 4, 2024
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
24 changes: 24 additions & 0 deletions examples/python-webhooks/.gitignore
Original file line number Diff line number Diff line change
@@ -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
3 changes: 3 additions & 0 deletions examples/python-webhooks/.sample.env
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
MIRO_CLIENT_ID="<your-client-id>"
MIRO_CLIENT_SECRET="<your-client-secret>"
MIRO_REDIRECT_URL="<your-ngrok-url>"
109 changes: 109 additions & 0 deletions examples/python-webhooks/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,109 @@
# Python Webhooks

This app shows how to get webhook events on your Miro board using Python and Flask.
horeaporutiu marked this conversation as resolved.
Show resolved Hide resolved

# 👨🏻‍💻 App Demo

https://github.com/user-attachments/assets/0ccffb46-daab-4fc9-8ff8-f5720237f75a

# 📒 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 <a name="features"></a>

- [Miro Node Client Library with Python](https://miroapp.github.io/api-clients/python/miro_api.html)
- [miro.exchangeCodeForAccessToken()](https://miroapp.github.io/api-clients/python/miro_api.html#Miro.exchange_code_for_access_token)
- [miro.isAuthorized()](https://miroapp.github.io/api-clients/python/miro_api.html#Miro.is_authorized)
- [miro.getAuthUrl()](https://miroapp.github.io/api-clients/python/miro_api.html#Miro.get_auth_url)
- [api.get_all_boards()](https://miroapp.github.io/api-clients/python/miro_api/api_extended.html#MiroApiExtended.get_all_boards)

# 🛠️ Tools and Technologies <a name="tools"></a>

- [Python](https://www.python.org/)
- [Flask](https://flask.palletsprojects.com/en/3.0.x/)

# ✅ Prerequisites <a name="prerequisites"></a>

- 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 [Python](https://www.python.org/) 3.9 or a later version.
- Your development environment includes [pip](https://www.python.org/) 24.0 or a later version.
- Your development environment includes [ngrok](https://ngrok.com/) or something similar.

# 📖 Associated Developer Tutorial <a name="tutorial"></a>

> To view a more in depth developer tutorial
> of this app (including code explanations) see the [Getting started with Miro webhooks using Python tutorial](https://developers.miro.com/docs/getting-started-with-webhooks-python) on Miro's Developer documentation.
horeaporutiu marked this conversation as resolved.
Show resolved Hide resolved

# 🏃🏽‍♂️ Run the app locally <a name="run"></a>

0. It is recommended to use a virtual env to run this app example. Go to where you .venv is and then activate it with the
`source ./bin/activate` command. Read more about venvs [here](https://docs.python.org/3/library/venv.html).

1. Create a new Miro app on [developers.miro.com](https://developers.miro.com/). This will take you to the app settings page. There you
will find the `MIRO_CLIENT_ID` and `MIRO_CLIENT_SECRET` to be added to your `.env` file. Ensure that your app URL is `http://localhost:5000` since that is what port Flask will be running on. Ensure that `boards:read` scope is checked,
and then go ahead and install the app on your developer team. You will get an **access token** which you will need later to
authenticate the creation of your webhook subscription.

2. In a new terminal session, run:

```
ngrok http 5000
```

This will output something like this:

```
Forwarding https:<your-ngrok-url> -> http://localhost:5000
```

The `https:<your-ngrok-url>` 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 `pip install -r requirements.txt` to install dependencies.

5. In a separate terminal from the ngrok terminal (leave ngrok running) go to `app-examples/examples/python-webhooks` directiory. Run `flask --app app run` to start the server. Your server should be running on port 5000.

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/<boardId>.
>
> **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 🎉.
horeaporutiu marked this conversation as resolved.
Show resolved Hide resolved

# 🗂️ Folder structure <a name="folder"></a>

```
.
├── app.py - main logic to receive webhooks and start the server
├── .sample.env <-- File with sample env variables. Need to rename to .env and then add in your variables.
├── app-manifest.yaml <-- File with sample manifest file for easy copy paste into your developer app settings manifest.
├── requirements.txt <-- File with libraries which the project depends on, including versions.
```

# 🫱🏻‍🫲🏽 Contributing <a name="contributing"></a>

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 <a name="license"></a>

[MIT License](https://github.com/miroapp/app-examples/blob/main/LICENSE).
5 changes: 5 additions & 0 deletions examples/python-webhooks/app-manifest.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
# See https://developers.miro.com/docs/app-manifest on how to use this
appName: Python Webhooks
sdkUri: "http://localhost:5000"
scopes:
- boards:read
53 changes: 53 additions & 0 deletions examples/python-webhooks/app.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
from flask import Flask, session, request, json, Response
from miro_api import Miro
from miro_api.storage import Storage

from dotenv import load_dotenv

load_dotenv()

app = Flask(__name__)

app.secret_key = b"very_random_secret"


class SessionStorage(Storage):
session_key = "miro_state"

def get(self):
return session[self.session_key]

def set(self, state):
if not state:
session.pop(self.session_key, None)
return
session[self.session_key] = state


miro = Miro(storage=SessionStorage())


def render_boards():
boards = miro.api.get_all_boards()
names = "<br/>".join([board.name for board in boards])
return f"<p>List of boards in the team: {names}</p>"


@app.route("/", methods=["GET", "POST"])
def hello_world():
if request.method == "GET":
if miro.is_authorized:
return render_boards()

if code := request.args.get("code", ""):
miro.exchange_code_for_access_token(code)
return render_boards()

return f"<a href='{miro.auth_url}'>Login to Miro</a>"
if request.method == "POST":
webhook_data = request.json
formatted_webhook_data = json.dumps(webhook_data, indent=4)
print(f"webhook event: {formatted_webhook_data}")
return Response(
json.dumps(webhook_data), status=200, mimetype="application/json"
)
16 changes: 16 additions & 0 deletions examples/python-webhooks/requirements.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
annotated-types==0.7.0
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

i think most of these are not required here except for Flask, miro_api and missing dotenv package

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

good point! I'll re work this file.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@janza should be ready now.

blinker==1.8.2
click==8.1.7
Flask==3.0.3
itsdangerous==2.2.0
Jinja2==3.1.4
MarkupSafe==2.1.5
miro_api==2.2.0
pydantic==2.8.2
pydantic_core==2.20.1
python-dateutil==2.9.0.post0
python-dotenv==1.0.1
six==1.16.0
typing_extensions==4.12.2
urllib3==1.26.19
Werkzeug==3.0.3
Loading