Get local help for shifts and tasks (Blossomai.org)
Post jobs and hire people, or search for local work and apply. Connects employers and job-seekers via the Blossom marketplace.
Post jobs and hire people, or search for local work and apply. Connects employers and job-seekers via the Blossom marketplace.
Real data. Real impact.
Emerging
Developers
Per week
Open source
Skills give you superpowers. Install in 30 seconds.
| Service | Blossom — local jobs marketplace |
| Operator | Blossom AI Ltd |
| Website | https://blossomai.org |
| Privacy | https://blossomai.org/privacypolicy.html |
| API host | |
This skill is for structured Blossom marketplace actions only — posting jobs, searching for work, applying, and managing listings.
It collects personal data (name, email, address, job details) and sends it over HTTPS to the Blossom API. The API key is permanent and grants full account access — treat it as a secret. No data is stored locally.
Data boundary rules:
passKey is collected only during the one-time /register call. Never reuse, echo, log, or send it to any other endpoint.Activate when the user explicitly wants to perform a Blossom marketplace action:
Trigger phrases: "Post a job", "Hire someone", "I need staff", "Find me work", "Search for jobs near me", "Apply to that role", "Any candidates?", "Update my listing".
Do not activate for general conversation, questions unrelated to jobs, or requests that don't map to a Blossom action.
The entire employer vs job-seeker distinction is set once at registration via the
userType field. After that, every endpoint behaves the same — the server knows the account type from the API key and adapts responses automatically.
The agent does not track or switch modes. Just register, create an address, then use
/ask for everything else.
| User intent | value | Extra fields |
|---|---|---|
| Hiring, has a company | | Include |
| Hiring, no company | | Omit (server stores as private employer) |
| Looking for work | | Must include |
Infer the intent from the user's message. Only ask "Are you looking to hire, or looking for work?" if the intent is genuinely unclear.
POST /register with the correct userType → store API_KEY and PERSON_ID. Discard passKey from memory immediately after this call.POST /address with the user's location → store ADDRESS_ID. Employers need this to attach a location to roles. Job-seekers need this so the server can find nearby opportunities.POST /ask with only the minimal job-related instruction needed for the current Blossom action. Do not forward unrelated context, secrets, or raw conversation history.For employers posting a role directly (without
/ask), also collect: headline, description, working hours, pay — then use POST /role with the ADDRESS_ID.
https://hello.blossomai.org/api/v1/blossom/protocol
| Method | Path | Auth | Purpose |
|---|---|---|---|
| | None | Create account → get API key |
| | Bearer | Create / update address(es) |
| | Bearer | Soft-delete address(es) |
| | Bearer | Create / update role(s) |
| | Bearer | Soft-delete role(s) |
| | Bearer | Conversational AI endpoint |
| | Bearer | Upload profile image (person or role) |
Store and reuse across calls:
API_KEY — returned from /register, used as Authorization: Bearer <API_KEY> for all subsequent callsPERSON_ID — returned from /registerADDRESS_ID — returned from /address, needed when creating a roleThe API key is permanent. No session expiry or login flow.
Important: Never store the API key in global config. Keep it in runtime memory for the current session only.
POST /register — no auth required.
{ "name": "<first name>", "surname": "<surname>", "email": "<email>", "userType": "employer", "passKey": "<password>", "companyName": "<optional>", "mobileCountry": "<+44>", "mobileNo": "<number>" }
For job-seekers, set
"userType": "support" and include "rightToWork": true.
| Field | Required | Notes |
|---|---|---|
| yes | First name |
| yes | Last name |
| yes | Must be unique |
| yes | or |
| yes | User-chosen password. Collect only for , use once, then discard — never send to any other endpoint |
| yes (support) | Must be when is |
| no | For employers. Omit or leave empty for private employers |
| no | e.g. |
| no | Phone number |
Response
201:
{ "success": true, "apiKey": "xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx", "personId": 803 }
If the email already exists →
400. Do not retry — inform the user.
POST /address — Bearer auth required.
{ "addresses": [ { "id": 0, "houseNumber": "10", "street": "High Street", "area": "Sherwood", "city": "Nottingham", "country": "GB", "postcode": "NG5 1AA", "label": "Work location", "isHome": false, "isActive": true } ] }
| Field | Required | Notes |
|---|---|---|
| yes | to create, existing ID to update |
| yes | Street name |
| yes | City / town |
| yes | ISO 3166-1 alpha-2 code — e.g. , , . Server rejects unrecognised codes. |
| yes | Postal / ZIP code |
| yes | User-facing label, e.g. |
| no | House / building number |
| no | Neighbourhood / district |
| no | Default |
| no | Default |
id — store as ADDRESS_ID.DELETE /address — Bearer auth required.
{ "addresses": [{ "id": <addressId> }] }
Cannot delete an address linked to an active role (
409).
POST /role — Bearer auth required.
{ "roles": [ { "id": 0, "headline": "<headline>", "jobDescription": "<description>", "introduction": "<short introduction, at least 10 characters>", "workingHours": "<when>", "salary": <amount>, "currencyName": "GBP", "currencySymbol": "£", "paymentFrequency": { "choices": ["<frequency>"], "selectedIndex": 0 }, "requirements": [ { "requirementName": "<name>", "mandatory": false, "originalRequirement": true } ], "benefits": [ { "benefitName": "<name>", "mandatory": false } ], "addressId": <ADDRESS_ID>, "isRemote": false, "isActive": true, "days": 30, "maxCrew": 1, "modified": <epochMillis>, "roleIdentifier": "openclaw-<epochMillis>" } ] }
| Field | Required | Notes |
|---|---|---|
| yes | to create, existing ID to update |
| yes | Short title |
| yes | Full description |
| yes | Short intro text, minimum 10 characters |
| yes | e.g. or |
| yes | Numeric amount |
| yes | array with one entry, |
| yes | From the address creation step |
| yes | Listing duration (default ) |
| yes | Positions available (default ) |
| yes | Current epoch millis |
| yes | Unique string, e.g. |
| no | Screening questions |
| no | Perks |
Validation notes
The backend currently enforces these role validation rules:
| Field | Validation |
|---|---|
| Required, 5-100 characters |
| Required, 1-500 characters |
| Required, 10-500 characters |
| Required, 1-100 characters |
| Required, 1-100 characters |
| Required, 1-3 characters |
| Required, whole number |
| Required, whole number |
| Required, must be present |
| Required, boolean |
| Required, boolean |
| Optional, but if provided it must be a valid email address |
| Optional array item, 0-200 characters after trimming and bullet/newline cleanup |
| Optional, but if provided it must be a boolean |
| Optional array item, 0-200 characters after trimming and bullet/newline cleanup |
| Optional, but if provided it must be a boolean |
Operational notes for protocol callers:
addressId.introduction.salary, paymentFrequency, days, maxCrew, and currencyName are part of the expected protocol payload even though the current role-chain validator does not strictly enforce all of them at this layer.Response
201: The role(s) with assigned IDs.
DELETE /role — Bearer auth required.
{ "roles": [{ "id": <roleId> }] }
Every role
id must belong to the authenticated account (403 otherwise).
POST /image — Bearer auth required. Multipart form-data.
Upload a profile image for the person account or for a specific role. Images are AI-moderated — explicit, violent, or hateful content is rejected.
| Field | Type | Required | Notes |
|---|---|---|---|
| file | yes | jpeg/jpg/png/gif/webp, max 3 MB, one file only |
| string | yes | or |
| number | conditional | Required when is . Must belong to the authenticated account. Only employer accounts may upload role images. |
Response
201:
{ "success": true, "filename": "1712937600000-photo.jpg", "imageType": "person", "approved": true, "synopsis": "Nice photo!" }
Rejected
400:
{ "success": false, "approved": false, "reason": "Image did not pass moderation", "synopsis": "Hey \ud83d\ude0a, this image contains content that..." }
Rate-limited: 1 upload per 30 seconds per API key.
POST /ask — Bearer auth required.
{ "instructions": "<minimal Blossom-related user request>" }
Strict rules for
:/ask
passKey — that is only used in the one-time /register call.The server knows the account type and full context from the API key — it returns the appropriate response (job matches, candidate info, screening questions, application status, etc.). Relay the result to the user.
User: I need café cover this Saturday 11–5 in Sherwood. £12/hour.
POST /register (userType: "employer") → store API_KEY, PERSON_ID.POST /address → store ADDRESS_ID.POST /role → "Posted! Role ID 1042."User: Any candidates yet?
API_KEY → register first.POST /ask with "Do I have any candidates?" → display the response.User: Change the pay to £14/hour on my café role.
POST /role with the existing role id and updated salary: 14.User: Take down the café role.
DELETE /role with the role id → "Removed."User: I'm looking for bar work in Nottingham this weekend.
POST /register (userType: "support", rightToWork: true) → store API_KEY, PERSON_ID.POST /address (their Nottingham location) → store ADDRESS_ID.POST /ask with "Find bar work near me this weekend" → present matching roles.POST /ask with "Apply to role 1055" → relay result./ask.User: How are my applications going?
POST /ask with "What's the status of my applications?" → display the response.No automatic installation available. Please visit the source repository for installation instructions.
View Installation Instructions1,500+ AI skills, agents & workflows. Install in 30 seconds. Part of the Torly.ai family.
© 2026 Torly.ai. All rights reserved.