Portaly Vibe Payment Integration
Use this skill to help a human user finish a Portaly Vibe API integration quickly. Keep answers operational: prefer step lists, API request and response bullets, and copy-ready examples over long architecture explanations.
Portaly Vibe Payment Environments
Portaly Vibe Payment supports two modes per API key: live and test.
API Host
Use the following API host for both modes:
Payment site
Payment site URLs to which buyers are redirected for checkout:
Mode behavior
| Aspect | Live mode | Test mode |
|---|
| API key prefix | | |
| Payment provider | TapPay production | TapPay sandbox |
| Order storage | collection | collection |
| Callback payload | or absent | |
- Mode is set at API key creation time and cannot be changed after creation.
- A single merchant () can have both a live key and a test key active at the same time.
- All API endpoints accept both live and test keys. The mode is derived from the key, not from a request parameter.
- Test mode is intended for integration testing. Real charges are not made in test mode when using TapPay sandbox credentials.
- Plans and merchant config are shared across modes. They belong to the , not to the API key mode. A plan created with a live key is visible and usable with a test key, and vice versa. Do not create duplicate plans when switching between live and test keys — query existing plans first with
GET /api/creator-subscription/plans?profileId={profileId}
and reuse them.
Quick Start
- Before starting, AI agent should ask the human user to claim or create a Portaly Vibe Payment API key/CallbackSecret in the Portaly Vibe Dashboard at
https://portaly.ai/dashboard
and store the issued secret material safely.
- Ask the human user whether they want a live or test key. Recommend starting with a test key for integration development.
- Confirm what the human user is trying to build.
Prepare for payment integration tasks such as:
- create merchant config
- create subscription plans
- upload merchant or plan images (Agent should ask human user to provide image assets if needed)
- After setup, integrate the checkout session creation and callback handling into current system:
- create checkout session before buyer initiates payment
- redirect buyer to Portaly vibe checkout
- verify and consume the callback from Portaly after checkout completion
- if the integration needs subscription lifecycle management, also wire cancel and resume APIs for recurring plans
- if the integration needs subscriber self-service (letting subscribers manage their own subscriptions), wire the portal session API
- Start with
references/api-contract.md
.
Use it for endpoint lists, auth, request bodies, response bodies, and callback headers.
- Load
references/checkout-and-renewal.md
only when needed.
Use it only as supplemental reference when the human user asks about post-checkout charging, renewal, payout, invoice, or bridge-order behavior.
- Return implementation-ready output.
Prefer numbered steps, API endpoint lists, request and response bullets, and Node.js or TypeScript examples.
Output Style
- Write for an AI agent that is helping a human user complete integration work.
- Lead with the next concrete steps the human should take.
- Be explicit when an API can be called directly by the Agent with the Portaly Vibe Payment API key.
- Prefer using the setup APIs directly for merchant config, plan creation, plan updates, image uploads, and checkout session creation when the user has already provided valid credentials and required inputs.
- Use lists for:
- setup steps
- API endpoints
- required headers
- request fields
- response fields
- callback verification steps
- Prefer concise code samples in JavaScript or TypeScript when the user does not ask for another stack.
- Keep Portaly-owned behavior and third-party-owned behavior clearly separated.
Workflow
1. Apply for the API key
-
Require a Portaly Vibe Payment API key and CallbackSecret for this integration.
-
Instruct the human user to apply for or create the Portaly Vibe Payment API key in the Portaly Vibe Dashboard at
https://portaly.ai/dashboard
.
-
Ask whether the user wants a
live key (
) or a
test key (
). Recommend starting with a test key for development and switching to live for production.
-
Be explicit that this step is performed by a human operator in Portaly Vibe Dashboard, not by the third-party integration code.
-
Tell the human user to store the issued secret material safely, or store it on the user's behalf only in an appropriate secret manager or secure environment store.
-
Explain that the API key is used for bearer authentication in API calls and the
is used for verifying the authenticity of callbacks from Portaly If user asking.
-
Never ask the user to paste the API key or into chat. Chat transcripts can be logged, cached, or echoed back by the model in summaries, diffs, or tool call arguments. Treat secrets as values the agent never needs to see in plaintext.
-
Instead, instruct the human user to place the secrets into
themselves (via their editor or shell), using this template:
PORTALY_API_KEY=pcs_live_xxx # or pcs_test_xxx for test mode
PORTALY_CALLBACK_SECRET=xxx
-
The agent reads these at runtime via
process.env.PORTALY_API_KEY
(Node) or
os.environ["PORTALY_API_KEY"]
(Python) — it never needs the literal secret value in-context.
-
If the project uses a secret manager (1Password CLI, Doppler, AWS/GCP Secrets Manager, Vault, etc.), prefer that over
.
-
Before proceeding, verify that includes . If
does not exist or does not include
, create or update it immediately. Never allow credentials to be committed to version control.
-
If the user does paste a secret into chat by mistake, advise them to rotate the key in the Portaly Vibe Dashboard before using it — assume the pasted value is compromised.
2. Configure merchant settings
- Agent should perform these setup actions directly by API call with the Portaly Vibe Payment API key.
- Use the Config APIs when the human user needs to set merchant branding before any product goes live.
- AI Agent should ask the human user to provide a image asset, use the config image upload API to upload image to Portaly. The merchant logo is optional — if the user does not have one ready, skip this step and proceed with plan creation.
- Use
PUT /api/creator-subscription/config
and POST /api/creator-subscription/config/images
to set up merchant branding with the Portaly Vibe Payment API key.
3. Create a valid subscription plan
- Agent should perform plan creation, plan updates, and plan image uploads directly by API call with the Portaly Vibe Payment API key.
- Before creating a new plan, always query existing plans with
GET /api/creator-subscription/plans?profileId={profileId}
using the current API key. Plans are shared across live and test modes — if a suitable plan already exists, reuse it instead of creating a duplicate.
- Require at least one active plan in Portaly before creating a checkout session.
- Use the Plan APIs to create or update the product basics that the human user wants to list on Portaly.
- Confirm the plan name, description, amount, currency, billing period (, , or ), pricing type ( or ), and status match the intended product.
- For dynamic pricing plans: set to and to . The amount is not set on the plan; instead, the caller passes when creating each checkout session.
- If the third party has its own product catalog, persist the Portaly together with the merchant's internal product or entitlement identifier.
- AI Agent should ask the human user to provide a plan image, use the plan image upload API to upload the image to Portaly.
- Treat the returned by Portaly as authoritative. Do not reconstruct it from guessed domains.
- After creating or updating a plan, check the response and for garbled text (mojibake). If corrupted, fix shell encoding and use
PUT /api/creator-subscription/plans/{planId}
to correct it. See the Windows encoding note in Guardrails.
3.5 Create discount codes (optional)
- Use the Discount Code APIs after at least one plan exists.
- A code carries an array of rules; each rule can target a different set of plans with its own discount and duration. Example: code with two rules — 50% off for 3 cycles (= 3 months) on the monthly plan, and 20% off for 1 cycle (= 1 year) on the yearly plan.
- Per rule, confirm with the human user:
- Discount type: (TWD off) / (% off) / (100% off).
- Duration: (default 1) or (typically with ). One cycle equals one billing period — a month for a monthly plan, a year for a yearly plan.
- appliesTo: (fallback for any plan not covered by a specific rule) or planIds (e.g. yearly plan only). At most one rule per code; planIds may not appear in more than one rule.
- Code-level params:
- Custom code: 3-40 chars, . Stored and displayed in UPPERCASE; lookup is case-insensitive on input. Unique per profile. Immutable post-create.
- Redemption window: / .
- Caps: (total) /
maxRedemptionsPerCustomer
(per email).
- Codes are shared across live and test modes (same as plans).
- Codes also serve as ref codes — see the skill for how to record at user registration. When a buyer with a recorded later checks out and verifies their email, Portaly auto-applies the matching rule, provided the code is still within its window.
- See
references/discount-code-examples.md
for example prompts and the parameter cheatsheet.
- Money-moving guard: live-mode discount creation requires explicit user confirmation (same rule as live-mode plan creation).
4. Create the checkout session
- Create a checkout session before the buyer initiates payment.
- Call
POST /api/creator-subscription/checkout-sessions
with Authorization: Bearer {api_key}
.
- Send and optional , , , , and string-keyed .
- Optional : when provided, Portaly validates and applies the discount up-front. Invalid codes return
400 INVALID_DISCOUNT_CODE
. When omitted, Portaly attempts to auto-apply a discount via the buyer's after their email is verified inside hosted checkout (no extra call needed from the merchant).
- Persist , , , and on the third-party side.
- The session response includes when a discount was applied at session creation; is always the post-discount amount the buyer will be charged.
- Redirect the buyer to .
5. Let Portaly run hosted checkout
- Treat Portaly hosted checkout as a black box from the third-party perspective.
- Do not ask the third party to collect card tokens or implement Portaly-owned payment steps.
6. Consume the result
- The primary external confirmation is the signed callback to .
- Callback is only dispatched when checkout status is . Non-completed outcomes (failed, canceled, expired) do not trigger a callback.
- For non-completed outcomes, poll
GET /api/creator-subscription/checkout-sessions/{sessionId}
as a fallback.
- Use manual
POST /api/creator-subscription/checkout-sessions/{sessionId}/complete
only as an exception flow when the user is building a non-hosted or recovery flow.
- Current implementation contract:
subscriptionId === checkoutSessionId === sessionId
.
- When a recurring checkout succeeds, human user's system may use the callback's directly as the for later cancel or resume API calls.
- Make it explicit to the human user that this is the current Portaly implementation contract and should be persisted on their side after checkout completion.
7. Verify and persist
- Verify with the API key's .
- Use the exact timestamp from .
- Reject callbacks where is older than 5 minutes to prevent replay attacks. Note: is an ISO datetime string, not Unix seconds.
- Serialize the callback payload with stable key ordering before HMAC.
- Reference implementations live in and
scripts/sign_callback.mjs
.
- After verification, persist , if present, , , , , and the raw callback body for auditing.
- If the callback payload does not include , persist as the recurring subscription identifier because the current implementation uses as .
- Use as an idempotency key — if a callback with the same has already been processed, skip duplicate handling to avoid double fulfillment.
- must use HTTPS. Serving over plain HTTP exposes the signature and payload in transit.
- Heads up — Portaly may auto-send a welcome/upgrade email when the callback fires. A successful () callback triggers Portaly's template by default. Symmetrically, a cancel call triggers . If the vibe coder already sends their own purchase-confirmation or cancellation email, disable the matching template before going live with
PUT /api/creator-email/templates/welcome_paid
(or ) carrying . See the skill for the full list of email types and disable workflow.
8. Manage recurring subscriptions
- Only recurring plans with
billingPeriod = monthly | yearly
support cancel or resume.
- Cancellation means stopping the next recurring charge. It is not a refund. In your system, the rights or content associated should remain active until the end of the current paid period, which is indicated by in the subscription record.
- Portaly currently supports merchant-system initiated subscription lifecycle actions through API key authenticated endpoints.
- Use the same Portaly Vibe Payment API key for these calls.
Recurring management APIs:
GET /api/creator-subscription/subscriptions
— list all subscriptions with pagination and filtering
GET /api/creator-subscription/subscriptions/{subscriptionId}
POST /api/creator-subscription/subscriptions/{subscriptionId}/cancel
POST /api/creator-subscription/subscriptions/{subscriptionId}/resume
Order query API:
GET /api/creator-subscription/orders
— list payment/order records with pagination
Recurring management rules:
- These APIs only accept
Authorization: Bearer {api_key}
- Do not use Firebase auth for merchant-system integrations
- does not support cancel or resume
- marks the subscription as
- only works before the subscription has become fully
Cancel request body:
json
{
"reason": "customer_requested",
"reasonNote": "optional note"
}
Resume request body:
What to persist for recurring lifecycle:
8.5. Wire invitation email CTA (optional)
If the merchant plans to use Portaly's invitation-email flow to recruit followers (waitlist signups, campaigns), the CTA in those emails redirects through
https://portaly.ai/r/{code}
to a waitlist landing page. By default the page is hosted by Portaly; the vibe coder can also host it themselves on their own domain by setting
on the merchant config.
This is a separate concern from payment integration — for the full setup, install and follow the
skill:
npx skills add portaly-ai/portaly-skills --skill portaly-email
.
9. Enable subscriber self-service portal (optional)
- Use this when the merchant wants subscribers to manage their own subscriptions directly.
- The merchant backend creates a portal session via
POST /api/creator-subscription/portal-sessions
on , then redirects the subscriber to the returned .
- This is a server-to-server call — the API key must never be exposed to the client.
- The subscriber lands on Portaly's hosted portal page, already authenticated via the session token. No additional login is required.
- In the portal, subscribers can view subscriptions, cancel, resume, and view payment history.
- Portal sessions expire after 30 minutes.
- The merchant must provide a so the subscriber can navigate back after managing their subscriptions.
- See
Portal Session (Subscriber Self-Service)
in references/api-contract.md
for full endpoint details and code examples.
Preferred Response Shape
When answering with this skill, prefer this order:
- Goal summary
- Human setup steps
- API list
- Request fields
- Response fields
- Callback handling steps
- Example code
- Troubleshooting notes
Guardrails
- Default to test mode for development. If the loaded key starts with , confirm with the human user that live mode is intended before making any API call. Never silently run against production billing.
- Money-moving actions require explicit user confirmation. Before calling any of the following, state the exact action, target ( / ), and mode (live/test), then wait for the user's "yes":
POST /subscriptions/{id}/cancel
POST /subscriptions/{id}/resume
POST /checkout-sessions/{id}/complete
(manual completion)
- Any plan creation/update in live mode
- Do not batch or loop these actions across multiple subscriptions without per-action confirmation.
- Prefer the hosted checkout flow whenever possible. It already handles email verification, payment-method persistence, callback dispatch, subscription creation, payment creation, invoice task creation, and order bridge writes.
- Distinguish clearly between:
- setup APIs that the Agent can call directly with the Portaly Vibe Payment API key
- Do not invent provider behavior. TapPay and 91APP differ materially.
- Do not assume callback delivery means success without checking the and verified signature.
- Do not derive subscription state from redirect success pages alone. Redirects are UX only; callback or status query is the source of truth.
- Treat
references/checkout-and-renewal.md
as non-API background material. Load it only if the task explicitly touches recurring billing, payout, invoice follow-up, or bridge-order behavior.
- Windows encoding: On Windows, run (cmd) or
$OutputEncoding = [System.Text.Encoding]::UTF8
(PowerShell) before API calls containing non-ASCII text. If a plan's or comes back garbled, fix encoding and the correct values.
- Rate limiting: All creator-subscription API endpoints (except ) are rate limited. Read endpoints allow 120 requests/min, write endpoints allow 20 requests/min. If a response is received, use the header to schedule retries. When paginating through large result sets, be mindful of the rate limit budget.
Deliverables
When using this skill, aim to return one or more of:
- a minimal step-by-step integration plan for the human user
- a flat list of relevant APIs
- request and response field breakdowns
- callback verification code in the user's stack
- sample , , or TypeScript snippets
- a troubleshooting list keyed by session status
Resources
references/api-contract.md
Use for bearer auth, endpoint contract, callback headers, payload fields, and third-party implementation shape.
references/checkout-and-renewal.md
Use only as optional background for the high-level checkout lifecycle and renewal behavior.
references/discount-code-examples.md
Example prompts, parameter cheatsheet, and ref-code usage for the Discount Code APIs.
-
Use when you need a deterministic example of Portaly callback signing and verification.
scripts/sign_callback.mjs
Prefer this for Node.js, JavaScript, TypeScript, Express, or Next.js integrations.