Get started

The Jemeni.NET API enables interaction with its services, including user management, transactions, OTP verification, and payment handling

To use this API, you need an API key. Please sign in to get your own API key.

    {BASE_URL} : https://jemeni.net/api
    

Authentication Signature

To authenticate API requests, you need to include an auth-signature in the request headers. This signature is generated using the following formula:


    SHA-512(SK + "+" + TK + "+" + METHOD + "+" + URL + "+" + BODY_REQUEST + "+" + TIMESTAMP)
    

Parameters:

  • SK: Secret Key (provided in your API credentials)
  • TK: Token Key (your API access token)
  • METHOD: HTTP Method (e.g., GET, POST, PUT, DELETE)
  • URL: The full API endpoint URL
  • BODY_REQUEST: The raw request body (empty string "" for GET requests)
  • TIMESTAMP: Current UNIX timestamp (in seconds)

Example HTTP Request:


    POST /resource HTTP/1.1
    Host: api.example.com
    Authorization: Bearer your_token_key
    auth-signature: 2a4e9b10a57e4c3e7b2...
    Content-Type: application/json
    Timestamp: 1700000000

    {
        "data": "test"
    }
    
---

Implementation in Different Languages

🔹 C# Example


    using System;
    using System.Security.Cryptography;
    using System.Text;

    class Program
    {
        static void Main()
        {
            string secretKey = "your_secret_key"; // SK
            string tokenKey = "your_token_key";   // TK
            string method = "POST";               // HTTP Method
            string url ={BASE_URL} +"/resource"; // API URL
            string bodyRequest = "{\"data\":\"test\"}";      // JSON Request Body
            long timestamp = DateTimeOffset.UtcNow.ToUnixTimeSeconds(); // UNIX timestamp

            // Concatenate the signature string
            string signatureString = $"{secretKey}+{tokenKey}+{method}+{url}+{bodyRequest}+{timestamp}";

            // Compute the SHA-512 hash
            using (SHA512 sha512 = SHA512.Create())
            {
                byte[] hashBytes = sha512.ComputeHash(Encoding.UTF8.GetBytes(signatureString));
                string authSignature = BitConverter.ToString(hashBytes).Replace("-", "").ToLower();
                
                Console.WriteLine($"auth-signature: {authSignature}");
            }
        }
    }
    
---

🔹 PHP Example


    <?php
    $secretKey = "your_secret_key"; // SK
    $tokenKey = "your_token_key";   // TK
    $method = "POST";               // HTTP Method
    $url = {BASE_URL} +"/resource"; // API URL
    $bodyRequest = '{"data":"test"}';  // JSON Request Body
    $timestamp = time(); // UNIX timestamp

    // Concatenate the signature string
    $signatureString = "$secretKey+$tokenKey+$method+$url+$bodyRequest+$timestamp";

    // Compute SHA-512 hash
    $authSignature = hash("sha512", $signatureString);

    echo "auth-signature: " . $authSignature;
    ?>
    
---

🔹 Node.js Example


    const crypto = require('crypto');

    const secretKey = "your_secret_key"; // SK
    const tokenKey = "your_token_key";   // TK
    const method = "POST";               // HTTP Method
    const url = {BASE_URL} +"/resource"; // API URL
    const bodyRequest = JSON.stringify({ data: "test" }); // JSON Request Body
    const timestamp = Math.floor(Date.now() / 1000); // UNIX timestamp

    // Concatenate the signature string
    const signatureString = `${secretKey}+${tokenKey}+${method}+${url}+${bodyRequest}+${timestamp}`;

    // Compute SHA-512 hash
    const authSignature = crypto.createHash('sha512').update(signatureString).digest('hex');

    console.log("auth-signature:", authSignature);
    
---

🔹 Flutter (Dart) Example


    import 'dart:convert';
    import 'dart:crypto';

    void main() {
    String secretKey = "your_secret_key"; // SK
    String tokenKey = "your_token_key";   // TK
    String method = "POST";               // HTTP Method
    String url = {BASE_URL} +"/resource"; // API URL
    String bodyRequest = '{"data":"test"}';  // JSON Request Body
    int timestamp = DateTime.now().millisecondsSinceEpoch ~/ 1000; // UNIX timestamp

    // Concatenate the signature string
    String signatureString = "$secretKey+$tokenKey+$method+$url+$bodyRequest+$timestamp";

    // Compute SHA-512 hash
    var bytes = utf8.encode(signatureString);
    var digest = sha512.convert(bytes);

    print("auth-signature: ${digest.toString()}");
    }
    
---

🚀 Final Notes

  • The generated auth-signature must be included in the request headers.
  • Ensure your secret key (SK) is never exposed in client-side applications.
  • The TIMESTAMP should always be the current UNIX timestamp to prevent replay attacks.

Get Transaction State

    
        curl -X GET "{BASE_URL}/sandbox/payment/verify/state?transaction_id={id}" \
        -H "auth-apiKey: pk_test_37........bf76b6dc218" \
        -H "auth-timestamp: 2024-04-24 16:55:00" \
        -H "auth-token: xxxxxxxxxxxxxxxxxxxxxxxxxxxx" \
        -H "auth-signature: xxxxxxxxx ............ xxxx"
        -H 'Content-Type: application/json' \
        -H 'Accept: application/json' \
    

To get Transaction State you need to make a GET call to the following url :
{BASE_URL}/sandbox/payment/verify/state?transaction_id={id}


HEADERS PARAMETERS

Field Type Description
auth-apiKey String (required) Your API key.
auth-timestamp String (required) request timestamp.
auth-token String (required) User token

Example Response:


{
    "status": "",
    "message": "",
    "data": {
        "id": 0,
        "amount": "",
        "state": 0,
        "transaction_id": "",
        "created_at": "",
        "updated_at": "",
        "customer": {
            "id": 0,
            "first_name": "",
            "last_name": "",
            "email": "",
            "phone": "",
            "country_id": 0,
            "created_at": "",
            "updated_at": ""
        },
        "client_merchants": [
            {
                "id": 0,
                "client_id": 0,
                "code": "",
                "carrier_id": "",
                "active": true,
                "archived": true,
                "created_at": "",
                "updated_at": "",
                "deleted_at": null,
                "carrier": {
                    "id": 0,
                    "carrier_name": "",
                    "country_id": 0,
                    "display_name": "",
                    "active": true,
                    "archived": true
                }
            }
        ],
        "client": {
            "id": 0,
            "name": "",
            "client_type_id": 0,
            "status": 0,
            "verified_at": null,
            "location": "",
            "created_at": "",
            "updated_at": ""
        },
        "payment": {
            "id": "",
            "subscription_id":"",
            "amount": "",
            "due_at": "2024-11-22 19:57:14",
            "nature": null,
            "archived": false,
            "created_at": "2024-11-18T19:57:14.000000Z",
            "updated_at": "2024-11-18T19:57:14.000000Z",
            "deleted_at": null,
            "status": "",
            "payment_reference": "JP-XXXXXXXXXX"
        }
    }
}


Create Transaction (Send Payment)

This endpoint is used to send a payment. It is an HTTP POST request to the specified URL.

                
                curl -X POST "BASE_URL/sandbox/send/payment" \
                -H "auth-apiKey: pk_test_37........bf76b6dc218" \
                -H "auth-timestamp: 2024-04-24 16:55:00" \
                -H "auth-token: xxxxxxxxxxxxxxxxxxxxxxxxxxxx" \
                -H "auth-signature: xxxxxxxxx ............ xxxx"
                -H 'Content-Type: application/json' \
                -H 'Accept: application/json' \
                --data '{
                    "customer_phone":"",
                    "amount":"",
                    "device":"",
                    "lang":"",
                    "version":"",
                    "country_id":1,
                    "return_url":"https://your_domain.com/callback"
                }'
            
            

To Create transaction you need to make a POST call to the following url :
{BASE_URL}/sandbox/send/payment


HEADERS PARAMETERS

Field Type Description
auth-apiKey String (required) Your API key.
auth-timestamp String (required) request timestamp.
auth-token String (required) User token

Request Body

Field Type Description
customer_phone String (required) The phone number of the customer..
amount Number (required): The amount of the payment.
device String (text, optional): The device used for the transaction
lang String (optional): The language preference.
version String (optional): The version of API v1.
country_id Number (required): The ID of the country.
return_url String (optional): The URL to return to after the transaction.

Example Response:


{
    "status": "",
    "message": "",
    "data": {
        "id": 0,
        "amount": "",
        "state": 0,
        "transaction_id": "",
        "created_at": "",
        "updated_at": "",
        "customer": {
            "id": 0,
            "first_name": "",
            "last_name": "",
            "email": "",
            "phone": "",
            "country_id": 0,
            "created_at": "",
            "updated_at": ""
        },
        "client_merchants": [
            {
                "id": 0,
                "client_id": 0,
                "code": "",
                "carrier_id": "",
                "active": true,
                "archived": true,
                "created_at": "",
                "updated_at": "",
                "deleted_at": null,
                "carrier": {
                    "id": 0,
                    "carrier_name": "",
                    "country_id": 0,
                    "display_name": "",
                    "active": true,
                    "archived": true
                }
            }
        ],
        "client": {
            "id": 0,
            "name": "",
            "client_type_id": 0,
            "status": 0,
            "verified_at": null,
            "location": "",
            "created_at": "",
            "updated_at": ""
        },
        
    }
}


Create Subscription

This endpoint allows you to create a new subscription for a customer.

        
            curl --location '{BASE_URL}/sandbox/subscriptions/create' \
                --header 'auth-apiKey: pk_test_fce1••••••c821c60e' \
                --header 'auth-timestamp: request date' \
                --header 'auth-token: ••••••••••••••••••' \
                --header 'auth-signature: ••••••••••••••••••••••••' \
                --header 'Content-Type: application/json' \
                --header 'Accept: application/json' \
                --data '{
                    "phone" : "0122030465",
                    "plan_id" : 1,
                    "frequency_id":1,
                    "start_at" : "2024-10-27"
                }'
        
    

To Create Subscription you need to make a POST call to the following url :
{BASE_URL}/sandbox/subscriptions/create


HEADERS PARAMETERS

Field Type Description
auth-apiKey String (required) Your API key.
auth-timestamp String (required) request timestamp.
auth-token String (required) User token

Response Body

Field Type Description
plan_id String The unique identifier for the plan.
name string The name of the subscription plan.
description Number A brief description of the plan.
price number The price of the plan.
currency string The currency in which the price is denominated.

Example Response:


{
    "status": "Success",
    "message": "Subscription created successfully",
    "data": {
        "id": "",
        "client_id": "",
        "plan_id": "",
        "customer_id": 2,
        "status": "",
        "start_at": "2024-10-27 00:00:00",
        "expire_at": "2025-01-09 21:55:07",
        "archived": false,
        "created_at": "2025-01-02T21:55:07.000000Z",
        "updated_at": "2025-01-02T21:55:07.000000Z",
        "deleted_at": null,
        "renewal_date": null,
        "renewed": false
    }
}

List of plans

This endpoint is used to retrieve a list of subscription plans.

        
            curl --location '{BASE_URL}/sandbox/subscriptions/plans' \
                --header 'auth-timestamp: 2024-09-17 23:08' \
                --header 'auth-signature: ••••••••••••••••••••••••••••••••••••' \
                --header 'Content-Type: application/json' \
                --header 'Accept: application/json' \                
        
    

To get List of plans you need to make a POST call to the following url :
{BASE_URL}/sandbox/subscriptions/plans


HEARDERS PARAMETERS

Field Type Description
auth-apiKey String (required) Your API key.
auth-timestamp String (required) request timestamp.
auth-token String (required) User token

Request Body

Field Type Description
phone String (required) The phone number of the customer..
plan_id Number (required):The ID of the subscription plan.
frequency_id Number (required):The ID of the frequency for the subscription.
start_at String (string) - The start date of the subscription in the format "YYYY-MM-DD".

Example Reponse :

    
        {
            "status": "Success",
            "message": "List of subscription plans",
            "data": [
                {
                    "id": null,
                    "client_id": null,
                    "name": "",
                    "price": "",
                    "frequency_id": null,
                    "archived": false,
                    "created_at": "2024-09-14T18:41:00.000000Z",
                    "updated_at": "2024-09-14T18:41:00.000000Z",
                    "deleted_at": null
                },
                ...
            ]
        }
     

Disable Subscription

This endpoint is used to disable a subscription for a user.

    
        curl --location '{BASE_URL}/sandbox/subscriptions/disable' \
            --header 'auth-apiKey: ••••••••••••••••••••••••••••••••••••••••••' \
            --header 'auth-timestamp: requet_date' \
            --header 'auth-token: ••••••••••••••••••••••••' \
            --header 'auth-signature: ••••••••••••••••••••••••••••••••••••••••••' \
            --header 'Content-Type: application/json' \
            --header 'Accept: application/json' \
            --data '{
                "phone" : "0122030465"
            }'             
    

To Disable Subscription you need to make a POST call to the following url :
{BASE_URL}/sandbox/subscriptions/disable


HEADERS PARAMETERS

Field Type Description
auth-apiKey String (required) Your API key.
auth-timestamp String (required) request timestamp.
auth-token String (required) User token

Request Body

Field Type Description
phone String (required) The phone number of the customer..

Response Body

The response will include the details of the disabled subscription in the following JSON schema:

Example Response:


    {
        "status": "Success",
        "message": "Customer subscription disabled successfully",
        "data": {
            "id": "",
            "first_name": "",
            "last_name": "",
            "email": "email@email.ml",
            "phone": "0102030504",
            "remember_token": null,
            "created_at": "2024-09-24T22:06:25.000000Z",
            "updated_at": "2024-09-24T22:06:25.000000Z",
            "deleted_at": null,
            "country_id": 1
        }
    }


Streamline Payments with Our Payment Notification System

Our system offers an alternative to payment links by enabling you to send payment notifications directly to JemeniPay customers. This approach is ideal if you prefer not to use or share a payment link while still ensuring a seamless payment experience.

How It Works:

  • Initiate a Payment Notification: Use our system to send a detailed payment notification directly to the customer’s JemeniPay account.
  • Customer Receives the Notification: The customer is instantly notified about the payment request.
  • Direct Payment: The customer can pay you directly through JemeniPay without needing a separate payment link.

This feature ensures security, convenience, and flexibility for both you and your customers. It's a perfect solution for businesses seeking streamlined payment processes without relying on shared links.

Start using our Payment Notification System today and enjoy a faster, more direct way to receive payments!

How JemeniPay Webhook Works

JemeniPay's webhook system is designed to allow seamless communication between our platform and your application. Webhooks are a way for JemeniPay to notify your system in real-time when specific events occur, enabling automated and efficient workflows.

Steps of the Webhook Process:

  • Event Occurs: When a specific event takes place (e.g., a payment is successful, an invoice is created, or a subscription is renewed), JemeniPay generates an event notification.
  • Notification Sent: The event data is sent to a predefined URL (your webhook endpoint) via an HTTP POST request. This notification contains details about the event, typically in JSON format.
  • Endpoint Receives the Data: Your server processes the incoming webhook request. It might validate the signature (to ensure the request is authentic) and extract the event details.
  • Action Triggered: Based on the event type, your system performs specific actions, such as updating your database, notifying your users, or triggering other processes.
  • Response to JemeniPay: Your webhook endpoint should return an HTTP status code (e.g., 200 OK) to confirm receipt of the notification. If JemeniPay does not receive a successful response, it retries sending the webhook a few times.

Key Features:

  • Security: JemeniPay uses signature validation to ensure that the webhook data comes from a trusted source.
  • Reliability: If your server does not respond, JemeniPay retries the notification multiple times.
  • Customizability: You can configure which events trigger webhook notifications to avoid unnecessary traffic.

Example Webhook Data:


{
    "event": "payment.success",
    "data": {
        "transaction_id": "TX12345",
        "amount": 1000,
        "currency": "XOF",
        "status": "success",
        "customer_id": "CUST6789"
    },
    "timestamp": "2024-12-30T12:34:56Z",
    "signature": "abcdef1234567890"
}

        

You can use this data to update your system in real-time, providing a seamless experience for your users.

Status: In Progress

We are continuously improving and adding new features to the JemeniPay webhook system. Stay tuned for updates!

HTTP Codes

The Jɛmɛnipay API uses the following error codes:

HTTP Code Descriptions for API Documentation

2xx Success

  • 200 OK: The request was successful, and the response contains the requested data.
  • 201 Created: The request was successful, and a new resource was created.
  • 202 Accepted: The request has been accepted for processing but is not yet complete.
  • 204 No Content: The request was successful, but there is no content to return.

3xx Redirection

  • 301 Moved Permanently: The resource has been permanently moved to a new URL.
  • 302 Found: The resource is temporarily located at a different URL.
  • 304 Not Modified: The resource has not been modified since the last request.

4xx Client Errors

  • 400 Bad Request: The server could not understand the request due to invalid syntax.
  • 401 Unauthorized: Authentication is required, or it has failed.
  • 403 Forbidden: The server understands the request but refuses to authorize it.
  • 404 Not Found: The requested resource could not be found.
  • 429 Too Many Requests: The client has sent too many requests in a given amount of time (rate limiting).

5xx Server Errors

  • 500 Internal Server Error: The server encountered an unexpected condition.
  • 501 Not Implemented: The server does not support the functionality required to fulfill the request.
  • 502 Bad Gateway: The server received an invalid response from the upstream server.
  • 503 Service Unavailable: The server is currently unable to handle the request due to maintenance or overload.
  • 504 Gateway Timeout: The server did not receive a timely response from the upstream server.

Custom Error Ranges

3000-3999 (Custom Redirection Codes)

Reserved for API-specific redirection behaviors.

4000-4999 (Custom Client Errors)

  • 4001 Invalid Input Data: The input data provided is invalid.
  • 4002 Resource Constraint Exceeded: A request exceeded a defined limit.

5000-5999 (Custom Server Errors)

  • 5001 Database Connection Failed: The server could not connect to the database.
  • 5002 Service Dependency Timeout: A dependent service did not respond in time.