Skip to content

Handle form submissions with Node.js

Learn how to handle form submissions in Node.js using the Forminit SDK.

Forminit is a form backend service that handles form submissions, data storage, and notifications — so you don’t have to.

With Forminit, you get all of this out of the box:

  • No database required — Submissions are stored securely in Forminit’s infrastructure
  • Built-in email notifications — Get notified instantly when forms are submitted
  • Automatic validation — Email, phone, URL, and country fields are validated server-side
  • File upload handling — Accept files up to 25 MB without configuring storage
  • Spam protection — Integrates with reCAPTCHA, hCaptcha, and honeypot
  • UTM tracking — Automatically capture marketing attribution data
  • Dashboard access — View, search, and export submissions from a web interface
  • Webhook support — Forward submissions to your own systems when needed

Forminit reduces form backend development from days to minutes. Focus on your application logic while Forminit handles the infrastructure.


Before integrating Forminit with your Node.js application:

  1. Create a Forminit account at forminit.com
  2. Create a form in your dashboard
  3. Create an API key

# npm
npm install forminit

# yarn
yarn add forminit

# pnpm
pnpm add forminit

Set authentication to Protected for server-side integrations. This enables higher rate limits and requires the x-api-key header.

Forminit Authentication Mode Protected

3. Create an API Token and Add to Environment

Section titled “3. Create an API Token and Add to Environment”

Generate your secret API token from Account → API Tokens in the Forminit dashboard.

# .env
FORMINIT_API_KEY="fi_your_secret_api_key"

Forminit uses a block-based system to structure form data. Each submission contains an array of blocks representing different field types.

For complete documentation on all available blocks, field naming conventions, and validation rules, see the Form Blocks Reference.

Forminit supports two submission formats: FormData and JSON.

Best for forms with file uploads or when processing HTML form data:

import { Forminit } from 'forminit';

const forminit = new Forminit({
  apiKey: process.env.FORMINIT_API_KEY,
});

const FORM_ID = 'frm_abc123xyz';

const formData = new FormData();
formData.append('fi-sender-firstName', 'John');
formData.append('fi-sender-lastName', 'Doe');
formData.append('fi-sender-email', 'john@example.com');
formData.append('fi-text-message', 'Hello world');

const { data, redirectUrl, error } = await forminit.submit(FORM_ID, formData);

if (error) {
  console.error('Submission failed:', error.message);
  return;
}

console.log('Submission successful:', data.hashId);

FormData Field Naming:

Block TypePatternExample
Sender propertiesfi-sender-{property}fi-sender-email
Field blocksfi-{type}-{name}fi-text-message

Best for programmatic submissions with structured data:

import { Forminit } from 'forminit';

const forminit = new Forminit({
  apiKey: process.env.FORMINIT_API_KEY,
});

const FORM_ID = 'frm_abc123xyz';

const { data, redirectUrl, error } = await forminit.submit(FORM_ID, {
  blocks: [
    {
      type: 'sender',
      properties: {
        email: 'john@example.com',
        firstName: 'John',
        lastName: 'Doe',
        phone: '+12025550123',
        company: 'Acme Corp',
      },
    },
    {
      type: 'text',
      name: 'message',
      value: 'I would like to learn more about your services.',
    },
    {
      type: 'select',
      name: 'plan',
      value: 'Enterprise',
    },
    {
      type: 'rating',
      name: 'urgency',
      value: 4,
    },
  ],
});

if (error) {
  console.error('Submission failed:', error.message);
  return;
}

console.log('Submission ID:', data.hashId);
console.log('Submitted at:', data.date);

File uploads require FormData and multipart handling. Here’s an example using Express with Multer:

import express from 'express';
import multer from 'multer';
import { Forminit } from 'forminit';

const app = express();
const upload = multer({ storage: multer.memoryStorage() });

const forminit = new Forminit({
  apiKey: process.env.FORMINIT_API_KEY,
});

const FORM_ID = 'frm_abc123xyz';

app.post('/apply', upload.fields([
  { name: 'resume', maxCount: 1 },
  { name: 'documents', maxCount: 5 }
]), async (req, res) => {
  const formData = new FormData();

  formData.append('fi-sender-firstName', req.body.firstName);
  formData.append('fi-sender-lastName', req.body.lastName);
  formData.append('fi-sender-email', req.body.email);
  formData.append('fi-text-cover_letter', req.body.coverLetter);

  if (req.files.resume) {
    const file = req.files.resume[0];
    formData.append('fi-file-resume', new Blob([file.buffer]), file.originalname);
  }

  if (req.files.documents) {
    for (const file of req.files.documents) {
      formData.append('fi-file-documents', new Blob([file.buffer]), file.originalname);
    }
  }

  const { data, error } = await forminit.submit(FORM_ID, formData);

  if (error) {
    return res.status(400).json({ success: false, message: error.message });
  }

  res.json({ success: true, submissionId: data.hashId });
});

app.listen(3000);

File Upload Limits:

LimitValue
Maximum upload size per submission25 MB
Maximum file blocks per submission20

The SDK returns { data, redirectUrl, error }. On successful submission:

const { data, redirectUrl, error } = await forminit.submit(FORM_ID, payload);

// data contains:
{
  "hashId": "7LMIBoYY74JOCp1k",
  "date": "2026-01-01 21:10:24",
  "blocks": {
    "sender": {
      "firstName": "John",
      "lastName": "Doe",
      "email": "john.doe@example.com"
    },
    "message": "Hello world",
    "plan": "Enterprise"
  }
}

// redirectUrl contains the thank you page URL
"https://forminit.com/thank-you"
FieldTypeDescription
data.hashIdstringUnique submission identifier
data.datestringSubmission timestamp (YYYY-MM-DD HH:mm:ss)
data.blocksobjectAll submitted field values
redirectUrlstringThank you page URL

When submission fails, error contains:

{
  "error": "FI_SCHEMA_FORMAT_EMAIL",
  "code": 400,
  "message": "Invalid email format for field: 'contact'. Please enter a valid email address."
}
FieldTypeDescription
errorstringError code identifier
codenumberHTTP status code
messagestringHuman-readable error message

Always wrap submissions in try-catch and handle errors appropriately:

import { Forminit } from 'forminit';

const forminit = new Forminit({
  apiKey: process.env.FORMINIT_API_KEY,
});

async function submitContactForm(formData) {
  try {
    const { data, redirectUrl, error } = await forminit.submit('frm_abc123xyz', formData);

    if (error) {
      switch (error.error) {
        case 'FI_SCHEMA_FORMAT_EMAIL':
          return { success: false, field: 'email', message: 'Invalid email address' };
        case 'FI_RULES_PHONE_INVALID':
          return { success: false, field: 'phone', message: 'Invalid phone number format' };
        case 'TOO_MANY_REQUESTS':
          return { success: false, message: 'Please wait before submitting again' };
        default:
          return { success: false, message: error.message };
      }
    }

    return {
      success: true,
      submissionId: data.hashId,
      redirectUrl,
    };
  } catch (err) {
    console.error('Submission error:', err);
    return { success: false, message: 'An unexpected error occurred' };
  }
}
Error CodeHTTP StatusDescription
FORM_NOT_FOUND404Form ID doesn’t exist or was deleted
FORM_DISABLED403Form is disabled by owner
MISSING_API_KEY401API key required but not provided
EMPTY_SUBMISSION400No fields with values submitted
FI_SCHEMA_FORMAT_EMAIL400Invalid email format
FI_RULES_PHONE_INVALID400Invalid phone number format
FI_SCHEMA_RANGE_RATING400Rating not between 1-5
FI_DATA_COUNTRY_INVALID400Invalid country code
TOO_MANY_REQUESTS429Rate limit exceeded

The SDK includes TypeScript definitions:

import { Forminit } from 'forminit';

interface ContactFormData {
  firstName: string;
  lastName: string;
  email: string;
  message: string;
}

const forminit = new Forminit({
  apiKey: process.env.FORMINIT_API_KEY!,
});

async function submitContact(formData: ContactFormData) {
  const { data, redirectUrl, error } = await forminit.submit('frm_abc123xyz', {
    blocks: [
      {
        type: 'sender',
        properties: {
          email: formData.email,
          firstName: formData.firstName,
          lastName: formData.lastName,
        },
      },
      {
        type: 'text',
        name: 'message',
        value: formData.message,
      },
    ],
  });

  if (error) {
    throw new Error(error.message);
  }

  return data;
}

Authenticated requests are limited to 5 requests per second.

When rate limited, you’ll receive a 429 status code:

{
  "error": "TOO_MANY_REQUESTS",
  "code": 429,
  "message": "Rate limit exceeded. Maximum 5 requests per second allowed.",
  "suggest": "Wait a moment before making another request."
}

  1. Store API keys in environment variables — Never hardcode keys in source code
  2. Use .gitignore — Exclude .env files from version control
  3. Validate input server-side — Don’t rely solely on client-side validation
  4. Use HTTPS — Always use secure connections in production
  5. Implement rate limiting — Protect your endpoints from abuse
  6. Sanitize user input — Prevent injection attacks
function validateContactForm(data) {
  const errors = [];

  if (!data.email || !/^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(data.email)) {
    errors.push('Valid email is required');
  }

  if (!data.firstName || data.firstName.length < 1) {
    errors.push('First name is required');
  }

  if (!data.message || data.message.length < 10) {
    errors.push('Message must be at least 10 characters');
  }

  return errors;
}

Enhance your forms with additional security: