Skip to content

This repo reads notes from a notebook, paper, or whiteboard and converts it into Markdown (*.md) using GenAI

License

Notifications You must be signed in to change notification settings

raffertyuy/Handwriting-to-Markdown-GPT

Repository files navigation

Handwriting-to-Markdown-GPT

This repo reads notes from a notebook, paper, or whiteboard and converts it into Markdown (*.md) using GenAI.

This code is the next iteration of a previous Handwriting-to-Markdown Azure Logic App which was doing something similar but using Azure AI Services - Computer Vision. In this implementation, Azure OpenAI gpt-4o is used instead.

  • ./az-function contains the main code using python and deployed to Azure Functions
  • ./az-logicapp contains code that triggers when a photo is added in a OneDrive source folder, and creates an .md file in a OneDrive target folder

Also see this blog post

Solution Flow

This solution is using Azure Logic Apps with the following flow:

  1. Watch for new files in a specific OneDrive folder
  2. Check for GPT-4o supported image formats
    • if the filetype is PDF, convert to JPG using OneDrive - Convert File (limited to the first page of the PDF file).
    • if the filetype is not supported, skip.
  3. Extract the text from the image file using GPT-4o (through a Function App).
  4. Copy the image and save a new markdown file output in a destination OneDrive folder.
flowchart LR
    H[Handwritten Notes/Image<br />from OneDrive] --> |image content| N{Detect Note Type}
    subgraph "Azure Function"
        subgraph "Azure OpenAI GPT-4o Prompts"
            N --> |Paper| P[Paper System Prompt]
            N --> |Whiteboard| W[Whiteboard System Prompt]
            N --> |Other| O[Generic System Prompt]
            P --> PR[Proof Read]
            W --> |text output| PR
            O --> PR
            PR --> S[Section<br />Headers]
            S --> ET[Extract<br />Title]
        end
         ET --> IT[Initial<br />Output]
         IT --> |add<br />metadata| Output[Final<br />Output]
    end
    Output --> |save| MD[Markdown File<br />in OneDrive]
Loading

Note

Prompt Flow was previously attempted, but the following gaps were encountered (as of 2024-06-28):

  • Prompt Flow in VS Code doesn't support an image input type.
  • Prompt Flow on AI Studio doesn't support GPT-4o (but GPT-4V is available in preview)

Instructions for Running Locally and for Deployment

There is no one-click deployment for this solution yet. If you want to try, feel free to Contribute. For now, here are the manual instructions.

Pre-requisites

  1. An active Azure Subscription
  2. An Azure OpenAI resource
  3. An Azure Key Vault resource

Running Locally

This app requires Microsoft Azure and can't be run locally. However, you may do the following to try and test the output:

  1. Run the Jupyter notebook
  • Create a .env file (refer to .env.example)
  • Modify the image path in this cell IMAGE_PATH="../media/notes-sample.png"
  • Run the notebook
  1. Run the Azure Function locally
  • Install the Azure Functions Core Tools
  • Create a file local.settings.json (refer to local.settings.example.json)
  • Go to the local CLI and run func start

Note

The current code assumes that you have an Azure Identity with the necessary access to your Azure OpenAI resource. To do so:

  1. Go to the Azure Portal
  2. Go to your Azure OpenAI resource
  3. Go to the Access control (IAM) tab
  4. Add a role assignment, and add yourself with the role Cognitive Services OpenAI User (or Contributor).

Azure Deployment

  1. Deploy the Azure Function
  • Opening the source in VS Code
  • Installing the Azure Functions extension
  • In the Explorer pane, right-click the az-function folder, and then click Deploy to Function App...
  • Test the function using the Test & View Output button
  1. Add Azure Function key in the Key Vault
  • Go to the deployed Azure Function
  • Go to the App keys tab and copy the default function key
  • Go to your Azure Key Vault resource (one of the pre-requisites)
  • Go to Secrets and add the function key as a secret
  1. Deploy the Azure Logic App
  • Going to the Azure Portal
  • In the search bar, search for "Template Deployment"
  • Load the ./az-logicapp/azuredeploy.json file
  • Fill in the required parameters
  • Deploy
  1. Configure Azure Logic App Idenity
  • Go to the deployed Azure Logic App
  • Click the Identity tab
  • Click "Identity""System assigned", and then turn Status = On
  • Add a role assignment, and add the Logic App with the role Key Vault Secrets User
  1. Check Azure Logic App connections
  • Check and fix the connection to your personal OneDrive account
  • Check and fix the connection to your Azure Key Vault
  • After deployment, go to the Logic App
  • Click "Identity""System assigned", and then turn Status = On

Tip

  • If you need to troubleshoot, check code.json. This is the copy-pasted code from the Logic App designer, included for reference.
  • If you're wondering why I didn't use managed identities to authorize the Logic App to the Azure Function, see Logic App Notes below.

Azure Function Notes

Python Programming Model

This Azure function is developed using the Python v2 programming model and is deployed using the VS Code Extension: Azure Functions.

Tip

Sometimes, the function does not show up on Azure after deploying. This usually happens when there is an error in the code. For example:

  • a missing dependency in the requirements.txt file
  • a missing environment variable in Azure (that is currently in local.settings.json)
  • or an actual error in the code.

Note

There is a previous attempt to have this consumed by logic apps using the Azure Function connector and Managed Identiites. It didn't work due to complexity of using MI with multipart/form-data. But here are additional notes in this previous attempt:

  • To call Azure Functions using managed identities, the function method must be set to ANONYMOUS and configured for managed identity authentication
  • Azure Logic Apps can't call Azure Functions with custom routes. For Python, this means leaving the route empty with @app.route(route="") as suggested in this issue.

For the current implementation, this function's AuthLevel is set to FUNCTION. Meaning, a function key is expected to be passed in the request.

Logic App Notes

Authentication

To authenticate with the Azure Function, a function key stored in Azure Key Vault is used.

Sending a multipart/form-data HTTP POST

Since we're passing a OneDrive file to Azure Functions as a multipart/form-data, the HTTP request body should look like this.

{
    "$content-type": "multipart/form-data",
    "$multipart": [
        {
            "body": ATTACHMENTS_CONTENT,
            "headers": {
                "Content-Disposition": "form-data; name=image; filename=\"ATTACHMENTS_NAME\"",
                "Content-Type": "ATTACHMENTS_CONTENT_TYPE"
            }
        }
    ]
}

Note

I eventually used the HTTP Request connector instead of an Azure Functions connector. This is because the response body is in JSON, and the Azure Functions connector returns this in a JSON-escaped string format. Since I'm using Obsidian for my second brain, my final markdown image link uses ![[image_path]] instead of the standard ![name](image_path) format.

About

This repo reads notes from a notebook, paper, or whiteboard and converts it into Markdown (*.md) using GenAI

Resources

License

Code of conduct

Security policy

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published