(core) API reworked to use POST to create webhook and DELET to remove it
Summary:
introduces POST /api/docs/{docId}/webhooks and DELETE /api/docs/{docId}/webhooks/{webhookId} on place of old _subscribe and _unsubscribe endpoints.
Remove checking for unsubscribeKey while deleting webhook - only owner can delete webhook using DELETE endpoint. subscription key is still needed for _unsubscribe endpoint.
old _unsubscribe and _subscribe endpoints are still active and work as before - no changes there.
Posting schema:
```
POST /api/docs/[docId]/webhooks
```
Request Body:
```
{
"webhooks": [
{
"fields": {
"url": "https://webhook.site/3bd02246-f122-445e-ba7f-bf5ea5bb6eb1",
"eventTypes": [
"add",
"update"
],
"enabled": true,
"name": "WebhookName",
"memo": "just a text",
"tableId": "Table1"
}
},
{
"fields": {
"url": "https://webhook.site/3bd02246-f122-445e-ba7f-bf5ea5bb6eb2",
"eventTypes": [
"add",
],
"enabled": true,
"name": "OtherWebhookName",
"memo": "just a text",
"tableId": "Table1"
}
}
]
}
```
Expected response: WebhookId for each webhook posted:
```
{
"webhooks": [
{
"id": "85c77108-f1e1-4217-a50d-acd1c5996da2"
},
{
"id": "d87a6402-cfd7-4822-878c-657308fcc8c3"
}
]
}
```
Deleting webhooks:
```
DELETE api/docs/[docId]/webhooks/[webhookId]
```
there is no payload in DELETE request. Therefore only one webhook can be deleted at once
Response:
```
{
"success": true
}
```
Test Plan: Old unit test improved to handle new endpoints, and one more added to check if endpoints are in fact created/removed
Reviewers: alexmojaki
Reviewed By: alexmojaki
Subscribers: paulfitz, alexmojaki
Differential Revision: https://phab.getgrist.com/D3916
2023-07-14 10:05:22 +00:00
|
|
|
export interface WebhookSubscribeCollection{
|
|
|
|
webhooks: Array<Webhook>
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface Webhook {
|
|
|
|
fields: WebhookFields;
|
|
|
|
}
|
|
|
|
|
2023-03-01 20:43:22 +00:00
|
|
|
export interface WebhookFields {
|
|
|
|
url: string;
|
|
|
|
eventTypes: Array<"add"|"update">;
|
|
|
|
tableId: string;
|
2024-04-12 20:04:37 +00:00
|
|
|
watchedColIds?: string[];
|
2023-03-01 20:43:22 +00:00
|
|
|
enabled?: boolean;
|
|
|
|
isReadyColumn?: string|null;
|
2023-05-08 22:06:24 +00:00
|
|
|
name?: string;
|
|
|
|
memo?: string;
|
2023-03-01 20:43:22 +00:00
|
|
|
}
|
|
|
|
|
2023-05-08 22:06:24 +00:00
|
|
|
// Union discriminated by type
|
|
|
|
export type WebhookBatchStatus = 'success'|'failure'|'rejected';
|
|
|
|
export type WebhookStatus = 'idle'|'sending'|'retrying'|'postponed'|'error'|'invalid';
|
|
|
|
|
|
|
|
|
2023-03-01 20:43:22 +00:00
|
|
|
// WebhookSubscribe should be `Omit<WebhookFields, 'tableId'>` (because subscribe endpoint read
|
|
|
|
// tableId from the url) but generics are not yet supported by ts-interface-builder
|
|
|
|
export interface WebhookSubscribe {
|
|
|
|
url: string;
|
|
|
|
eventTypes: Array<"add"|"update">;
|
2024-04-12 20:04:37 +00:00
|
|
|
watchedColIds?: string[];
|
2023-03-01 20:43:22 +00:00
|
|
|
enabled?: boolean;
|
|
|
|
isReadyColumn?: string|null;
|
2023-05-08 22:06:24 +00:00
|
|
|
name?: string;
|
|
|
|
memo?: string;
|
|
|
|
}
|
|
|
|
|
(core) API reworked to use POST to create webhook and DELET to remove it
Summary:
introduces POST /api/docs/{docId}/webhooks and DELETE /api/docs/{docId}/webhooks/{webhookId} on place of old _subscribe and _unsubscribe endpoints.
Remove checking for unsubscribeKey while deleting webhook - only owner can delete webhook using DELETE endpoint. subscription key is still needed for _unsubscribe endpoint.
old _unsubscribe and _subscribe endpoints are still active and work as before - no changes there.
Posting schema:
```
POST /api/docs/[docId]/webhooks
```
Request Body:
```
{
"webhooks": [
{
"fields": {
"url": "https://webhook.site/3bd02246-f122-445e-ba7f-bf5ea5bb6eb1",
"eventTypes": [
"add",
"update"
],
"enabled": true,
"name": "WebhookName",
"memo": "just a text",
"tableId": "Table1"
}
},
{
"fields": {
"url": "https://webhook.site/3bd02246-f122-445e-ba7f-bf5ea5bb6eb2",
"eventTypes": [
"add",
],
"enabled": true,
"name": "OtherWebhookName",
"memo": "just a text",
"tableId": "Table1"
}
}
]
}
```
Expected response: WebhookId for each webhook posted:
```
{
"webhooks": [
{
"id": "85c77108-f1e1-4217-a50d-acd1c5996da2"
},
{
"id": "d87a6402-cfd7-4822-878c-657308fcc8c3"
}
]
}
```
Deleting webhooks:
```
DELETE api/docs/[docId]/webhooks/[webhookId]
```
there is no payload in DELETE request. Therefore only one webhook can be deleted at once
Response:
```
{
"success": true
}
```
Test Plan: Old unit test improved to handle new endpoints, and one more added to check if endpoints are in fact created/removed
Reviewers: alexmojaki
Reviewed By: alexmojaki
Subscribers: paulfitz, alexmojaki
Differential Revision: https://phab.getgrist.com/D3916
2023-07-14 10:05:22 +00:00
|
|
|
|
(core) GET endpoint for webhooks returns now data in format {webhooks:[...]}
Summary:
Rework of endpoint GET for webhooks to make it coherent with other endpoints. Now data should be return in {webhooks:[{id:"...",fields:{"..."}]} format
```
{
"webhooks": [
{
"id": ...
"fields": {
"url": ...
"unsubscribeKey": ...
"eventTypes": [
"add",
"update"
],
"isReadyColumn": null,
"tableId": "...",
"enabled": false,
"name": "...",
"memo": "..."
},
"usage": {
"status": "idle",
"numWaiting": 0,
"lastEventBatch": null
}
},
{
"id": "...",
"fields": {
"url": "...",
"unsubscribeKey": "...",
"eventTypes": [
"add",
"update"
],
"isReadyColumn": null,
"tableId": "...",
"enabled": true,
"name": "...",
"memo": "..."
},
"usage": {
"status": "error",
"numWaiting": 0,
"updatedTime": 1689076978098,
"lastEventBatch": {
"status": "rejected",
"httpStatus": 404,
"errorMessage": "{\"success\":false,\"error\":{\"message\":\"Alias 5a9bf6a8-4865-403a-bec6-b4ko not found\",\"id\":null}}",
"size": 49,
"attempts": 5
},
"lastSuccessTime": null,
"lastFailureTime": 1689076978097,
"lastErrorMessage": "{\"success\":false,\"error\":{\"message\":\"Alias 5a9bf6a8-4865-403a-bec6-b4ko not found\",\"id\":null}}",
"lastHttpStatus": 404
}
}
]
}
```
Test Plan: new test added to check if GET data fromat is correct. Other tests fixed to handle changed endpoint.
Reviewers: paulfitz
Reviewed By: paulfitz
Differential Revision: https://phab.getgrist.com/D3966
2023-07-25 13:29:19 +00:00
|
|
|
export interface WebhookSummaryCollection {
|
|
|
|
webhooks: Array<WebhookSummary>;
|
|
|
|
}
|
2023-05-08 22:06:24 +00:00
|
|
|
export interface WebhookSummary {
|
|
|
|
id: string;
|
|
|
|
fields: {
|
|
|
|
url: string;
|
|
|
|
unsubscribeKey: string;
|
|
|
|
eventTypes: string[];
|
|
|
|
isReadyColumn: string|null;
|
|
|
|
tableId: string;
|
2024-04-12 20:04:37 +00:00
|
|
|
watchedColIds?: string[];
|
2023-05-08 22:06:24 +00:00
|
|
|
enabled: boolean;
|
|
|
|
name: string;
|
|
|
|
memo: string;
|
|
|
|
},
|
|
|
|
usage: WebhookUsage|null,
|
2023-03-01 20:43:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Describes fields to update a webhook
|
|
|
|
export interface WebhookUpdate {
|
|
|
|
id: string;
|
|
|
|
fields: WebhookPatch;
|
|
|
|
}
|
|
|
|
|
|
|
|
// WebhookPatch should be `Partial<WebhookFields>` but generics are not yet supported by
|
|
|
|
// ts-interface-builder
|
|
|
|
export interface WebhookPatch {
|
|
|
|
url?: string;
|
|
|
|
eventTypes?: Array<"add"|"update">;
|
|
|
|
tableId?: string;
|
2024-04-12 20:04:37 +00:00
|
|
|
watchedColIds?: string[];
|
2023-03-01 20:43:22 +00:00
|
|
|
enabled?: boolean;
|
|
|
|
isReadyColumn?: string|null;
|
2023-05-08 22:06:24 +00:00
|
|
|
name?: string;
|
|
|
|
memo?: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
export interface WebhookUsage {
|
|
|
|
// As minimum we need number of waiting events and status (by default pending).
|
|
|
|
numWaiting: number,
|
|
|
|
status: WebhookStatus;
|
|
|
|
updatedTime?: number|null;
|
|
|
|
lastSuccessTime?: number|null;
|
|
|
|
lastFailureTime?: number|null;
|
|
|
|
lastErrorMessage?: string|null;
|
|
|
|
lastHttpStatus?: number|null;
|
|
|
|
lastEventBatch?: null | {
|
|
|
|
size: number;
|
|
|
|
errorMessage: string|null;
|
|
|
|
httpStatus: number|null;
|
|
|
|
status: WebhookBatchStatus;
|
|
|
|
attempts: number;
|
|
|
|
},
|
|
|
|
numSuccess?: {
|
|
|
|
pastHour: number;
|
|
|
|
past24Hours: number;
|
|
|
|
},
|
2023-03-01 20:43:22 +00:00
|
|
|
}
|