1
0
mirror of https://github.com/laurent22/joplin.git synced 2024-12-21 09:38:01 +02:00
joplin/packages/tools/utils/discourse.ts
2024-11-09 12:44:12 +00:00

158 lines
4.4 KiB
TypeScript

import { substrWithEllipsis } from '@joplin/lib/string-utils';
import { msleep } from '@joplin/utils/time';
import fetch from 'node-fetch';
interface ApiConfig {
baseUrl: string;
key: string;
username: string;
newsCategoryId: number;
}
export enum HttpMethod {
GET = 'GET',
POST = 'POST',
DELETE = 'DELETE',
PUT = 'PUT',
PATCH = 'PATCH',
}
interface ForumTopPost {
id: number;
raw: string;
title: string;
}
interface ForumTopic {
id: number;
topic_id: string;
}
export const config: ApiConfig = {
baseUrl: 'https://discourse.joplinapp.org',
key: '',
username: '',
newsCategoryId: 9,
};
export const execApi = async (method: HttpMethod, path: string, body: Record<string, string | number> = null) => {
interface Request {
method: HttpMethod;
headers: Record<string, string>;
body?: string;
}
const headers: Record<string, string> = {
'Api-Key': config.key,
'Api-Username': config.username,
};
if (method !== HttpMethod.GET) headers['Content-Type'] = 'application/json;';
const request: Request = {
method,
headers,
};
if (body) request.body = JSON.stringify(body);
while (true) {
const response = await fetch(`${config.baseUrl}/${path}`, request);
if (!response.ok) {
const errorText = await response.text();
const error = new Error(`On ${method} ${path}: ${errorText}`);
let apiObject = null;
try {
apiObject = JSON.parse(errorText);
} catch (error) {
// Ignore - it just means that the error object is a plain string
}
// {
// "errors": [
// "You’ve performed this action too many times. Please wait 39 seconds before trying again."
// ],
// "error_type": "rate_limit",
// "extras": {
// "wait_seconds": 39,
// "time_left": "39 seconds"
// }
// }
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
(error as any).apiObject = apiObject;
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
(error as any).status = response.status;
if (apiObject?.extras?.wait_seconds) {
await msleep(apiObject.extras.wait_seconds * 1000 + 1000);
continue;
}
throw error;
}
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
return response.json() as any;
}
};
export const getForumTopPostByExternalId = async (externalId: string): Promise<ForumTopPost> => {
try {
const existingForumTopic = await execApi(HttpMethod.GET, `t/external_id/${externalId}.json`);
const existingForumPost = await execApi(HttpMethod.GET, `posts/${existingForumTopic.post_stream.posts[0].id}.json`);
return {
id: existingForumPost.id,
title: existingForumTopic.title,
raw: existingForumPost.raw,
};
} catch (error) {
if (error.status === 404) return null;
if (error.apiObject && error.apiObject.error_type === 'not_found') return null;
throw error;
}
};
export const getTopicByExternalId = async (externalId: string): Promise<ForumTopic> => {
try {
const existingForumTopic = await execApi(HttpMethod.GET, `t/external_id/${externalId}.json`);
return existingForumTopic;
} catch (error) {
if (error.status === 404) return null;
if (error.apiObject && error.apiObject.error_type === 'not_found') return null;
throw error;
}
};
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
export const createTopic = async (topic: any): Promise<ForumTopic> => {
return execApi(HttpMethod.POST, 'posts', topic);
};
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
export const createPost = async (topicId: number, post: any): Promise<ForumTopic> => {
return execApi(HttpMethod.POST, 'posts', {
topic_id: topicId,
...post,
});
};
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
export const updatePost = async (postId: number, content: any): Promise<void> => {
await execApi(HttpMethod.PUT, `posts/${postId}.json`, content);
};
export const trimPostToMaximumLength = (postBody: string) => {
// Discourse has a maximum post length of 65_000:
const maximumLength = 65_000;
if (postBody.length > maximumLength) {
return [
substrWithEllipsis(postBody, 0, maximumLength - 150),
'**Note**: The full content of this post is longer than permitted by Discourse.',
].join('\n\n');
}
return postBody;
};