☁️️ Quickstart: Workflows - Run code (with Airtable example)

Starter examples for using Run Code in workflows

With Run Code you can run your own JavaScript code within a Harbour workflow
(TLDR- jump to the example code)



This can be quite useful for:

  • Pulling data in from a third-party systems (e.g., get a list of available cost centers from Airtable for dynamic approvals routing)
  • Running your own custom calculations (e.g., determine pricing tier based on inputs)
  • Sending data into a third-party system (e.g., your in-house CMS or project tracker)

Starter code samples:

//Hello world - really basic example
function main(){
return "Hello world~!";
}
//Example Airtable integration (🔗 demo Airtable base)

//Get your Airtable API token: https://airtable.com/create/tokens (***can be restricted to be read-only from a single base)
const AIRTABLE_TOKEN = "<INSERT>";

//Get your base/table IDs from URL: https://airtable.com/<BASE_ID>/<TABLE_ID>/viw7yUZ6mSRuM3t3h?blocks=hide
const BASE_ID = "apppziZc3tEldBHHn"; //BASE_ID
const TABLE_ID = "tblTJO2NfSPnSezom"; //TABLE_ID

//Set lookup key and target (aka return field)
//Reference visual: https://cdn.zappy.app/f5f255354ebc8f9c1b04f32e0d0bc157.png
const lookupField = "Role";
const lookupValue = "CFO";
const targetField = "Email";

//Get current, matching Airtable value (CFO email)
const main = async () => {
    try {
        const API_URL = `https://api.airtable.com/v0/${BASE_ID}/${TABLE_ID}`;
        const response = await fetch(API_URL, {
            method: "GET",
            headers: {
                "Authorization": `Bearer ${AIRTABLE_TOKEN}`,
                "Content-Type": "application/json"
            }
        });
        const data = await response.json();
        const match = data.records.filter(r => r.fields[lookupField] == lookupValue)[0];
        return match.fields[targetField];
    } catch(error) {
        return "#ERROR"
    }
}
//Hello world - basic example, with an upstream input value
function main(inputs){
return "Hello " + inputs.signer1_name + "~!";
}
//Email extraction example - returning multiple values, useable in downstream actions
function main(inputs){
const email = inputs.signer1_email; //e.g., "ada@lovelace.com"
const username_value = email.split("@")[0];
const domain_name_value = email.split("@")[1];
return {"username":username_value, "domain_name":domain_name_value};
}
//Get data example - async function with third-party data
async function main(inputs){
const r = await fetch("https://api.github.com/repositories/5906854/pulls/37/comments");
const json = await r.json();
return "Hello " + json[0].user.login + "~!";
}
//Import module example (Lodash) - async function with imported code and emoji
import * as _ from "https://unpkg.com/lodash-es@4.17.21/lodash.js";
async function main(inputs){
return _.startCase("🦄 hello universe~!");
}


The returned value (from above examples) is then available here (Inputs > 💻️ Result) and can be used within downstream emails, conditional step routing, approvals, and much more):

✅️ Approvals (with Run Code-based approver):

✉️️ Email (with Run Code-based personalization):


Technical notes

  • Full support for modern JavaScript features (ES 2024+) ✅️
  • Each run can take up to 1 minute and up to 500MB of memory 👌️🏎️️
  • We support both sync and async functions, third-party data requests, and third-party module importing (*see examples below) 🕸️️
  • Your JavaScript can both access available inputs from upstream action blocks (e.g., signer e-mail from an upstream Start agreement) and send results into downstream action blocks 🏈️
  • JavaScript errors are captured and sent to anyone in the Notifications section of the Workflow Settings ⚾🧤