Accessing a formData object in a Cloudflare Workers function

I’m trying to send a formData object to a cloudlflare workers object. Here’s the original fetch event

const response = await fetch('https://example.workers.dev/createproject', {
    method: 'POST',
    body: formData,
    mode: 'no-cors'
    if (response.ok) {
    ...
    }
    else {
    ...
    }

}

And here is my cloudflare workers index.js file

import { json, missing, ThrowableRouter, withParams } from 'itty-router-extras';
import { createClient } from '@supabase/supabase-js';
const router = ThrowableRouter();
// apiKey is the parsed JWT for a user.
// Not currently supported by Supabase's client, but eventually
// you'll be able to auth _as_ a user by providing this value.
const supabase = (apiKey) => createClient(SUPABASE_URL, SUPABASE_API_KEY);
const parseAuthHeader = (header) => {
    if (!header)
        return;
    const [_, token] = header.split("Bearer ");
    return token;
};

  router.post('/createproject', async ({ request }) => {
    try {
      const formData = await request.formData();
      console.log('success');
      console.log(formData);
    }
    catch (error) {
      console.log('error');
      console.log(error)
    }
    console.log(headers);
    console.log(request);
    return new Response('test');
});
  
  /*
  This is the last route we define, it will match anything that hasn't hit a route we've defined
  above, therefore it's useful as a 404 (and avoids us hitting worker exceptions, so make sure to include it!).
  
  Visit any page that doesn't exist (e.g. /foobar) to see it in action.
  */
  
router.all("*", () => new Response("404, not found!", { status: 404 }))

addEventListener("fetch", event => {
    event.respondWith(router.handle(event.request));
});

In the createproject route, everything that’s logged comes back empty (e.g. empty objects, empty arrays) — so it’s clearly not accessing the request body or formData object as expected, or I’m not sending the request / formData correctly from the original fetch call.

Can someone help me fix this code so that I can access the request body and formData inside the createproject route?

See the thread for this user’s formdata parser.

Hey @Judge, thank you so much for following-up on this.

There are no files like images in the formData I’m trying to send. Just plain form data and some json. Would I still need to use that parser?

If so, could you perhaps point to the specific aspects of the parser which are relevant for my use case?

From this code example, I’m not exactly sure which parts are the example data being sent, and which parts are the parser

const main = async () => {
  const o = {};
  const o2 = {};
  const body = new FormData();
  o['50'] = '';
  o['100'] = '';
  o['test.txt'] = 'data:text/plain;base64,YWFh'; //contains : 'aaa'
  o['test.json'] = 'data:application/json;base64,eyJKb2huIjoiRG9lIn0='; //contains : '{"John":"Doe"}'
  for (const k in o) { body.append(k, await (await fetch(o[k])).blob(), k); }
  o2['1'] = JSON.stringify({ name: 'Alice' });
  o2['2'] = JSON.stringify({ name: 'Bob' });
  o2['x'] = 'yyy';
  o2['u'] = "v\nv2";
  for (const k in o2) { body.append(k, o2[k]); }
  const res = new Response(body);
  //console.log(JSON.stringify(Object.fromEntries(res.headers)));
  //console.log(await res.clone().text());
  const event = { request: res, response: res }; // Fake event, fake request
  console.time('formData');
  const fd = await formData(event); // Returns an array [{name,value,filename,content-type}]
  console.timeEnd('formData');
  console.log(fd);
  if (!fd) { return; }
  const z = fd[1];
  const blob = new Blob([new Uint8Array(z.value, 0, z.value.byteLength)]);
  //const blob = new Blob(z.value);
  const url = URL.createObjectURL(blob);
  document.querySelector('body').insertAdjacentHTML('beforeend', `<img src='${url}' width=50>`);
  //URL.revokeObjectURL(url);
}
main();

const formData = async (event) => {
  // content-type headers must be valid and offer a boundary
  const boundary = await getBoundary(event);
  if (!boundary) { return null; }
  // Payload size must be under 1000kB
  const ab = await event.request.arrayBuffer();
  const max = ab.byteLength;
  if (!await lessThan1000kB(max)) { return null; }
  // Start to parse
  const t = new TextDecoder();
  const fd = Array();
  let o = Object();
  let i = 0;
  //console.log({ ab, i, max });
  while (i < max) {
    i = await getPart(ab, i, max, boundary, t, o);
    if (!o || !o.name) { break; }
    fd.push(o);
    o = Object();
  }
  return fd;
}

const lessThan1000kB = async (max) => {
  if (Math.ceil(max / 1000) > 1000) { return false; }
  return true;
}

const getBoundary = async (event) => {
  const ct = event.request.headers.get('content-type');
  return `--${(ct.match(/^multipart\/form\-data; boundary=(.+)$/))?.[1]}`;
}

const getPart = async (ab, start, max, boundary, t, o) => {
  let s = '';
  let i = start;
  //console.log('A');
  i = await readString(boundary, i, ab, t, max);
  if (i === max) { return max; }
  //console.log('B');
  if (await eof(i, ab, t, max)) { return max; }
  if (i === max) { return max; }
  //console.log('C');
  i = await readNewLine(i, ab, t, max);
  if (i === max) { return max; }
  //console.log('D');
  i = await readString('Content-Disposition: form-data; name="', i, ab, t, max);
  if (i === max) { return max; }
  //console.log('E');
  i = await captureUntil(['"'], i, ab, t, max, o, 'name');
  if (i === max) { return max; }
  //console.log('F');
  if (!o.name) { o = null; return max; }
  if (i === max) { return max; }
  //console.log('G');
  let x;
  if ((x = await readValue("\"\n\n", boundary, i, ab, t, max, o))) { return x; }
  if ((x = await readValue("\"\r\n\r\n", boundary, i, ab, t, max, o))) { return x; }
  return readFile(boundary, i, ab, t, max, o);
}

// Depending on carriage return prefix,
// Try to read a simple value (implicit text/plain)
const readValue = async (s, boundary, i, ab, t, max, o) => {
  if (!await checkString(s, i, ab, t, max)) { return false; }
  //console.log('I');
  i = await readString(s, i, ab, t, max);
  //console.log('I2');
  i = await captureUntil([/*`${"\r\n"}${boundary}`, */`${"\n"}${boundary}`], i, ab, t, max, o, 'value');
  //console.log('I3');
  i -= boundary.length - 1;
  return i;
}

// Read a file, like text, json or images
const readFile = async (boundary, i, ab, t, max, o) => {
  //console.log('J');
  i = await readString('"; filename="', i, ab, t, max);
  if (i === max) { return max; }
  //console.log('K');
  i = await captureUntil(['"'], i, ab, t, max, o, 'filename');
  if (i === max) { return max; }
  //console.log('L');
  i = await readString('"', i, ab, t, max);
  i = await readNewLine(i, ab, t, max);
  if (i === max) { return max; }
  //console.log('M');
  i = await readString('Content-Type: ', i, ab, t, max);
  if (i === max) { return max; }
  //console.log('N');
  i = await captureUntil(["\r", "\n"], i, ab, t, max, o, 'content-type');
  if (i === max) { return max; }
  //console.log('O');
  if (!o['content-type']) { o = null; return max; }
  if (i === max) { return max; }
  //console.log('P');
  i = await readNewLine(i, ab, t, max);
  if (i === max) { return max; }
  //console.log('Q');
  i = await readNewLine(i, ab, t, max);
  if (i === max) { return max; }
  //console.log('R');
  i = await captureUntil([`${"\r\n"}${boundary}`, `${"\n"}${boundary}`], i, ab, t, max, o, 'value', o['content-type']);
  if (i === max) { return max; }
  //console.log('S');
  i -= boundary.length - 1;
  return i;
}

const charAt = async (i, ab, t) => {
  return t.decode(ab.slice(i, i + 1));
}

// Check if end of file is found
// Meaningful only when used directly after readString(boundary)
const eof = async (i, ab, t, max) => {
  if (i + 1 === max) { return false; }
  const s = t.decode(ab.slice(i, i + 2));
  return s === '--';
}

// Try to read \n
// If \r is found, second chance is given
// If max reached, or if \n not found, return max to stop the machine
const readNewLine = async (i, ab, t, max) => {
  if (await charAt(i, ab, t) === "\r") { if (++i === max) { return max; } }
  if (await charAt(i, ab, t) === "\n") { return ++i; }
  return max;
}

const checkString = async (v, i, ab, t, max) => {
  return readString(v, i, ab, t, max, true)
}

// Try to read v
// If max reached, or if v not found, return max
const readString = async (v, i, ab, t, max, check = false) => {
  const start = i;
  let j = 0;
  const len = v.length;
  while (i < max && j < len) {
    ++i;
    ++j;
  }
  const s = t.decode(ab.slice(start, i));
  //console.log('expected : v :', v);
  //console.log(`${check ? 'check' : 'read '}${'    : s :'}`, s);
  if (s === v) { return check ? true : i; }
  return check ? false : max;
}

// Try to capture, until we get one the values inside a (Ex : \r or \n)
const captureUntil = async (a, i, ab, t, max, o, prop, contentType = 'text/plain') => {
  return readUntil(a, i, ab, t, max, true, o, prop, contentType);
}

// Try to read, until we get one the values inside a (Ex : \r or \n)
const readUntil = async (a, i, ab, t, max, capture = false, o = Object(), prop = '_', contentType = 'text/plain') => {
  const start = i;
  // Until end
  while (i < max) {
    //console.log(i, await charAt(i, ab, t));
    // For each token
    for (const v of a) {
      const end = Math.max(start, i - v.length);
      // Extract ending text, at the size of the token
      const tmp = t.decode(ab.slice(end, i));
      // If extracted ending text not equals to the token, do nothing
      if (tmp !== v) { continue; }
      // Capture data
      if (capture) { await captureData(ab, start, end, t, o, prop, contentType); }
      // Return current position
      --i;
      //console.log('readUntilOut1', i);
      return i;
    }
    // Go to next char
    ++i;
  }
  // console.log('readUntilOut2', i);
  return i;
}

const captureData = async (ab, start, end, t, o, prop, contentType) => {
  //console.log('capture', prop, contentType);
  //console.log(start, end);
  const sub = ab.slice(start, end);
  await captureJson(sub, t, o, prop, contentType);
  await captureTextAsJson(sub, t, o, prop, contentType);
  await captureText(sub, t, o, prop, contentType);
  await captureBlob(sub, o, prop, contentType);
}

const captureJson = async (sub, t, o, prop, contentType) => {
  if (contentType !== 'application/json') { return; }
  let s = t.decode(sub);
  //console.log('captureJson', s);
  o[prop] = JSON.parse(s);
}

const captureTextAsJson = async (sub, t, o, prop, contentType) => {
  const s = t.decode(sub);
  if (contentType !== 'text/plain' || s[0] !== '{') { return; }
  //console.log('captureTextAsJson1');
  let o2;
  //try { o2 = JSON.parse(s); } catch (e) { }
  try { o2 = JSON.parse(s); } catch (e) { }
  if (!o2) { return; }
  o[prop] = o2;
  o['content-type'] = 'application/json';
  //console.log('captureTextAsJson2', s);
}

const captureText = async (sub, t, o, prop, contentType) => {
  //console.log('captureText1', contentType);
  if (contentType !== 'text/plain' ||
    o['content-type'] && o['content-type'] !== 'text/plain') { return; }
  let s = t.decode(sub);
  //console.log('captureText2', s);
  o[prop] = s;
}

const captureBlob = async (sub, o, prop, contentType) => {
  if (contentType === 'text/plain' || contentType === 'application/json') { return; }
  //console.log('captureBlob');
  o[prop] = sub;
}