Create KYC Validation
Session-based validation
Create KYC Validation
Initiate a KYC verification session for a person entity — in the gu1 KYC API for identity verification flows, with examples for create validation use cases.
POST
Create KYC Validation
Overview
This endpoint creates a new KYC validation session for a person entity using a configured integration provider. After creating the validation, you’ll receive a verification URL that you can share with your customer to complete identity verification.Complete Flow Diagram
Full sequence from entity creation to KYC validation (production flow; in sandbox with test document numbers the provider step is skipped and the API returns the mock result and webhooks immediately—see Sandbox mock data):Key Points in the Flow
Key Points in the Flow
1. Entity and duplicate taxId
- Create the person entity first via
POST /api/entities(withcountryCode). If thetaxIdalready exists in the organization, the API returns 409 and does not create a duplicate; use the existing entity’s ID. - You need an existing
entityIdto create a KYC validation.
global_gueno_validation_kycis the standard code for full KYC verification and works in sandbox with no extra setup.
- In production, the response includes
providerSessionUrl. Send that URL to your user; they complete the flow on the provider’s hosted page (document + selfie). The URL is valid untilexpiresAt. - In sandbox, if the entity’s document number is in the test list, no provider session is used: the API returns 201 with
pendingstatus and moments later updates the validation and sends webhooks (e.g.kyc.validation_approvedorkyc.validation_rejected) with no user step. Important: You must have a webhook endpoint configured to receive the responses - see Sandbox mock data.
- When the validation finishes, the API sends a webhook to your URL. The event is one of:
kyc.validation_approved,kyc.validation_rejected,kyc.validation_abandoned,kyc.validation_expired,kyc.validation_cancelled(not a single “completed” event). The payload is the full validation object. - You can also poll
GET /api/kyc/validations/:idfor status updates.
Prerequisites
Before creating a KYC validation:- Person entity must exist: Create a person entity using the Entities API
- KYC integration configured: Your organization must have a KYC integration provider activated (e.g.,
global_gueno_validation_kyc) - Valid API key: Authenticate with your API key
Sandbox vs Production: Sandbox environments do NOT require profile setup or pre-configuration. You can test KYC validations immediately in sandbox with test data. Production environments require:
- Completed organization onboarding
- KYC integration provider activated by gu1 team
- Sufficient credit balance for KYC operations
Mock Data (Sandbox)
In sandbox, when the person entity’s document number (taxId) matches one of our test values, the API returns an immediate mock result (e.g. approved, rejected, cancelled) and sends the corresponding webhooks—no real verification is run. Document format does not matter (e.g. 99.990.001 and 99990001 both work).
For the full list of test document numbers by format (Argentina DNI/CUIT, Brazil CPF/CNPJ), expected outcomes, and example responses, see Sandbox mock data.
Important Behaviors
Duplicate taxId (POST /entities)
Multiple KYC Validations per Entity
You can create multiple KYC validations for the same entity:- Each validation gets a unique ID and session
- Only the most recent approved validation is marked as
isCurrent: true - Use cases: Re-verification, expired validations, failed attempts
Request
Endpoint
Headers
Query Parameters (Optional)
When
true, enables RENAPER double-check for Argentina entities. On terminal statuses the API queries the official registry (data and, when applicable, biometrics) and stores results in metadata. Only when OCR KYC verification returns status approved can a cross-check failure automatically set the validation to rejected; on in_review or rejected the check is informational (enforcementApplied: false).Type: boolean (e.g. ?doubleCheckRenaper=true)Requirements: Entity must be from Argentina (countryCode === 'AR'), and the organization must have RENAPER credentials configured in System Settings → Organizations → Config. If credentials are missing, the creation may fail or the double-check will be skipped depending on configuration.Body Parameters
The UUID of the person entity to verifyType:
string (uuid)The integration provider code for KYC validationStandard Value:
global_gueno_validation_kyc (recommended for most use cases)Type: string (min length: 1)What is integrationCode?The
integrationCode identifies which KYC provider integration to use for verification. Think of it as selecting the verification service.Available Integration Codes:global_gueno_validation_kyc- Recommended - Full KYC with document + selfie + face match + liveness- Custom codes may be configured for your organization (contact support)
- Log in to gu1 Dashboard
- Navigate to Settings → Integrations → KYC Providers
- Your active integration code will be listed there
global_gueno_validation_kyc works immediately with no configuration.Same as query param
doubleCheckRenaper. When true, enables RENAPER double-check for Argentina. Can be sent in the body or as ?doubleCheckRenaper=true. If both are present, the query param takes precedence.Optional list of KYC warning risk codes (exact strings). Stored on the validation as
metadata.omitWarnings. After the session completes, if the validation would be in_review, warnings is non-empty, and every code in warnings appears in this list, the API sets status to approved and keeps all warnings for display and audit. If any warning is not in omitWarnings, status stays in_review. If warnings is empty while status is in_review, this rule does not auto-approve. Invalid or unknown codes in the request body return 400. When the rule applies, metadata.kycOmitWarningsApplied records the decision timestamp and matched warnings.Non-omittable codes (e.g. GUENO_CROSS_ENTITY_DUPLICATED) are rejected in this field with 400 and always block omit auto-approve even if present in warnings.Type: string[] (each element must be an allowlisted code; duplicates are ignored)Response
Success Response (201 Created)
metadata.doubleChecks.renaper is set to true at creation. After OCR KYC verification completes and the double-check runs, metadata.responseDoubleChecks.renaper is populated (see RENAPER double-check below).
Response Fields
Unique identifier for this KYC validation
The person entity being verified
Your organization ID
Identity verification session identifier
Current validation status. Possible values:
pending- Validation created, waiting for customer to startin_progress- Customer is completing verification (filling out form)in_review- Verification completed, requires manual review from compliance teamapproved- Verification successfulrejected- Verification failedexpired- Verification session expired (e.g. after 7 days)abandoned- Customer started but didn’t completecancelled- Validation manually cancelled
KYC provider name
The verification URL to share with your customer
Whether this is the current active validation for the entity
Custom metadata. When RENAPER double-check is requested, includes
doubleChecks: { renaper: true }. After approval and double-check execution, may include responseDoubleChecks.renaper (see RENAPER double-check section).Timestamp when validation was created
Timestamp of last update
After approval, media keys appear inside
decision. To download files (images, video), use GET /api/kyc/validations/:id/media?key=... with Authorization: Bearer. Full details: Get KYC validation media.Example Request
Error Responses
Entity Not Found (404)
Invalid Entity Type (400)
KYC Not Configured (400)
RENAPER Double-Check (Argentina)
WhendoubleCheckRenaper is true and the entity is from Argentina, on each terminal status (approved, rejected, in_review) the API runs a cross-check against the official Argentine registry (RENAPER) when sufficient OCR data exists. Automatic rejection by RENAPER only applies when OCR KYC verification returned status approved.
How to Send It
- Body:
{ "entityId": "...", "integrationCode": "...", "doubleCheckRenaper": true } - Query:
POST /api/kyc/validations?doubleCheckRenaper=truewith the same body (without the field). Query param overrides body if both are present.
When It Runs
- Validation is created with
metadata.doubleChecks.renaper: true. - User completes verification; OCR KYC verification returns a terminal status.
- API calls RENAPER (data + biometrics when applicable) and stores
comparisonResultsin metadata. - If status is
approvedand the cross-check fails → validation becomesrejected. - If status is
in_revieworrejected→ results are informational; mismatch codes are added towarningswithout replacing OCR verification warnings. - Manual approve from
in_reviewdoes not re-run RENAPER; the human decision stands on the data already shown in review.
Where the Result Is Stored
All RENAPER double-check results are stored inmetadata.responseDoubleChecks.renaper. Mismatch codes are added to metadata.warnings alongside OCR KYC verification warnings. errorCode on the renaper object keeps the first failure for compatibility; UIs can list every code from comparisonResults and warnings.
Shape of metadata.responseDoubleChecks.renaper:
| Field | Type | Description |
|---|---|---|
verified | boolean | true if the double-check passed. |
matchResult | string | "match" | "mismatch" | "error". |
verifiedAt | string | ISO timestamp when the check was run. |
personalNumber | string | Transaction number from KYC (normalized). |
idTramitePrincipal | string | Transaction number from RENAPER. |
renaperData | object | Raw RENAPER API response (see renaperData shape). |
comparisonResults | object | Per-field OCR vs registry comparison (when applicable). |
renaperBiometric | object | ABIS biometric result (validate-dni): matchResult, score, renaperData (raw response). |
errorCode | string | Present when failed; see error codes below. |
error | string | Optional legacy message; prefer using errorCode and translating in your UI. |
renaperData shape
This is the unmodified body returned by the registry via ms-providers (POST …/provider-records/renaper/data). Gu1 forwards it as-is in metadata.responseDoubleChecks.renaper.renaperData. Field names use snake_case; all fields are optional depending on what RENAPER returns for each lookup.
Example (successful double-check, matchResult: "match"):
| Situation | Typical renaperData |
|---|---|
Error before registry data (e.g. RENAPER_DNI_MISSING) | {} |
Service unavailable (RENAPER_VERIFICATION_UNAVAILABLE) | { "error": "…", "code": "SERVICE_UNAVAILABLE" } |
| Failed cross-check with partial registry payload | Subset of fields (e.g. id_tramite_principal, apellido, nombres, fecha_nacimiento) |
comparisonResults shape
Per-field map (dni, tramite, name, ejemplar, dateOfBirth, expirationDate). Each entry may include:
| Field | Type | Description |
|---|---|---|
compared | boolean | true if comparison was attempted; false if skipped due to missing data. |
passed | boolean | Outcome when compared is true. |
skipReason | string | ocr_missing, renaper_missing, or both_missing when not compared. |
ocrValue | string | OCR-extracted value (when applicable). |
renaperValue | string | Registry value from RENAPER (when applicable). |
similarityPercent | number | For name only: Levenshtein similarity (0–100). |
threshold | number | For name only: applied threshold (e.g. 80). |
errorCode | string | Field failure code (e.g. RENAPER_DNI_NOT_MATCH). |
ejemplar, the OCR value is extractedData.ejemplar (see extractedData fields). Comparison runs when both OCR and RENAPER values exist.
Example — extractedData on an approved KYC validation (Argentina):
renaperBiometric shape
Nested under metadata.responseDoubleChecks.renaper.renaperBiometric when the org has biometric credentials and the KYC session provides a selfie. Gu1 sends one selfie to validate-dni; RENAPER compares it to the ID photo on file.
| Field | Type | Description |
|---|---|---|
verified | boolean | true when the ABIS biometric check returns resultado.match. |
matchResult | string | "match" | "mismatch" | "error". |
score | number | Score from the biometric check (when present). |
verifiedAt | string | ISO timestamp of the biometric check. |
renaperData | object | Raw validate-dni response (includes resultado.match, resultado.score, etc.). |
submittedSelfieRef | string | Reference of the selfie sent: storage path (kyc/...) or URL (https://...). |
submittedSelfieRefKind | string | Format of submittedSelfieRef: s3_key (internal path) | url (public URL). |
submittedSelfiePickedFrom | string | Source in decision: liveness_reference_image or face_match_target_image. |
errorCode | string | On failure (e.g. RENAPER_BIOMETRIC_NOT_MATCH). |
skipReason | string | When not run (e.g. selfie unavailable). |
Full responseDoubleChecks.renaper example
renaperBiometric and comparisonResults entries may be omitted depending on credentials, OCR data, or selfie availability.
Error Codes (When RENAPER Fails)
When the double-check fails,metadata.warnings contains one of the following codes (and metadata.responseDoubleChecks.renaper.errorCode is set to the same value). Your UI should translate these codes into user-facing messages.
| Code | Meaning |
|---|---|
RENAPER_DNI_MISSING | No document number (DNI) was obtained from the KYC verification. |
RENAPER_GENDER_MISSING | Gender (M/F) is required to call RENAPER and was not available. |
RENAPER_VERIFICATION_UNAVAILABLE | Could not complete the cross-check with the registry (e.g. network/API error). Retry later. |
RENAPER_DNI_NOT_MATCH | The document number from the verification does not match the official registry. |
RENAPER_TRAMITE_DATA_MISSING | Transaction number could not be compared (missing from KYC or RENAPER response). |
RENAPER_TRAMITE_ID_NOT_MATCH | The document transaction number does not match the registry. Common when the person used an older copy of the ID, the document is expired, or it was reissued; they should verify again with their current, valid ID. |
RENAPER_NAME_NOT_MATCH | OCR name is below the 80% similarity threshold vs the RENAPER registry. |
RENAPER_EJEMPLAR_NOT_MATCH | ID copy letter (ejemplar) does not match the RENAPER registry. |
RENAPER_DOB_NOT_MATCH | OCR date of birth does not match the RENAPER registry. |
RENAPER_EXPIRY_NOT_MATCH | OCR expiration date does not match the RENAPER registry. |
RENAPER_BIOMETRIC_NOT_MATCH | Selfie does not match the ID photo in the biometric registry. |
RENAPER_BIOMETRIC_UNAVAILABLE | Biometric check could not be completed (network, credentials, or service). |
When RENAPER enforcement applies (automatic rejection)
| OCR KYC verification status | RENAPER runs? | Can RENAPER reject? |
|---|---|---|
approved (direct approval) | Yes | Yes — cross-check failure → rejected |
in_review | Yes | No — informational; compliance decides via manual review |
rejected (OCR verification or Gu1 rules) | Yes (if OCR available) | No — informational; registry data kept in metadata |
Manual approve from in_review (POST …/approve) | No (reuses stored check) | No — human decision |
Why the validation may be rejected
RENAPER can change status torejected only on the direct approved path. On in_review / rejected / manual approve, failures appear as codes in metadata.warnings and in metadata.responseDoubleChecks.renaper without auto-rejection. Use the table above and translation keys (e.g. kyc.rejectionReasonCodes.*).
Next Steps
After creating a validation:- Extract the verification URL from
providerSessionUrl - Share the URL with your customer via email, SMS, or in-app
- Set up webhook endpoint to receive completion notifications
- Monitor validation status using the validation ID
- After approval, read media keys from
decisionand download files — see Get KYC validation media
Get KYC URL
Learn how to retrieve the URL
Webhook Integration
Configure webhook notifications