mirror of
https://github.com/nostr-protocol/nips.git
synced 2024-12-22 16:35:52 -05:00
working
This commit is contained in:
parent
70ede5e67d
commit
b6f3e249cd
229
105.md
Normal file
229
105.md
Normal file
|
@ -0,0 +1,229 @@
|
||||||
|
NIP-105
|
||||||
|
======
|
||||||
|
|
||||||
|
API Service Marketplace
|
||||||
|
-------------------------------
|
||||||
|
|
||||||
|
`draft` `optional` `author:coachchuckff` `author:unclejim21` `author:cmdruid`
|
||||||
|
|
||||||
|
This NIP defines `kind:31402` (a _parameterized replaceable event_) for broadcasting API services, endpoints and their costs in mSats. API services are offered ala carte to be paid via lightning invoices.
|
||||||
|
|
||||||
|
API service providers will issue a `kind:31402` event as an API service offering. Clients can fetch `s` (service) tag offerings, call their endpoints, and pay per use. Creating an API service marketplace.
|
||||||
|
|
||||||
|
## Protocol flow
|
||||||
|
|
||||||
|
1. API Service provider creates and hosts API service offerings, they then issue a `kind:31402` event for each `s` ( service ) tag.
|
||||||
|
2. Client fetches API offerings by `s` tag.
|
||||||
|
3. Client chooses a provider(s) and POSTs to their `endpoint`. The POST body should match exactly what is required by the underlying API less the API key. ( Or following the note's `schema` )
|
||||||
|
4. The service provider should validate their request and return a lightning invoice with a `successAction` url. The `successAction` url should be formatted as such: `<endpoint>/<invoice_hash>/get_result`
|
||||||
|
5. The client pays.
|
||||||
|
6. The client will poll the `successAction` until the service provider has returned the result(s) matching exactly what the underlying API will produce ( Or following the note's `outputSchema` ).
|
||||||
|
|
||||||
|
## Server Functions
|
||||||
|
### Create Invoice
|
||||||
|
|
||||||
|
Initally, the client will make a POST request to the `endpoint`. The POST body should be exactly what the underlying API is expecting less the API key. This information should also be represented in the `schema` portion of the `content` field.
|
||||||
|
|
||||||
|
The service provider will then do the following:
|
||||||
|
|
||||||
|
1. Validate their API request
|
||||||
|
2. Issue a lightning invoice
|
||||||
|
|
||||||
|
The issued lightning invoice will contain a `successAction` url formatted as so:
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
{
|
||||||
|
tag: "url",
|
||||||
|
url: `https://example.api/chat/${paymentHash}/get_result`,
|
||||||
|
description: "Open to get the confirmation code for your purchase."
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
From here the service provider can either listen and execute their API request "on-payment" or wait until the client calls `get_results`.
|
||||||
|
|
||||||
|
### Get Results
|
||||||
|
|
||||||
|
Once the client has gotten the `successAction` url, they can issue a GET request. The service provider should respond in one of three ways:
|
||||||
|
|
||||||
|
1. 200 ( DONE ) - the service provider returns the results of the API as if it was coming from the product itself ( Errors included )
|
||||||
|
2. 202 ( WORKING ) - the service provider returns a status that indicates the results are not ready yet.
|
||||||
|
3. 402 ( NOT PAID YET ) - the service provider should respond with a 402 error if the client has not paid yet.
|
||||||
|
|
||||||
|
It is up to the Client to poll the `successAction` until a terminal result is reached.
|
||||||
|
|
||||||
|
## Event Reference and Examples
|
||||||
|
|
||||||
|
### Offering Event
|
||||||
|
|
||||||
|
`kind:31402`
|
||||||
|
|
||||||
|
`.content` should be a JSON stringified version of the following JSON:
|
||||||
|
```typescript
|
||||||
|
enum OfferingStatus {
|
||||||
|
up = 'UP',
|
||||||
|
down = 'DOWN',
|
||||||
|
closed = 'CLOSED'
|
||||||
|
}
|
||||||
|
|
||||||
|
interface OfferingContent = {
|
||||||
|
endpoint: string, // The POST endpoint you call to pay/fetch
|
||||||
|
status: OfferingStatus, // UP/DOWN/CLOSED
|
||||||
|
cost: number, // The cost per call in mSats
|
||||||
|
schema?: Object, // Reccomended - JSON schema for the POST body of the endpoint
|
||||||
|
outputSchema?: Object, // Reccomended - JSON schema for the response of the call
|
||||||
|
description?: string // Optional - Description for the end user
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
`.tag` MUST include the following:
|
||||||
|
|
||||||
|
- `s`, the service tag should simply be the underlying API endpoint of the service provided. For example if you are offering a ChatGPT service, you would set `s` = `https://api.openai.com/v1/chat/completions`. This way the service they are buying is implicit.
|
||||||
|
- `d`, following **parameterized replaceable events** in [NIP-01](https://github.com/nostr-protocol/nips/blob/master/01.md), this tag allows this event to be replaceable. Since there should only be one service per pubkey, the `d` tag should match the `s` tag.
|
||||||
|
|
||||||
|
### Example Service Event
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
|
||||||
|
const content = {
|
||||||
|
endpoint: "https://example.api/chat/",
|
||||||
|
status: "UP",
|
||||||
|
cost: 5000,
|
||||||
|
schema: {...},
|
||||||
|
outputSchema: {...},
|
||||||
|
description: "This will call the ChatGPT endpoint"
|
||||||
|
}
|
||||||
|
|
||||||
|
const event = {
|
||||||
|
"kind": 31402,
|
||||||
|
"created_at": 1675642635,
|
||||||
|
"content": JSON.stringify(content),
|
||||||
|
"tags": [
|
||||||
|
["s", "https://api.openai.com/v1/chat/completions"]
|
||||||
|
["d", "https://api.openai.com/v1/chat/completions"],
|
||||||
|
],
|
||||||
|
"pubkey": "<pubkey>",
|
||||||
|
"id": "<id>"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Example Schema
|
||||||
|
|
||||||
|
The following is for the `gpt-3.5-turbo` input schema:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"model": {
|
||||||
|
"type": "string",
|
||||||
|
"enum": ["gpt-3.5-turbo"]
|
||||||
|
},
|
||||||
|
"messages": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"role": {
|
||||||
|
"type": "string",
|
||||||
|
"enum": ["system", "user"]
|
||||||
|
},
|
||||||
|
"content": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": ["role", "content"]
|
||||||
|
},
|
||||||
|
"minItems": 1
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": ["model", "messages"]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
The following is for the `gpt-3.5-turbo` output schema:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"type": "object",
|
||||||
|
"required": ["id", "object", "created", "model", "choices", "usage"],
|
||||||
|
"properties": {
|
||||||
|
"id": {
|
||||||
|
"type": "string",
|
||||||
|
"pattern": "^chatcmpl-[a-zA-Z0-9-]+$"
|
||||||
|
},
|
||||||
|
"object": {
|
||||||
|
"type": "string",
|
||||||
|
"enum": ["chat.completion"]
|
||||||
|
},
|
||||||
|
"created": {
|
||||||
|
"type": "integer"
|
||||||
|
},
|
||||||
|
"model": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"choices": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "object",
|
||||||
|
"required": ["index", "message"],
|
||||||
|
"properties": {
|
||||||
|
"index": {
|
||||||
|
"type": "integer"
|
||||||
|
},
|
||||||
|
"message": {
|
||||||
|
"type": "object",
|
||||||
|
"required": ["role", "content"],
|
||||||
|
"properties": {
|
||||||
|
"role": {
|
||||||
|
"type": "string",
|
||||||
|
"enum": ["assistant"]
|
||||||
|
},
|
||||||
|
"content": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"finish_reason": {
|
||||||
|
"type": "string",
|
||||||
|
"enum": ["stop"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"usage": {
|
||||||
|
"type": "object",
|
||||||
|
"required": ["prompt_tokens", "completion_tokens", "total_tokens"],
|
||||||
|
"properties": {
|
||||||
|
"prompt_tokens": {
|
||||||
|
"type": "integer"
|
||||||
|
},
|
||||||
|
"completion_tokens": {
|
||||||
|
"type": "integer"
|
||||||
|
},
|
||||||
|
"total_tokens": {
|
||||||
|
"type": "integer"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
### Safey
|
||||||
|
|
||||||
|
It is not mandatory, but to raise the barrier to entry, clients should screen service provider's NIP-05 identifier. The domain used in their NIP-05, should be the same domain used for their endpoint.
|
||||||
|
|
||||||
|
Clients may wish to create a whitelist of trusted service providers once tested.
|
||||||
|
|
||||||
|
### Problems
|
||||||
|
|
||||||
|
- No data integrity - service providers can store/redistribute any data passed to them
|
||||||
|
- Service providers could take payment and never return the product
|
||||||
|
- Service providers are not gaurenteed to call the endpoint specified in the `s` tag
|
||||||
|
- No recourse for errored API fetches
|
||||||
|
- The `cost` field may not match the actual final price
|
||||||
|
- No proof of purchase
|
||||||
|
|
||||||
|
### Example Implementations
|
||||||
|
|
||||||
|
- [Server](https://github.com/Team-Pleb-TabConf-2023/nip-105-server)
|
||||||
|
- [Client](https://github.com/Team-Pleb-TabConf-2023/nip-105-client)
|
Loading…
Reference in New Issue
Block a user