@drodil/backstage-plugin-qeta-backend 3.55.0 → 3.55.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1 +1 @@
1
- {"version":3,"file":"posts.cjs.js","sources":["../../../src/service/routes/posts.ts"],"sourcesContent":["import { getCreated, mapAdditionalFields } from '../util';\nimport { durationToMilliseconds, HumanDuration } from '@backstage/types';\nimport Ajv from 'ajv';\nimport { Request, Router } from 'express';\nimport {\n findEntityMentions,\n PostsQuery,\n qetaCreateCommentPermission,\n qetaCreatePostPermission,\n qetaCreatePostReviewPermission,\n qetaDeleteCommentPermission,\n qetaDeletePostPermission,\n qetaEditCommentPermission,\n qetaEditPostPermission,\n qetaReadAnswerPermission,\n qetaReadCommentPermission,\n qetaReadPostPermission,\n qetaReadPostReviewPermission,\n qetaReadTagPermission,\n} from '@drodil/backstage-plugin-qeta-common';\nimport addFormats from 'ajv-formats';\nimport {\n CommentSchema,\n DeleteMetadataSchema,\n PostQuerySchema,\n PostReviewBodySchema,\n PostSchema,\n PostsQuerySchema,\n RouteOptions,\n URLMetadataSchema,\n} from '../types';\nimport { Response } from 'express-serve-static-core';\nimport {\n entityToJsonObject,\n extractMetadata,\n signalPostStats,\n validateDateRange,\n wrapAsync,\n} from './util';\nimport { getEntities, getTags } from './routeUtil';\nimport { PostOptions } from '../../database/QetaStore';\n\nconst ajv = new Ajv({ coerceTypes: 'array' });\naddFormats(ajv);\n\nexport const postsRoutes = (router: Router, options: RouteOptions) => {\n const {\n database,\n events,\n config,\n cache,\n signals,\n notificationMgr,\n auditor,\n permissionMgr,\n } = options;\n\n const postsOlderThan = config.getOptional<HumanDuration>(\n 'qeta.contentHealth.postsOlderThan',\n ) ?? { months: 6 };\n const reviewThresholdMs = durationToMilliseconds(postsOlderThan);\n\n const getPostFilters = async (request: Request, opts: PostOptions) => {\n return await Promise.all([\n permissionMgr.getAuthorizeConditions(request, qetaReadPostPermission, {\n allowServicePrincipal: true,\n }),\n opts.includeTags\n ? permissionMgr.getAuthorizeConditions(request, qetaReadTagPermission, {\n allowServicePrincipal: true,\n })\n : undefined,\n opts.includeComments\n ? permissionMgr.getAuthorizeConditions(\n request,\n qetaReadCommentPermission,\n { allowServicePrincipal: true },\n )\n : undefined,\n opts.includeAnswers\n ? permissionMgr.getAuthorizeConditions(\n request,\n qetaReadAnswerPermission,\n { allowServicePrincipal: true },\n )\n : undefined,\n ]);\n };\n\n const getPostAndCheckStatus = async (\n request: Request,\n response: Response,\n recordView?: boolean,\n allowServiceToken?: boolean,\n ) => {\n const username = await permissionMgr.getUsername(\n request,\n allowServiceToken,\n );\n const postId = Number.parseInt(request.params.id, 10);\n if (Number.isNaN(postId)) {\n response.status(400).send({ errors: 'Invalid post id', type: 'body' });\n return null;\n }\n\n const [tagsFilter, commentsFilter, answersFilter] = await Promise.all([\n permissionMgr.getAuthorizeConditions(request, qetaReadTagPermission, {\n allowServicePrincipal: allowServiceToken,\n }),\n permissionMgr.getAuthorizeConditions(request, qetaReadCommentPermission, {\n allowServicePrincipal: allowServiceToken,\n }),\n permissionMgr.getAuthorizeConditions(request, qetaReadAnswerPermission, {\n allowServicePrincipal: allowServiceToken,\n }),\n ]);\n\n const post = await database.getPost(username, postId, recordView, {\n tagsFilter,\n commentsFilter,\n answersFilter,\n includeHealth: true,\n reviewThresholdMs,\n });\n\n if (!post) {\n response.status(404).send({ errors: 'Post not found', type: 'query' });\n return null;\n }\n\n if (\n post.status === 'deleted' &&\n !(await permissionMgr.isModerator(request))\n ) {\n response.status(404).send({ errors: 'Post not found', type: 'query' });\n return null;\n }\n\n if (post.status === 'draft' && post.author !== username) {\n response.status(404).send({ errors: 'Post not found', type: 'query' });\n return null;\n }\n\n return {\n post,\n username,\n postId,\n tagsFilter,\n commentsFilter,\n answersFilter,\n };\n };\n\n // GET /posts\n router.get(`/posts`, async (request, response) => {\n // Validation\n const username = await permissionMgr.getUsername(request, true);\n const validateQuery = ajv.compile(PostsQuerySchema);\n if (!validateQuery(request.query)) {\n response\n .status(400)\n .send({ errors: validateQuery.errors, type: 'query' });\n return;\n }\n\n const validDate = validateDateRange(\n request.query.fromDate as string,\n request.query.toDate as string,\n );\n if (!validDate?.isValid) {\n response.status(400).send(validDate);\n return;\n }\n\n const opts = request.query as PostsQuery;\n\n if (opts.reviewNeeded) {\n opts.obsolete = false;\n opts.includeHealth = true;\n }\n\n const [filter, tagsFilter, commentsFilter, answersFilter] =\n await getPostFilters(request, opts);\n\n if (\n opts.status === 'deleted' &&\n !(await permissionMgr.isModerator(request))\n ) {\n response.status(404).send({ errors: 'Post not found', type: 'query' });\n return;\n }\n\n // Act\n const posts = await database.getPosts(username, opts, filter, {\n tagsFilter,\n commentsFilter,\n answersFilter,\n includeAnswers: false,\n includeComments: false,\n includeAttachments: false,\n includeExperts: false,\n includeHealth: opts.includeHealth,\n reviewThresholdMs,\n reviewNeeded: opts.reviewNeeded,\n });\n\n await mapAdditionalFields(request, posts.posts, options, {\n checkRights: opts.checkAccess ?? false,\n username,\n });\n response.json(posts);\n });\n\n router.post(`/posts/query`, async (request, response) => {\n // Validation\n const username = await permissionMgr.getUsername(request, true);\n const validateQuery = ajv.compile(PostsQuerySchema);\n if (!validateQuery(request.body)) {\n response.status(400).send({ errors: validateQuery.errors, type: 'body' });\n return;\n }\n\n const validDate = validateDateRange(\n request.body.fromDate as string,\n request.body.toDate as string,\n );\n if (!validDate?.isValid) {\n response.status(400).send(validDate);\n return;\n }\n\n const opts = request.body;\n\n if (\n opts.status === 'deleted' &&\n !(await permissionMgr.isModerator(request))\n ) {\n response.status(404).send({ errors: 'Post not found', type: 'query' });\n return;\n }\n\n const [filter, tagsFilter, commentsFilter, answersFilter] =\n await getPostFilters(request, opts);\n\n if (opts.reviewNeeded) {\n opts.obsolete = false;\n opts.includeHealth = true;\n }\n\n // Act\n const posts = await database.getPosts(username, opts, filter, {\n tagsFilter,\n commentsFilter,\n answersFilter,\n includeAnswers: false,\n includeComments: false,\n includeAttachments: false,\n includeExperts: false,\n includeHealth: opts.includeHealth,\n reviewThresholdMs,\n reviewNeeded: opts.reviewNeeded,\n });\n await mapAdditionalFields(request, posts.posts, options, {\n checkRights: opts.checkAccess ?? false,\n username,\n });\n response.json(posts);\n });\n\n // POST /posts/suggest\n router.post(`/posts/suggest`, async (request, response) => {\n const username = await permissionMgr.getUsername(request, true);\n if (!request.body.title) {\n response.status(400).send({ errors: 'Title is required', type: 'body' });\n return;\n }\n\n const { title, content, tags, entities } = request.body;\n\n const [filter, tagsFilter, commentsFilter, answersFilter] =\n await getPostFilters(request, {\n includeAnswers: true,\n includeComments: false,\n includeAttachments: false,\n includeExperts: false,\n });\n\n const posts = await database.suggestPosts(\n username,\n title,\n content,\n tags,\n entities,\n filter,\n { tagsFilter, commentsFilter, answersFilter },\n );\n\n await mapAdditionalFields(request, posts.posts, options, {\n checkRights: false,\n username,\n });\n response.json(posts);\n });\n\n // GET /posts/list/:type\n router.get(`/posts/list/:type`, async (request, response) => {\n // Validation\n const username = await permissionMgr.getUsername(request, true);\n const validateQuery = ajv.compile(PostsQuerySchema);\n if (!validateQuery(request.query)) {\n response\n .status(400)\n .send({ errors: validateQuery.errors, type: 'query' });\n return;\n }\n\n const optionOverride: PostsQuery = { status: 'active' };\n const type = request.params.type;\n if (type === 'unanswered') {\n optionOverride.random = true;\n optionOverride.noAnswers = true;\n } else if (type === 'incorrect') {\n optionOverride.noCorrectAnswer = true;\n optionOverride.random = true;\n } else if (type === 'hot') {\n optionOverride.includeTrend = true;\n optionOverride.orderBy = 'trend';\n } else if (type === 'own') {\n optionOverride.author = username;\n }\n const opts = { ...request.query, ...optionOverride };\n\n const [filter, tagsFilter, commentsFilter, answersFilter] =\n await getPostFilters(request, opts);\n\n // Act\n const posts = await database.getPosts(username, opts, filter, {\n includeTotal: false,\n includeAnswers: false,\n includeAttachments: false,\n includeEntities: false,\n includeTags: false,\n includeVotes: false,\n includeExperts: false,\n includeComments: false,\n commentsFilter,\n tagsFilter,\n answersFilter,\n });\n\n await mapAdditionalFields(request, posts.posts, options, {\n checkRights: opts.checkAccess ?? false,\n username,\n });\n response.json(posts);\n });\n\n // GET /posts/:id\n router.get(`/posts/:id`, async (request, response) => {\n const validateQuery = ajv.compile(PostQuerySchema);\n if (!validateQuery(request.query)) {\n response\n .status(400)\n .send({ errors: validateQuery.errors, type: 'query' });\n return;\n }\n const anonymous = request.query.anonymous as undefined | boolean;\n\n const ret = await getPostAndCheckStatus(\n request,\n response,\n anonymous ? !anonymous : true,\n true,\n );\n if (!ret) return;\n const { post, username } = ret;\n\n await permissionMgr.authorize(\n request,\n [{ permission: qetaReadPostPermission, resource: post }],\n { throwOnDeny: true },\n );\n\n await mapAdditionalFields(request, [post], options, { username });\n if (!anonymous) {\n signalPostStats(signals, post);\n }\n\n auditor?.createEvent({\n eventId: 'read-post',\n severityLevel: 'low',\n request,\n meta: { post: entityToJsonObject(post) },\n });\n\n response.json(post);\n });\n\n // POST /posts/:id/comments\n router.post(`/posts/:id/comments`, async (request, response) => {\n const ret = await getPostAndCheckStatus(request, response, false, true);\n if (!ret) return;\n const { post, username, answersFilter, tagsFilter, commentsFilter } = ret;\n\n const created = await getCreated(request, options);\n const validateRequestBody = ajv.compile(CommentSchema);\n if (!validateRequestBody(request.body)) {\n response\n .status(400)\n .send({ errors: validateRequestBody.errors, type: 'body' });\n return;\n }\n\n if (post.status === 'obsolete') {\n response.status(400).send({\n errors: 'Cannot add comments to obsolete posts',\n type: 'body',\n });\n return;\n }\n\n await permissionMgr.authorize(\n request,\n [\n { permission: qetaReadPostPermission, resource: post },\n { permission: qetaCreateCommentPermission },\n ],\n { throwOnDeny: true },\n );\n\n const updatedPost = await database.commentPost(\n post.id,\n username,\n request.body.content,\n created,\n { tagsFilter, commentsFilter, answersFilter },\n );\n\n if (updatedPost === null) {\n response\n .status(400)\n .send({ errors: 'Failed to comment post', type: 'body' });\n return;\n }\n\n await mapAdditionalFields(request, [updatedPost], options, { username });\n\n wrapAsync(async () => {\n if (!updatedPost || updatedPost.status !== 'active') {\n return;\n }\n const followingUsers = await Promise.all([\n database.getUsersForTags(updatedPost.tags),\n database.getUsersForEntities(updatedPost.entities),\n database.getFollowingUsers(username),\n database.getUsersWhoFavoritedPost(updatedPost.id),\n ]);\n\n const sent = await notificationMgr.onNewPostComment(\n username,\n updatedPost,\n request.body.content,\n followingUsers.flat(),\n );\n const mentions = findEntityMentions(request.body.content);\n if (mentions.length > 0) {\n await notificationMgr.onMention(\n username,\n updatedPost,\n mentions,\n sent,\n true,\n );\n }\n });\n\n events?.publish({\n topic: 'qeta',\n eventPayload: {\n post: updatedPost,\n comment: request.body.content,\n author: username,\n },\n metadata: { action: 'comment_post' },\n });\n\n auditor?.createEvent({\n eventId: 'comment-post',\n severityLevel: 'medium',\n request,\n meta: {\n post: entityToJsonObject(updatedPost),\n comment: request.body.content,\n },\n });\n\n response.status(201).json(updatedPost);\n });\n\n // POST /posts/:id/comments/:commentId\n router.post(`/posts/:id/comments/:commentId`, async (request, response) => {\n // Validation\n // Act\n const postId = Number.parseInt(request.params.id, 10);\n const commentId = Number.parseInt(request.params.commentId, 10);\n if (Number.isNaN(postId) || Number.isNaN(commentId)) {\n response.status(400).send({ errors: 'Invalid id', type: 'body' });\n return;\n }\n\n const ret = await getPostAndCheckStatus(request, response, false, true);\n if (!ret) return;\n const { post, username, answersFilter, commentsFilter, tagsFilter } = ret;\n\n if (post.status === 'obsolete') {\n response.status(400).send({\n errors: 'Cannot edit comments on obsolete posts',\n type: 'body',\n });\n return;\n }\n\n const comment = await database.getComment(commentId, { postId });\n\n if (!comment) {\n response.status(404).send({ errors: 'Comment not found', type: 'body' });\n return;\n }\n\n await permissionMgr.authorize(\n request,\n [{ permission: qetaEditCommentPermission, resource: comment }],\n { throwOnDeny: true },\n );\n\n const updatedPost = await database.updatePostComment(\n postId,\n commentId,\n username,\n request.body.content,\n {\n tagsFilter,\n commentsFilter,\n answersFilter,\n },\n );\n\n if (updatedPost === null) {\n response\n .status(400)\n .send({ errors: 'Failed to update post comment', type: 'body' });\n return;\n }\n\n auditor?.createEvent({\n eventId: 'update-comment',\n severityLevel: 'medium',\n request,\n meta: {\n post: entityToJsonObject(updatedPost),\n from: entityToJsonObject(comment),\n to: request.body.content,\n },\n });\n\n await mapAdditionalFields(request, [updatedPost], options, { username });\n\n // Response\n response.json(updatedPost);\n });\n\n // DELETE /posts/:id/comments/:commentId\n router.delete(`/posts/:id/comments/:commentId`, async (request, response) => {\n // Validation\n // Act\n const postId = Number.parseInt(request.params.id, 10);\n const commentId = Number.parseInt(request.params.commentId, 10);\n if (Number.isNaN(postId) || Number.isNaN(commentId)) {\n response.status(400).send({ errors: 'Invalid id', type: 'body' });\n return;\n }\n\n const ret = await getPostAndCheckStatus(request, response, false, true);\n if (!ret) return;\n const { post, username, answersFilter, tagsFilter, commentsFilter } = ret;\n\n if (post.status === 'obsolete') {\n response.status(400).send({\n errors: 'Cannot delete comments on obsolete posts',\n type: 'body',\n });\n return;\n }\n\n const comment = await database.getComment(commentId, { postId });\n\n if (!comment) {\n response.status(404).send({ errors: 'Comment not found', type: 'body' });\n return;\n }\n\n await permissionMgr.authorize(\n request,\n [{ permission: qetaDeleteCommentPermission, resource: comment }],\n { throwOnDeny: true },\n );\n\n let updatedPost = null;\n if (comment.status === 'deleted' || request.body?.permanent === true) {\n if (!(await permissionMgr.isModerator(request))) {\n response\n .status(404)\n .send({ errors: 'Comment not found', type: 'query' });\n return;\n }\n updatedPost = await database.deletePostComment(\n postId,\n commentId,\n username,\n true,\n {\n tagsFilter,\n commentsFilter,\n answersFilter,\n },\n );\n } else {\n updatedPost = await database.deletePostComment(\n postId,\n commentId,\n username,\n false,\n {\n tagsFilter,\n commentsFilter,\n answersFilter,\n },\n );\n }\n\n if (updatedPost === null) {\n response\n .status(400)\n .send({ errors: 'Failed to delete post comment', type: 'body' });\n return;\n }\n\n auditor?.createEvent({\n eventId: 'delete-comment',\n severityLevel: 'medium',\n request,\n meta: {\n post: entityToJsonObject(updatedPost),\n comment: entityToJsonObject(comment),\n },\n });\n\n await mapAdditionalFields(request, [updatedPost], options, { username });\n\n // Response\n response.json(updatedPost);\n });\n\n // POST /posts\n router.post(`/posts`, async (request, response) => {\n // Validation\n const validateRequestBody = ajv.compile(PostSchema);\n if (!validateRequestBody(request.body)) {\n response\n .status(400)\n .json({ errors: validateRequestBody.errors, type: 'body' });\n return;\n }\n await permissionMgr.authorize(\n request,\n [{ permission: qetaCreatePostPermission }],\n { throwOnDeny: true },\n );\n\n const existingTags = await database.getTags();\n const [tags, entities, username, created] = await Promise.all([\n getTags(request, options, existingTags),\n getEntities(request, config),\n permissionMgr.getUsername(request),\n getCreated(request, options),\n ]);\n\n if (request.body.author && request.body.author !== username) {\n if (!(await permissionMgr.isModerator(request))) {\n response\n .status(400)\n .json({ errors: validateRequestBody.errors, type: 'body' });\n return;\n }\n }\n\n // Act\n const post = await database.createPost({\n user_ref: username,\n title: request.body.title,\n content: request.body.content,\n author: request.body.author,\n created,\n tags,\n entities,\n images: request.body.images,\n anonymous: request.body.anonymous || username === 'user:default/guest',\n type: request.body.type,\n headerImage: request.body.headerImage,\n url: request.body.url,\n status: request.body.status || 'active',\n opts: {\n includeComments: false,\n includeVotes: false,\n includeAnswers: false,\n },\n });\n\n if (!post) {\n response.status(400).send({ errors: 'Failed to post', type: 'body' });\n return;\n }\n\n wrapAsync(async () => {\n if (!post || post.status !== 'active') {\n return;\n }\n const followingUsers = await Promise.all([\n database.getUsersForTags(tags),\n database.getUsersForEntities(entities),\n database.getFollowingUsers(username),\n database.getUsersWhoFavoritedPost(post.id),\n ]);\n const sent = await notificationMgr.onNewPost(\n username,\n post,\n followingUsers.flat(),\n );\n const mentions = findEntityMentions(request.body.content);\n if (mentions.length > 0) {\n await notificationMgr.onMention(username, post, mentions, sent);\n }\n });\n\n events?.publish({\n topic: 'qeta',\n eventPayload: {\n post,\n author: username,\n },\n metadata: { action: 'new_post' },\n });\n\n auditor?.createEvent({\n eventId: 'create-post',\n severityLevel: 'medium',\n request,\n meta: {\n post: entityToJsonObject(post),\n },\n });\n\n await mapAdditionalFields(request, [post], options, { username });\n\n // Response\n response.status(201).json(post);\n });\n\n // POST /posts/:id\n router.post(`/posts/:id`, async (request, response) => {\n // Validation\n\n const validateRequestBody = ajv.compile(PostSchema);\n if (!validateRequestBody(request.body)) {\n response\n .status(400)\n .json({ errors: validateRequestBody.errors, type: 'body' });\n return;\n }\n\n const ret = await getPostAndCheckStatus(request, response, false, true);\n if (!ret) return;\n const {\n post: originalPost,\n postId,\n username,\n answersFilter,\n tagsFilter,\n commentsFilter,\n } = ret;\n\n await permissionMgr.authorize(\n request,\n [{ permission: qetaEditPostPermission, resource: originalPost }],\n { throwOnDeny: true },\n );\n\n if (request.body.status !== 'active' && originalPost.status === 'active') {\n if (!(await permissionMgr.isModerator(request))) {\n response\n .status(400)\n .json({ errors: validateRequestBody.errors, type: 'body' });\n return;\n }\n }\n\n if (\n request.body.author &&\n request.body.author !== username &&\n request.body.author !== originalPost.author\n ) {\n if (!(await permissionMgr.isModerator(request))) {\n response\n .status(400)\n .json({ errors: validateRequestBody.errors, type: 'body' });\n return;\n }\n }\n\n const existingTags = await database.getTags();\n const [tags, entities] = await Promise.all([\n getTags(request, options, existingTags),\n getEntities(request, config),\n ]);\n\n // Act\n const post = await database.updatePost({\n id: postId,\n user_ref: username,\n author: request.body.author,\n title: request.body.title,\n content: request.body.content,\n tags,\n entities,\n images: request.body.images,\n headerImage: request.body.headerImage,\n url: request.body.url,\n status: request.body.status || 'active',\n setUpdatedBy: originalPost.status !== 'draft',\n opts: { tagsFilter, commentsFilter, answersFilter },\n });\n\n if (!post) {\n response.sendStatus(401);\n return;\n }\n\n wrapAsync(async () => {\n if (!post || post.status !== 'active') {\n return;\n }\n const newTags = tags.filter(t => !originalPost.tags?.includes(t));\n const newEntities = entities.filter(\n e => !originalPost.entities?.includes(e),\n );\n\n const followingUsers = await Promise.all([\n database.getUsersForTags(newTags),\n database.getUsersForEntities(newEntities),\n ]);\n\n const sent = await notificationMgr.onPostEdit(\n username,\n post,\n followingUsers.flat(),\n );\n const originalMentions = findEntityMentions(originalPost.content);\n const mentions = findEntityMentions(request.body.content);\n const newMentions = mentions.filter(m => !originalMentions.includes(m));\n\n if (newMentions.length > 0) {\n await notificationMgr.onMention(username, post, newMentions, sent);\n }\n });\n\n events?.publish({\n topic: 'qeta',\n eventPayload: {\n post,\n author: username,\n },\n metadata: { action: 'update_post' },\n });\n\n await mapAdditionalFields(request, [post], options, { username });\n\n auditor?.createEvent({\n eventId: 'update-post',\n severityLevel: 'medium',\n request,\n meta: {\n from: entityToJsonObject(originalPost),\n to: entityToJsonObject(post),\n },\n });\n\n // Response\n response.json(post);\n });\n\n // DELETE /posts/:id\n router.delete('/posts/:id', async (request, response) => {\n const ret = await getPostAndCheckStatus(request, response, false, true);\n if (!ret) return;\n const validateRequestBody = ajv.compile(DeleteMetadataSchema);\n if (!validateRequestBody(request.body)) {\n response.status(400).send({ errors: 'Invalid data', type: 'body' });\n return;\n }\n\n const { post, username } = ret;\n\n await permissionMgr.authorize(\n request,\n [{ permission: qetaDeletePostPermission, resource: post }],\n { throwOnDeny: true },\n );\n\n let deleted = false;\n if (post.status === 'deleted' || request.body?.permanent === true) {\n if (!(await permissionMgr.isModerator(request))) {\n response.status(404).send({ errors: 'Post not found', type: 'query' });\n return;\n }\n deleted = await database.deletePost(post.id, true);\n } else {\n deleted = await database.deletePost(post.id);\n if (deleted) {\n notificationMgr.onPostDelete(username, post, request.body.reason);\n }\n }\n\n if (deleted) {\n events?.publish({\n topic: 'qeta',\n eventPayload: {\n post,\n author: post.author,\n reason: request.body.reason,\n },\n metadata: { action: 'delete_post' },\n });\n\n auditor?.createEvent({\n eventId: 'delete-post',\n severityLevel: 'medium',\n request,\n meta: { post: entityToJsonObject(post), reason: request.body.reason },\n });\n }\n\n response.sendStatus(deleted ? 204 : 404);\n });\n\n const votePost = async (\n request: Request<any>,\n response: Response,\n score: number,\n ) => {\n const ret = await getPostAndCheckStatus(request, response);\n if (!ret) return;\n const {\n post,\n postId,\n username,\n answersFilter,\n tagsFilter,\n commentsFilter,\n } = ret;\n\n await permissionMgr.authorize(\n request,\n [{ permission: qetaReadPostPermission, resource: post }],\n { throwOnDeny: true },\n );\n if (post.own) {\n response\n .status(400)\n .send({ errors: 'You cannot vote your own post', type: 'body' });\n return;\n }\n\n const voted = await database.votePost(username, postId, score);\n\n if (!voted) {\n response.status(404).send({ errors: 'Post not found', type: 'body' });\n return;\n }\n\n const resp = await database.getPost(username, postId, false, {\n answersFilter,\n tagsFilter,\n commentsFilter,\n includeComments: false,\n includeAnswers: false,\n includeAttachments: false,\n includeTags: false,\n includeEntities: false,\n });\n if (resp === null) {\n response.sendStatus(404);\n return;\n }\n\n events?.publish({\n topic: 'qeta',\n eventPayload: {\n resp,\n author: username,\n score,\n },\n metadata: { action: 'vote_post' },\n });\n\n await mapAdditionalFields(request, [resp], options, { username });\n resp.ownVote = score;\n\n auditor?.createEvent({\n eventId: 'vote',\n severityLevel: 'low',\n request,\n meta: { post: entityToJsonObject(post), score },\n });\n\n signalPostStats(signals, resp);\n\n // Response\n response.json(resp);\n };\n\n // POST /posts/:id/restore\n router.post('/posts/:id/restore', async (request, response) => {\n const ret = await getPostAndCheckStatus(request, response, false, true);\n if (!ret) return;\n\n if (!(await permissionMgr.isModerator(request))) {\n response.status(404).send({ errors: 'Post not found', type: 'query' });\n return;\n }\n const {\n post: originalPost,\n postId,\n username,\n answersFilter,\n tagsFilter,\n commentsFilter,\n } = ret;\n\n // Act\n const post = await database.updatePost({\n id: postId,\n status: 'active',\n user_ref: username,\n setUpdatedBy: false,\n opts: { tagsFilter, commentsFilter, answersFilter },\n });\n\n if (!post) {\n response.sendStatus(401);\n return;\n }\n\n events?.publish({\n topic: 'qeta',\n eventPayload: {\n post,\n author: username,\n },\n metadata: { action: 'restore_post' },\n });\n\n await mapAdditionalFields(request, [post], options, { username });\n\n auditor?.createEvent({\n eventId: 'restore-post',\n severityLevel: 'medium',\n request,\n meta: {\n from: entityToJsonObject(originalPost),\n to: entityToJsonObject(post),\n },\n });\n\n // Response\n response.json(post);\n });\n\n // PUT /posts/:id/click\n router.put('/posts/:id/click', async (request, response) => {\n const ret = await getPostAndCheckStatus(request, response);\n if (!ret) {\n response.status(404).send({ errors: 'Post not found', type: 'query' });\n return;\n }\n const { postId, username } = ret;\n await database.clickPost(username, postId);\n\n const resp = await database.getPost(username, postId, false, {\n includeComments: false,\n includeAnswers: false,\n includeAttachments: false,\n includeTags: false,\n includeEntities: false,\n });\n\n if (!resp) {\n response.status(404).send({ errors: 'Post not found', type: 'query' });\n return;\n }\n\n signalPostStats(signals, resp);\n response.status(200).send({});\n });\n\n // GET /posts/:id/upvote\n router.get(`/posts/:id/upvote`, async (request, response) => {\n return await votePost(request, response, 1);\n });\n\n // GET /posts/:id/downvote\n router.get(`/posts/:id/downvote`, async (request, response) => {\n return await votePost(request, response, -1);\n });\n\n router.delete('/posts/:id/vote', async (request, response) => {\n // Validation\n const ret = await getPostAndCheckStatus(request, response);\n if (!ret) return;\n const {\n post,\n postId,\n username,\n answersFilter,\n tagsFilter,\n commentsFilter,\n } = ret;\n\n await permissionMgr.authorize(\n request,\n [{ permission: qetaReadPostPermission, resource: post }],\n { throwOnDeny: true },\n );\n\n const deleted = await database.deletePostVote(username, postId);\n if (!deleted) {\n response.sendStatus(404);\n return;\n }\n\n const resp = await database.getPost(username, postId, false, {\n answersFilter,\n tagsFilter,\n commentsFilter,\n includeComments: false,\n includeAnswers: false,\n includeAttachments: false,\n includeTags: false,\n includeEntities: false,\n });\n\n if (resp === null) {\n response.sendStatus(404);\n return;\n }\n\n events?.publish({\n topic: 'qeta',\n eventPayload: {\n post,\n author: username,\n },\n metadata: { action: 'delete_vote' },\n });\n\n await mapAdditionalFields(request, [resp], options, { username });\n resp.ownVote = undefined;\n\n signalPostStats(signals, resp);\n\n auditor?.createEvent({\n eventId: 'delete-vote',\n severityLevel: 'low',\n request,\n meta: { post: entityToJsonObject(post) },\n });\n response.json(resp);\n });\n\n // POST /posts/:id/review\n router.post('/posts/:id/review', async (request, response) => {\n const validateRequestBody = ajv.compile(PostReviewBodySchema);\n if (!validateRequestBody(request.body)) {\n response\n .status(400)\n .json({ errors: validateRequestBody.errors, type: 'body' });\n return;\n }\n\n const ret = await getPostAndCheckStatus(request, response);\n if (!ret) return;\n const { post, postId } = ret;\n\n const user = await options.permissionMgr.getUsername(request, true);\n\n await options.permissionMgr.authorize(\n request,\n [\n { permission: qetaReadPostPermission, resource: post },\n { permission: qetaCreatePostReviewPermission },\n ],\n { throwOnDeny: true },\n );\n\n const updatedPost = await options.database.reviewPost(\n user,\n postId,\n request.body.status,\n request.body.comment,\n );\n if (!updatedPost) {\n response.sendStatus(404);\n return;\n }\n response.json(updatedPost);\n });\n\n // GET /posts/:id/reviews\n router.get('/posts/:id/reviews', async (request, response) => {\n const ret = await getPostAndCheckStatus(request, response);\n if (!ret) return;\n const { post, postId } = ret;\n\n await options.permissionMgr.authorize(\n request,\n [\n { permission: qetaReadPostPermission, resource: post },\n { permission: qetaReadPostReviewPermission },\n ],\n { throwOnDeny: true },\n );\n\n const reviews = await options.database.getPostReviews(postId);\n response.json(reviews);\n });\n\n // GET /posts/:id/favorite\n router.get(`/posts/:id/favorite`, async (request, response) => {\n const ret = await getPostAndCheckStatus(request, response);\n if (!ret) return;\n const {\n post,\n postId,\n username,\n answersFilter,\n tagsFilter,\n commentsFilter,\n } = ret;\n\n await permissionMgr.authorize(\n request,\n [{ permission: qetaReadPostPermission, resource: post }],\n { throwOnDeny: true },\n );\n\n const favorited = await database.favoritePost(username, postId);\n\n if (!favorited) {\n response.sendStatus(404);\n return;\n }\n\n const updatedPost = await database.getPost(\n username,\n Number.parseInt(request.params.id, 10),\n false,\n {\n answersFilter,\n tagsFilter,\n commentsFilter,\n includeComments: false,\n includeAnswers: false,\n includeEntities: false,\n includeVotes: false,\n includeTags: false,\n includeAttachments: false,\n },\n );\n\n if (!updatedPost) {\n response.sendStatus(404);\n return;\n }\n\n auditor?.createEvent({\n eventId: 'favorite-post',\n severityLevel: 'low',\n request,\n meta: { post: entityToJsonObject(updatedPost) },\n });\n\n await mapAdditionalFields(request, [updatedPost], options, { username });\n\n // Response\n response.json(updatedPost);\n });\n\n // GET /posts/:id/unfavorite\n router.get(`/posts/:id/unfavorite`, async (request, response) => {\n const ret = await getPostAndCheckStatus(request, response);\n if (!ret) return;\n const {\n post,\n postId,\n username,\n answersFilter,\n tagsFilter,\n commentsFilter,\n } = ret;\n\n await permissionMgr.authorize(\n request,\n [{ permission: qetaReadPostPermission, resource: post }],\n { throwOnDeny: true },\n );\n\n const unfavorited = await database.unfavoritePost(username, postId);\n\n if (!unfavorited) {\n response.sendStatus(404);\n return;\n }\n\n const updatedPost = await database.getPost(\n username,\n Number.parseInt(request.params.id, 10),\n false,\n {\n answersFilter,\n tagsFilter,\n commentsFilter,\n includeComments: false,\n includeAnswers: false,\n includeEntities: false,\n includeVotes: false,\n includeTags: false,\n includeAttachments: false,\n },\n );\n\n if (!updatedPost) {\n response.sendStatus(404);\n return;\n }\n\n auditor?.createEvent({\n eventId: 'unfavorite-post',\n severityLevel: 'low',\n request,\n meta: { post: entityToJsonObject(updatedPost) },\n });\n\n await mapAdditionalFields(request, [updatedPost], options, { username });\n\n // Response\n response.json(updatedPost);\n });\n\n // POST /url\n router.post(`/url`, async (request, response) => {\n const validateQuery = ajv.compile(URLMetadataSchema);\n if (!validateQuery(request.body)) {\n response.status(400).send({ errors: validateQuery.errors, type: 'body' });\n return;\n }\n\n const url = new URL(request.body.url);\n if (url.protocol !== 'http:' && url.protocol !== 'https:') {\n response\n .status(400)\n .send({ errors: 'Invalid URL protocol', type: 'url' });\n return;\n }\n if (url.hostname === 'localhost') {\n response\n .status(400)\n .send({ errors: 'localhost not allowed', type: 'url' });\n return;\n }\n\n const cacheKey = `url:metadata:${url.toString()}`;\n const cached = await cache?.get(cacheKey);\n if (cached) {\n response.json(cached);\n return;\n }\n\n const metadata = await extractMetadata(url, options.logger);\n await cache?.set(cacheKey, metadata, { ttl: { weeks: 2 } });\n response.json(metadata);\n });\n};\n"],"names":["Ajv","addFormats","durationToMilliseconds","qetaReadPostPermission","qetaReadTagPermission","qetaReadCommentPermission","qetaReadAnswerPermission","PostsQuerySchema","validateDateRange","mapAdditionalFields","PostQuerySchema","signalPostStats","entityToJsonObject","getCreated","CommentSchema","qetaCreateCommentPermission","wrapAsync","findEntityMentions","qetaEditCommentPermission","qetaDeleteCommentPermission","PostSchema","qetaCreatePostPermission","getTags","getEntities","qetaEditPostPermission","DeleteMetadataSchema","qetaDeletePostPermission","PostReviewBodySchema","qetaCreatePostReviewPermission","qetaReadPostReviewPermission","URLMetadataSchema","extractMetadata"],"mappings":"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"}
1
+ {"version":3,"file":"posts.cjs.js","sources":["../../../src/service/routes/posts.ts"],"sourcesContent":["import { getCreated, mapAdditionalFields } from '../util';\nimport { durationToMilliseconds, HumanDuration } from '@backstage/types';\nimport Ajv from 'ajv';\nimport { Request, Router } from 'express';\nimport {\n findEntityMentions,\n PostsQuery,\n qetaCreateCommentPermission,\n qetaCreatePostPermission,\n qetaCreatePostReviewPermission,\n qetaDeleteCommentPermission,\n qetaDeletePostPermission,\n qetaEditCommentPermission,\n qetaEditPostPermission,\n qetaReadAnswerPermission,\n qetaReadCommentPermission,\n qetaReadPostPermission,\n qetaReadPostReviewPermission,\n qetaReadTagPermission,\n} from '@drodil/backstage-plugin-qeta-common';\nimport addFormats from 'ajv-formats';\nimport {\n CommentSchema,\n DeleteMetadataSchema,\n PostQuerySchema,\n PostReviewBodySchema,\n PostSchema,\n PostsQuerySchema,\n RouteOptions,\n URLMetadataSchema,\n} from '../types';\nimport { Response } from 'express-serve-static-core';\nimport {\n entityToJsonObject,\n extractMetadata,\n signalPostStats,\n validateDateRange,\n wrapAsync,\n} from './util';\nimport { getCachedData, getEntities, getTags } from './routeUtil';\nimport { PostOptions } from '../../database/QetaStore';\n\nconst ajv = new Ajv({ coerceTypes: 'array' });\naddFormats(ajv);\n\nexport const postsRoutes = (router: Router, options: RouteOptions) => {\n const {\n database,\n events,\n config,\n cache,\n signals,\n notificationMgr,\n auditor,\n permissionMgr,\n } = options;\n\n const postsOlderThan = config.getOptional<HumanDuration>(\n 'qeta.contentHealth.postsOlderThan',\n ) ?? { months: 6 };\n const reviewThresholdMs = durationToMilliseconds(postsOlderThan);\n\n const getPostFilters = async (request: Request, opts: PostOptions) => {\n return await Promise.all([\n permissionMgr.getAuthorizeConditions(request, qetaReadPostPermission, {\n allowServicePrincipal: true,\n }),\n opts.includeTags\n ? permissionMgr.getAuthorizeConditions(request, qetaReadTagPermission, {\n allowServicePrincipal: true,\n })\n : undefined,\n opts.includeComments\n ? permissionMgr.getAuthorizeConditions(\n request,\n qetaReadCommentPermission,\n { allowServicePrincipal: true },\n )\n : undefined,\n opts.includeAnswers\n ? permissionMgr.getAuthorizeConditions(\n request,\n qetaReadAnswerPermission,\n { allowServicePrincipal: true },\n )\n : undefined,\n ]);\n };\n\n const getPostAndCheckStatus = async (\n request: Request,\n response: Response,\n recordView?: boolean,\n allowServiceToken?: boolean,\n ) => {\n const username = await permissionMgr.getUsername(\n request,\n allowServiceToken,\n );\n const postId = Number.parseInt(request.params.id, 10);\n if (Number.isNaN(postId)) {\n response.status(400).send({ errors: 'Invalid post id', type: 'body' });\n return null;\n }\n\n const [tagsFilter, commentsFilter, answersFilter] = await Promise.all([\n permissionMgr.getAuthorizeConditions(request, qetaReadTagPermission, {\n allowServicePrincipal: allowServiceToken,\n }),\n permissionMgr.getAuthorizeConditions(request, qetaReadCommentPermission, {\n allowServicePrincipal: allowServiceToken,\n }),\n permissionMgr.getAuthorizeConditions(request, qetaReadAnswerPermission, {\n allowServicePrincipal: allowServiceToken,\n }),\n ]);\n\n const post = await database.getPost(username, postId, recordView, {\n tagsFilter,\n commentsFilter,\n answersFilter,\n includeHealth: true,\n reviewThresholdMs,\n });\n\n if (!post) {\n response.status(404).send({ errors: 'Post not found', type: 'query' });\n return null;\n }\n\n if (\n post.status === 'deleted' &&\n !(await permissionMgr.isModerator(request))\n ) {\n response.status(404).send({ errors: 'Post not found', type: 'query' });\n return null;\n }\n\n if (post.status === 'draft' && post.author !== username) {\n response.status(404).send({ errors: 'Post not found', type: 'query' });\n return null;\n }\n\n return {\n post,\n username,\n postId,\n tagsFilter,\n commentsFilter,\n answersFilter,\n };\n };\n\n // GET /posts\n router.get(`/posts`, async (request, response) => {\n // Validation\n const username = await permissionMgr.getUsername(request, true);\n const validateQuery = ajv.compile(PostsQuerySchema);\n if (!validateQuery(request.query)) {\n response\n .status(400)\n .send({ errors: validateQuery.errors, type: 'query' });\n return;\n }\n\n const validDate = validateDateRange(\n request.query.fromDate as string,\n request.query.toDate as string,\n );\n if (!validDate?.isValid) {\n response.status(400).send(validDate);\n return;\n }\n\n const opts = request.query as PostsQuery;\n\n if (opts.reviewNeeded) {\n opts.obsolete = false;\n opts.includeHealth = true;\n }\n\n const [filter, tagsFilter, commentsFilter, answersFilter] =\n await getPostFilters(request, opts);\n\n if (\n opts.status === 'deleted' &&\n !(await permissionMgr.isModerator(request))\n ) {\n response.status(404).send({ errors: 'Post not found', type: 'query' });\n return;\n }\n\n // Act\n const posts = await database.getPosts(username, opts, filter, {\n tagsFilter,\n commentsFilter,\n answersFilter,\n includeAnswers: false,\n includeComments: false,\n includeAttachments: false,\n includeExperts: false,\n includeHealth: opts.includeHealth,\n reviewThresholdMs,\n reviewNeeded: opts.reviewNeeded,\n });\n\n await mapAdditionalFields(request, posts.posts, options, {\n checkRights: opts.checkAccess ?? false,\n username,\n });\n response.json(posts);\n });\n\n router.post(`/posts/query`, async (request, response) => {\n // Validation\n const username = await permissionMgr.getUsername(request, true);\n const validateQuery = ajv.compile(PostsQuerySchema);\n if (!validateQuery(request.body)) {\n response.status(400).send({ errors: validateQuery.errors, type: 'body' });\n return;\n }\n\n const validDate = validateDateRange(\n request.body.fromDate as string,\n request.body.toDate as string,\n );\n if (!validDate?.isValid) {\n response.status(400).send(validDate);\n return;\n }\n\n const opts = request.body;\n\n if (\n opts.status === 'deleted' &&\n !(await permissionMgr.isModerator(request))\n ) {\n response.status(404).send({ errors: 'Post not found', type: 'query' });\n return;\n }\n\n const [filter, tagsFilter, commentsFilter, answersFilter] =\n await getPostFilters(request, opts);\n\n if (opts.reviewNeeded) {\n opts.obsolete = false;\n opts.includeHealth = true;\n }\n\n // Act\n const posts = await database.getPosts(username, opts, filter, {\n tagsFilter,\n commentsFilter,\n answersFilter,\n includeAnswers: false,\n includeComments: false,\n includeAttachments: false,\n includeExperts: false,\n includeHealth: opts.includeHealth,\n reviewThresholdMs,\n reviewNeeded: opts.reviewNeeded,\n });\n await mapAdditionalFields(request, posts.posts, options, {\n checkRights: opts.checkAccess ?? false,\n username,\n });\n response.json(posts);\n });\n\n // POST /posts/suggest\n router.post(`/posts/suggest`, async (request, response) => {\n const username = await permissionMgr.getUsername(request, true);\n if (!request.body.title) {\n response.status(400).send({ errors: 'Title is required', type: 'body' });\n return;\n }\n\n const { title, content, tags, entities } = request.body;\n\n const [filter, tagsFilter, commentsFilter, answersFilter] =\n await getPostFilters(request, {\n includeAnswers: true,\n includeComments: false,\n includeAttachments: false,\n includeExperts: false,\n });\n\n const posts = await database.suggestPosts(\n username,\n title,\n content,\n tags,\n entities,\n filter,\n { tagsFilter, commentsFilter, answersFilter },\n );\n\n await mapAdditionalFields(request, posts.posts, options, {\n checkRights: false,\n username,\n });\n response.json(posts);\n });\n\n // GET /posts/list/:type\n router.get(`/posts/list/:type`, async (request, response) => {\n // Validation\n const username = await permissionMgr.getUsername(request, true);\n const validateQuery = ajv.compile(PostsQuerySchema);\n if (!validateQuery(request.query)) {\n response\n .status(400)\n .send({ errors: validateQuery.errors, type: 'query' });\n return;\n }\n\n const optionOverride: PostsQuery = { status: 'active' };\n const type = request.params.type;\n if (type === 'unanswered') {\n optionOverride.random = true;\n optionOverride.noAnswers = true;\n } else if (type === 'incorrect') {\n optionOverride.noCorrectAnswer = true;\n optionOverride.random = true;\n } else if (type === 'hot') {\n optionOverride.includeTrend = true;\n optionOverride.orderBy = 'trend';\n } else if (type === 'own') {\n optionOverride.author = username;\n }\n const opts = { ...request.query, ...optionOverride };\n\n const [filter, tagsFilter, commentsFilter, answersFilter] =\n await getPostFilters(request, opts);\n\n // Act\n const key = `qeta:posts:list:${type}:${username}:${JSON.stringify(opts)}`;\n const ttl = 300 * 1000;\n\n const posts = await getCachedData(\n options.cache,\n key,\n ttl,\n async () => {\n return await database.getPosts(username, opts, filter, {\n includeTotal: false,\n includeAnswers: false,\n includeAttachments: false,\n includeEntities: false,\n includeTags: false,\n includeVotes: false,\n includeExperts: false,\n includeComments: false,\n commentsFilter,\n tagsFilter,\n answersFilter,\n });\n },\n options.logger,\n );\n\n await mapAdditionalFields(request, posts.posts, options, {\n checkRights: opts.checkAccess ?? false,\n username,\n });\n response.json(posts);\n });\n\n // GET /posts/:id\n router.get(`/posts/:id`, async (request, response) => {\n const validateQuery = ajv.compile(PostQuerySchema);\n if (!validateQuery(request.query)) {\n response\n .status(400)\n .send({ errors: validateQuery.errors, type: 'query' });\n return;\n }\n const anonymous = request.query.anonymous as undefined | boolean;\n\n const ret = await getPostAndCheckStatus(\n request,\n response,\n anonymous ? !anonymous : true,\n true,\n );\n if (!ret) return;\n const { post, username } = ret;\n\n await permissionMgr.authorize(\n request,\n [{ permission: qetaReadPostPermission, resource: post }],\n { throwOnDeny: true },\n );\n\n await mapAdditionalFields(request, [post], options, { username });\n if (!anonymous) {\n signalPostStats(signals, post);\n }\n\n auditor?.createEvent({\n eventId: 'read-post',\n severityLevel: 'low',\n request,\n meta: { post: entityToJsonObject(post) },\n });\n\n response.json(post);\n });\n\n // POST /posts/:id/comments\n router.post(`/posts/:id/comments`, async (request, response) => {\n const ret = await getPostAndCheckStatus(request, response, false, true);\n if (!ret) return;\n const { post, username, answersFilter, tagsFilter, commentsFilter } = ret;\n\n const created = await getCreated(request, options);\n const validateRequestBody = ajv.compile(CommentSchema);\n if (!validateRequestBody(request.body)) {\n response\n .status(400)\n .send({ errors: validateRequestBody.errors, type: 'body' });\n return;\n }\n\n if (post.status === 'obsolete') {\n response.status(400).send({\n errors: 'Cannot add comments to obsolete posts',\n type: 'body',\n });\n return;\n }\n\n await permissionMgr.authorize(\n request,\n [\n { permission: qetaReadPostPermission, resource: post },\n { permission: qetaCreateCommentPermission },\n ],\n { throwOnDeny: true },\n );\n\n const updatedPost = await database.commentPost(\n post.id,\n username,\n request.body.content,\n created,\n { tagsFilter, commentsFilter, answersFilter },\n );\n\n if (updatedPost === null) {\n response\n .status(400)\n .send({ errors: 'Failed to comment post', type: 'body' });\n return;\n }\n\n await mapAdditionalFields(request, [updatedPost], options, { username });\n\n wrapAsync(async () => {\n if (!updatedPost || updatedPost.status !== 'active') {\n return;\n }\n const followingUsers = await Promise.all([\n database.getUsersForTags(updatedPost.tags),\n database.getUsersForEntities(updatedPost.entities),\n database.getFollowingUsers(username),\n database.getUsersWhoFavoritedPost(updatedPost.id),\n ]);\n\n const sent = await notificationMgr.onNewPostComment(\n username,\n updatedPost,\n request.body.content,\n followingUsers.flat(),\n );\n const mentions = findEntityMentions(request.body.content);\n if (mentions.length > 0) {\n await notificationMgr.onMention(\n username,\n updatedPost,\n mentions,\n sent,\n true,\n );\n }\n });\n\n events?.publish({\n topic: 'qeta',\n eventPayload: {\n post: updatedPost,\n comment: request.body.content,\n author: username,\n },\n metadata: { action: 'comment_post' },\n });\n\n auditor?.createEvent({\n eventId: 'comment-post',\n severityLevel: 'medium',\n request,\n meta: {\n post: entityToJsonObject(updatedPost),\n comment: request.body.content,\n },\n });\n\n response.status(201).json(updatedPost);\n });\n\n // POST /posts/:id/comments/:commentId\n router.post(`/posts/:id/comments/:commentId`, async (request, response) => {\n // Validation\n // Act\n const postId = Number.parseInt(request.params.id, 10);\n const commentId = Number.parseInt(request.params.commentId, 10);\n if (Number.isNaN(postId) || Number.isNaN(commentId)) {\n response.status(400).send({ errors: 'Invalid id', type: 'body' });\n return;\n }\n\n const ret = await getPostAndCheckStatus(request, response, false, true);\n if (!ret) return;\n const { post, username, answersFilter, commentsFilter, tagsFilter } = ret;\n\n if (post.status === 'obsolete') {\n response.status(400).send({\n errors: 'Cannot edit comments on obsolete posts',\n type: 'body',\n });\n return;\n }\n\n const comment = await database.getComment(commentId, { postId });\n\n if (!comment) {\n response.status(404).send({ errors: 'Comment not found', type: 'body' });\n return;\n }\n\n await permissionMgr.authorize(\n request,\n [{ permission: qetaEditCommentPermission, resource: comment }],\n { throwOnDeny: true },\n );\n\n const updatedPost = await database.updatePostComment(\n postId,\n commentId,\n username,\n request.body.content,\n {\n tagsFilter,\n commentsFilter,\n answersFilter,\n },\n );\n\n if (updatedPost === null) {\n response\n .status(400)\n .send({ errors: 'Failed to update post comment', type: 'body' });\n return;\n }\n\n auditor?.createEvent({\n eventId: 'update-comment',\n severityLevel: 'medium',\n request,\n meta: {\n post: entityToJsonObject(updatedPost),\n from: entityToJsonObject(comment),\n to: request.body.content,\n },\n });\n\n await mapAdditionalFields(request, [updatedPost], options, { username });\n\n // Response\n response.json(updatedPost);\n });\n\n // DELETE /posts/:id/comments/:commentId\n router.delete(`/posts/:id/comments/:commentId`, async (request, response) => {\n // Validation\n // Act\n const postId = Number.parseInt(request.params.id, 10);\n const commentId = Number.parseInt(request.params.commentId, 10);\n if (Number.isNaN(postId) || Number.isNaN(commentId)) {\n response.status(400).send({ errors: 'Invalid id', type: 'body' });\n return;\n }\n\n const ret = await getPostAndCheckStatus(request, response, false, true);\n if (!ret) return;\n const { post, username, answersFilter, tagsFilter, commentsFilter } = ret;\n\n if (post.status === 'obsolete') {\n response.status(400).send({\n errors: 'Cannot delete comments on obsolete posts',\n type: 'body',\n });\n return;\n }\n\n const comment = await database.getComment(commentId, { postId });\n\n if (!comment) {\n response.status(404).send({ errors: 'Comment not found', type: 'body' });\n return;\n }\n\n await permissionMgr.authorize(\n request,\n [{ permission: qetaDeleteCommentPermission, resource: comment }],\n { throwOnDeny: true },\n );\n\n let updatedPost = null;\n if (comment.status === 'deleted' || request.body?.permanent === true) {\n if (!(await permissionMgr.isModerator(request))) {\n response\n .status(404)\n .send({ errors: 'Comment not found', type: 'query' });\n return;\n }\n updatedPost = await database.deletePostComment(\n postId,\n commentId,\n username,\n true,\n {\n tagsFilter,\n commentsFilter,\n answersFilter,\n },\n );\n } else {\n updatedPost = await database.deletePostComment(\n postId,\n commentId,\n username,\n false,\n {\n tagsFilter,\n commentsFilter,\n answersFilter,\n },\n );\n }\n\n if (updatedPost === null) {\n response\n .status(400)\n .send({ errors: 'Failed to delete post comment', type: 'body' });\n return;\n }\n\n auditor?.createEvent({\n eventId: 'delete-comment',\n severityLevel: 'medium',\n request,\n meta: {\n post: entityToJsonObject(updatedPost),\n comment: entityToJsonObject(comment),\n },\n });\n\n await mapAdditionalFields(request, [updatedPost], options, { username });\n\n // Response\n response.json(updatedPost);\n });\n\n // POST /posts\n router.post(`/posts`, async (request, response) => {\n // Validation\n const validateRequestBody = ajv.compile(PostSchema);\n if (!validateRequestBody(request.body)) {\n response\n .status(400)\n .json({ errors: validateRequestBody.errors, type: 'body' });\n return;\n }\n await permissionMgr.authorize(\n request,\n [{ permission: qetaCreatePostPermission }],\n { throwOnDeny: true },\n );\n\n const existingTags = await database.getTags();\n const [tags, entities, username, created] = await Promise.all([\n getTags(request, options, existingTags),\n getEntities(request, config),\n permissionMgr.getUsername(request),\n getCreated(request, options),\n ]);\n\n if (request.body.author && request.body.author !== username) {\n if (!(await permissionMgr.isModerator(request))) {\n response\n .status(400)\n .json({ errors: validateRequestBody.errors, type: 'body' });\n return;\n }\n }\n\n // Act\n const post = await database.createPost({\n user_ref: username,\n title: request.body.title,\n content: request.body.content,\n author: request.body.author,\n created,\n tags,\n entities,\n images: request.body.images,\n anonymous: request.body.anonymous || username === 'user:default/guest',\n type: request.body.type,\n headerImage: request.body.headerImage,\n url: request.body.url,\n status: request.body.status || 'active',\n opts: {\n includeComments: false,\n includeVotes: false,\n includeAnswers: false,\n },\n });\n\n if (!post) {\n response.status(400).send({ errors: 'Failed to post', type: 'body' });\n return;\n }\n\n wrapAsync(async () => {\n if (!post || post.status !== 'active') {\n return;\n }\n const followingUsers = await Promise.all([\n database.getUsersForTags(tags),\n database.getUsersForEntities(entities),\n database.getFollowingUsers(username),\n database.getUsersWhoFavoritedPost(post.id),\n ]);\n const sent = await notificationMgr.onNewPost(\n username,\n post,\n followingUsers.flat(),\n );\n const mentions = findEntityMentions(request.body.content);\n if (mentions.length > 0) {\n await notificationMgr.onMention(username, post, mentions, sent);\n }\n });\n\n events?.publish({\n topic: 'qeta',\n eventPayload: {\n post,\n author: username,\n },\n metadata: { action: 'new_post' },\n });\n\n auditor?.createEvent({\n eventId: 'create-post',\n severityLevel: 'medium',\n request,\n meta: {\n post: entityToJsonObject(post),\n },\n });\n\n await mapAdditionalFields(request, [post], options, { username });\n\n // Response\n response.status(201).json(post);\n });\n\n // POST /posts/:id\n router.post(`/posts/:id`, async (request, response) => {\n // Validation\n\n const validateRequestBody = ajv.compile(PostSchema);\n if (!validateRequestBody(request.body)) {\n response\n .status(400)\n .json({ errors: validateRequestBody.errors, type: 'body' });\n return;\n }\n\n const ret = await getPostAndCheckStatus(request, response, false, true);\n if (!ret) return;\n const {\n post: originalPost,\n postId,\n username,\n answersFilter,\n tagsFilter,\n commentsFilter,\n } = ret;\n\n await permissionMgr.authorize(\n request,\n [{ permission: qetaEditPostPermission, resource: originalPost }],\n { throwOnDeny: true },\n );\n\n if (request.body.status !== 'active' && originalPost.status === 'active') {\n if (!(await permissionMgr.isModerator(request))) {\n response\n .status(400)\n .json({ errors: validateRequestBody.errors, type: 'body' });\n return;\n }\n }\n\n if (\n request.body.author &&\n request.body.author !== username &&\n request.body.author !== originalPost.author\n ) {\n if (!(await permissionMgr.isModerator(request))) {\n response\n .status(400)\n .json({ errors: validateRequestBody.errors, type: 'body' });\n return;\n }\n }\n\n const existingTags = await database.getTags();\n const [tags, entities] = await Promise.all([\n getTags(request, options, existingTags),\n getEntities(request, config),\n ]);\n\n // Act\n const post = await database.updatePost({\n id: postId,\n user_ref: username,\n author: request.body.author,\n title: request.body.title,\n content: request.body.content,\n tags,\n entities,\n images: request.body.images,\n headerImage: request.body.headerImage,\n url: request.body.url,\n status: request.body.status || 'active',\n setUpdatedBy: originalPost.status !== 'draft',\n opts: { tagsFilter, commentsFilter, answersFilter },\n });\n\n if (!post) {\n response.sendStatus(401);\n return;\n }\n\n wrapAsync(async () => {\n if (!post || post.status !== 'active') {\n return;\n }\n const newTags = tags.filter(t => !originalPost.tags?.includes(t));\n const newEntities = entities.filter(\n e => !originalPost.entities?.includes(e),\n );\n\n const followingUsers = await Promise.all([\n database.getUsersForTags(newTags),\n database.getUsersForEntities(newEntities),\n ]);\n\n const sent = await notificationMgr.onPostEdit(\n username,\n post,\n followingUsers.flat(),\n );\n const originalMentions = findEntityMentions(originalPost.content);\n const mentions = findEntityMentions(request.body.content);\n const newMentions = mentions.filter(m => !originalMentions.includes(m));\n\n if (newMentions.length > 0) {\n await notificationMgr.onMention(username, post, newMentions, sent);\n }\n });\n\n events?.publish({\n topic: 'qeta',\n eventPayload: {\n post,\n author: username,\n },\n metadata: { action: 'update_post' },\n });\n\n await mapAdditionalFields(request, [post], options, { username });\n\n auditor?.createEvent({\n eventId: 'update-post',\n severityLevel: 'medium',\n request,\n meta: {\n from: entityToJsonObject(originalPost),\n to: entityToJsonObject(post),\n },\n });\n\n // Response\n response.json(post);\n });\n\n // DELETE /posts/:id\n router.delete('/posts/:id', async (request, response) => {\n const ret = await getPostAndCheckStatus(request, response, false, true);\n if (!ret) return;\n const validateRequestBody = ajv.compile(DeleteMetadataSchema);\n if (!validateRequestBody(request.body)) {\n response.status(400).send({ errors: 'Invalid data', type: 'body' });\n return;\n }\n\n const { post, username } = ret;\n\n await permissionMgr.authorize(\n request,\n [{ permission: qetaDeletePostPermission, resource: post }],\n { throwOnDeny: true },\n );\n\n let deleted = false;\n if (post.status === 'deleted' || request.body?.permanent === true) {\n if (!(await permissionMgr.isModerator(request))) {\n response.status(404).send({ errors: 'Post not found', type: 'query' });\n return;\n }\n deleted = await database.deletePost(post.id, true);\n } else {\n deleted = await database.deletePost(post.id);\n if (deleted) {\n notificationMgr.onPostDelete(username, post, request.body.reason);\n }\n }\n\n if (deleted) {\n events?.publish({\n topic: 'qeta',\n eventPayload: {\n post,\n author: post.author,\n reason: request.body.reason,\n },\n metadata: { action: 'delete_post' },\n });\n\n auditor?.createEvent({\n eventId: 'delete-post',\n severityLevel: 'medium',\n request,\n meta: { post: entityToJsonObject(post), reason: request.body.reason },\n });\n }\n\n response.sendStatus(deleted ? 204 : 404);\n });\n\n const votePost = async (\n request: Request<any>,\n response: Response,\n score: number,\n ) => {\n const ret = await getPostAndCheckStatus(request, response);\n if (!ret) return;\n const {\n post,\n postId,\n username,\n answersFilter,\n tagsFilter,\n commentsFilter,\n } = ret;\n\n await permissionMgr.authorize(\n request,\n [{ permission: qetaReadPostPermission, resource: post }],\n { throwOnDeny: true },\n );\n if (post.own) {\n response\n .status(400)\n .send({ errors: 'You cannot vote your own post', type: 'body' });\n return;\n }\n\n const voted = await database.votePost(username, postId, score);\n\n if (!voted) {\n response.status(404).send({ errors: 'Post not found', type: 'body' });\n return;\n }\n\n const resp = await database.getPost(username, postId, false, {\n answersFilter,\n tagsFilter,\n commentsFilter,\n includeComments: false,\n includeAnswers: false,\n includeAttachments: false,\n includeTags: false,\n includeEntities: false,\n });\n if (resp === null) {\n response.sendStatus(404);\n return;\n }\n\n events?.publish({\n topic: 'qeta',\n eventPayload: {\n resp,\n author: username,\n score,\n },\n metadata: { action: 'vote_post' },\n });\n\n await mapAdditionalFields(request, [resp], options, { username });\n resp.ownVote = score;\n\n auditor?.createEvent({\n eventId: 'vote',\n severityLevel: 'low',\n request,\n meta: { post: entityToJsonObject(post), score },\n });\n\n signalPostStats(signals, resp);\n\n // Response\n response.json(resp);\n };\n\n // POST /posts/:id/restore\n router.post('/posts/:id/restore', async (request, response) => {\n const ret = await getPostAndCheckStatus(request, response, false, true);\n if (!ret) return;\n\n if (!(await permissionMgr.isModerator(request))) {\n response.status(404).send({ errors: 'Post not found', type: 'query' });\n return;\n }\n const {\n post: originalPost,\n postId,\n username,\n answersFilter,\n tagsFilter,\n commentsFilter,\n } = ret;\n\n // Act\n const post = await database.updatePost({\n id: postId,\n status: 'active',\n user_ref: username,\n setUpdatedBy: false,\n opts: { tagsFilter, commentsFilter, answersFilter },\n });\n\n if (!post) {\n response.sendStatus(401);\n return;\n }\n\n events?.publish({\n topic: 'qeta',\n eventPayload: {\n post,\n author: username,\n },\n metadata: { action: 'restore_post' },\n });\n\n await mapAdditionalFields(request, [post], options, { username });\n\n auditor?.createEvent({\n eventId: 'restore-post',\n severityLevel: 'medium',\n request,\n meta: {\n from: entityToJsonObject(originalPost),\n to: entityToJsonObject(post),\n },\n });\n\n // Response\n response.json(post);\n });\n\n // PUT /posts/:id/click\n router.put('/posts/:id/click', async (request, response) => {\n const ret = await getPostAndCheckStatus(request, response);\n if (!ret) {\n response.status(404).send({ errors: 'Post not found', type: 'query' });\n return;\n }\n const { postId, username } = ret;\n await database.clickPost(username, postId);\n\n const resp = await database.getPost(username, postId, false, {\n includeComments: false,\n includeAnswers: false,\n includeAttachments: false,\n includeTags: false,\n includeEntities: false,\n });\n\n if (!resp) {\n response.status(404).send({ errors: 'Post not found', type: 'query' });\n return;\n }\n\n signalPostStats(signals, resp);\n response.status(200).send({});\n });\n\n // GET /posts/:id/upvote\n router.get(`/posts/:id/upvote`, async (request, response) => {\n return await votePost(request, response, 1);\n });\n\n // GET /posts/:id/downvote\n router.get(`/posts/:id/downvote`, async (request, response) => {\n return await votePost(request, response, -1);\n });\n\n router.delete('/posts/:id/vote', async (request, response) => {\n // Validation\n const ret = await getPostAndCheckStatus(request, response);\n if (!ret) return;\n const {\n post,\n postId,\n username,\n answersFilter,\n tagsFilter,\n commentsFilter,\n } = ret;\n\n await permissionMgr.authorize(\n request,\n [{ permission: qetaReadPostPermission, resource: post }],\n { throwOnDeny: true },\n );\n\n const deleted = await database.deletePostVote(username, postId);\n if (!deleted) {\n response.sendStatus(404);\n return;\n }\n\n const resp = await database.getPost(username, postId, false, {\n answersFilter,\n tagsFilter,\n commentsFilter,\n includeComments: false,\n includeAnswers: false,\n includeAttachments: false,\n includeTags: false,\n includeEntities: false,\n });\n\n if (resp === null) {\n response.sendStatus(404);\n return;\n }\n\n events?.publish({\n topic: 'qeta',\n eventPayload: {\n post,\n author: username,\n },\n metadata: { action: 'delete_vote' },\n });\n\n await mapAdditionalFields(request, [resp], options, { username });\n resp.ownVote = undefined;\n\n signalPostStats(signals, resp);\n\n auditor?.createEvent({\n eventId: 'delete-vote',\n severityLevel: 'low',\n request,\n meta: { post: entityToJsonObject(post) },\n });\n response.json(resp);\n });\n\n // POST /posts/:id/review\n router.post('/posts/:id/review', async (request, response) => {\n const validateRequestBody = ajv.compile(PostReviewBodySchema);\n if (!validateRequestBody(request.body)) {\n response\n .status(400)\n .json({ errors: validateRequestBody.errors, type: 'body' });\n return;\n }\n\n const ret = await getPostAndCheckStatus(request, response);\n if (!ret) return;\n const { post, postId } = ret;\n\n const user = await options.permissionMgr.getUsername(request, true);\n\n await options.permissionMgr.authorize(\n request,\n [\n { permission: qetaReadPostPermission, resource: post },\n { permission: qetaCreatePostReviewPermission },\n ],\n { throwOnDeny: true },\n );\n\n const updatedPost = await options.database.reviewPost(\n user,\n postId,\n request.body.status,\n request.body.comment,\n );\n if (!updatedPost) {\n response.sendStatus(404);\n return;\n }\n response.json(updatedPost);\n });\n\n // GET /posts/:id/reviews\n router.get('/posts/:id/reviews', async (request, response) => {\n const ret = await getPostAndCheckStatus(request, response);\n if (!ret) return;\n const { post, postId } = ret;\n\n await options.permissionMgr.authorize(\n request,\n [\n { permission: qetaReadPostPermission, resource: post },\n { permission: qetaReadPostReviewPermission },\n ],\n { throwOnDeny: true },\n );\n\n const reviews = await options.database.getPostReviews(postId);\n response.json(reviews);\n });\n\n // GET /posts/:id/favorite\n router.get(`/posts/:id/favorite`, async (request, response) => {\n const ret = await getPostAndCheckStatus(request, response);\n if (!ret) return;\n const {\n post,\n postId,\n username,\n answersFilter,\n tagsFilter,\n commentsFilter,\n } = ret;\n\n await permissionMgr.authorize(\n request,\n [{ permission: qetaReadPostPermission, resource: post }],\n { throwOnDeny: true },\n );\n\n const favorited = await database.favoritePost(username, postId);\n\n if (!favorited) {\n response.sendStatus(404);\n return;\n }\n\n const updatedPost = await database.getPost(\n username,\n Number.parseInt(request.params.id, 10),\n false,\n {\n answersFilter,\n tagsFilter,\n commentsFilter,\n includeComments: false,\n includeAnswers: false,\n includeEntities: false,\n includeVotes: false,\n includeTags: false,\n includeAttachments: false,\n },\n );\n\n if (!updatedPost) {\n response.sendStatus(404);\n return;\n }\n\n auditor?.createEvent({\n eventId: 'favorite-post',\n severityLevel: 'low',\n request,\n meta: { post: entityToJsonObject(updatedPost) },\n });\n\n await mapAdditionalFields(request, [updatedPost], options, { username });\n\n // Response\n response.json(updatedPost);\n });\n\n // GET /posts/:id/unfavorite\n router.get(`/posts/:id/unfavorite`, async (request, response) => {\n const ret = await getPostAndCheckStatus(request, response);\n if (!ret) return;\n const {\n post,\n postId,\n username,\n answersFilter,\n tagsFilter,\n commentsFilter,\n } = ret;\n\n await permissionMgr.authorize(\n request,\n [{ permission: qetaReadPostPermission, resource: post }],\n { throwOnDeny: true },\n );\n\n const unfavorited = await database.unfavoritePost(username, postId);\n\n if (!unfavorited) {\n response.sendStatus(404);\n return;\n }\n\n const updatedPost = await database.getPost(\n username,\n Number.parseInt(request.params.id, 10),\n false,\n {\n answersFilter,\n tagsFilter,\n commentsFilter,\n includeComments: false,\n includeAnswers: false,\n includeEntities: false,\n includeVotes: false,\n includeTags: false,\n includeAttachments: false,\n },\n );\n\n if (!updatedPost) {\n response.sendStatus(404);\n return;\n }\n\n auditor?.createEvent({\n eventId: 'unfavorite-post',\n severityLevel: 'low',\n request,\n meta: { post: entityToJsonObject(updatedPost) },\n });\n\n await mapAdditionalFields(request, [updatedPost], options, { username });\n\n // Response\n response.json(updatedPost);\n });\n\n // POST /url\n router.post(`/url`, async (request, response) => {\n const validateQuery = ajv.compile(URLMetadataSchema);\n if (!validateQuery(request.body)) {\n response.status(400).send({ errors: validateQuery.errors, type: 'body' });\n return;\n }\n\n const url = new URL(request.body.url);\n if (url.protocol !== 'http:' && url.protocol !== 'https:') {\n response\n .status(400)\n .send({ errors: 'Invalid URL protocol', type: 'url' });\n return;\n }\n if (url.hostname === 'localhost') {\n response\n .status(400)\n .send({ errors: 'localhost not allowed', type: 'url' });\n return;\n }\n\n const cacheKey = `url:metadata:${url.toString()}`;\n const cached = await cache?.get(cacheKey);\n if (cached) {\n response.json(cached);\n return;\n }\n\n const metadata = await extractMetadata(url, options.logger);\n await cache?.set(cacheKey, metadata, { ttl: { weeks: 2 } });\n response.json(metadata);\n });\n};\n"],"names":["Ajv","addFormats","durationToMilliseconds","qetaReadPostPermission","qetaReadTagPermission","qetaReadCommentPermission","qetaReadAnswerPermission","PostsQuerySchema","validateDateRange","mapAdditionalFields","getCachedData","PostQuerySchema","signalPostStats","entityToJsonObject","getCreated","CommentSchema","qetaCreateCommentPermission","wrapAsync","findEntityMentions","qetaEditCommentPermission","qetaDeleteCommentPermission","PostSchema","qetaCreatePostPermission","getTags","getEntities","qetaEditPostPermission","DeleteMetadataSchema","qetaDeletePostPermission","PostReviewBodySchema","qetaCreatePostReviewPermission","qetaReadPostReviewPermission","URLMetadataSchema","extractMetadata"],"mappings":"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"}
@@ -36,7 +36,40 @@ const getEntities = (request, config) => {
36
36
  }
37
37
  return entities;
38
38
  };
39
+ const getCachedData = async (cache, key, ttl, fetchFn, logger) => {
40
+ if (!cache) {
41
+ return fetchFn();
42
+ }
43
+ const cached = await cache.get(key);
44
+ let data;
45
+ if (cached) {
46
+ try {
47
+ data = JSON.parse(cached);
48
+ } catch (_e) {
49
+ }
50
+ }
51
+ const refresh = async () => {
52
+ try {
53
+ const fresh2 = await fetchFn();
54
+ await cache.set(key, JSON.stringify(fresh2), { ttl });
55
+ return fresh2;
56
+ } catch (error) {
57
+ logger?.warn(`Failed to refresh cache for ${key}: ${error}`);
58
+ return void 0;
59
+ }
60
+ };
61
+ if (data) {
62
+ refresh();
63
+ return data;
64
+ }
65
+ const fresh = await refresh();
66
+ if (!fresh) {
67
+ throw new Error("Failed to fetch data");
68
+ }
69
+ return fresh;
70
+ };
39
71
 
72
+ exports.getCachedData = getCachedData;
40
73
  exports.getEntities = getEntities;
41
74
  exports.getTags = getTags;
42
75
  //# sourceMappingURL=routeUtil.cjs.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"routeUtil.cjs.js","sources":["../../../src/service/routes/routeUtil.ts"],"sourcesContent":["import { Config } from '@backstage/config';\nimport { Request } from 'express';\nimport {\n qetaCreateTagPermission,\n TagsResponse,\n} from '@drodil/backstage-plugin-qeta-common';\nimport { RouteOptions } from '../types';\n\nexport const getTags = async (\n request: Request,\n options: RouteOptions,\n existingTags: TagsResponse,\n) => {\n if (!request.body.tags) {\n return [];\n }\n\n const maxTags = options.config.getOptionalNumber('qeta.tags.max') ?? 5;\n const allowedTags =\n options.config.getOptionalStringArray('qeta.tags.allowedTags') ?? [];\n const allowTagCreationResults = await options.permissionMgr.authorizeBoolean(\n request,\n [{ permission: qetaCreateTagPermission }],\n );\n const allowTagCreation = allowTagCreationResults[0] ?? false;\n\n const rawTags = request.body.tags;\n let tags: string[] = Array.isArray(rawTags) ? rawTags : rawTags.split(',');\n\n if (tags.length === 0) {\n return [];\n }\n\n if (!allowTagCreation) {\n tags = tags.filter(\n tag =>\n allowedTags.includes(tag) || existingTags.tags.some(t => t.tag === tag),\n );\n }\n\n return tags.slice(0, maxTags);\n};\n\nexport const getEntities = (request: Request, config: Config): string[] => {\n if (!request.body.entities) {\n return [];\n }\n const maxEntities = config.getOptionalNumber('qeta.entities.max') ?? 3;\n let entities = request.body.entities;\n if (Array.isArray(entities)) {\n entities = entities.slice(0, maxEntities);\n }\n return entities;\n};\n"],"names":["qetaCreateTagPermission"],"mappings":";;;;AAQO,MAAM,OAAU,GAAA,OACrB,OACA,EAAA,OAAA,EACA,YACG,KAAA;AACH,EAAI,IAAA,CAAC,OAAQ,CAAA,IAAA,CAAK,IAAM,EAAA;AACtB,IAAA,OAAO,EAAC;AAAA;AAGV,EAAA,MAAM,OAAU,GAAA,OAAA,CAAQ,MAAO,CAAA,iBAAA,CAAkB,eAAe,CAAK,IAAA,CAAA;AACrE,EAAA,MAAM,cACJ,OAAQ,CAAA,MAAA,CAAO,sBAAuB,CAAA,uBAAuB,KAAK,EAAC;AACrE,EAAM,MAAA,uBAAA,GAA0B,MAAM,OAAA,CAAQ,aAAc,CAAA,gBAAA;AAAA,IAC1D,OAAA;AAAA,IACA,CAAC,EAAE,UAAY,EAAAA,iDAAA,EAAyB;AAAA,GAC1C;AACA,EAAM,MAAA,gBAAA,GAAmB,uBAAwB,CAAA,CAAC,CAAK,IAAA,KAAA;AAEvD,EAAM,MAAA,OAAA,GAAU,QAAQ,IAAK,CAAA,IAAA;AAC7B,EAAI,IAAA,IAAA,GAAiB,MAAM,OAAQ,CAAA,OAAO,IAAI,OAAU,GAAA,OAAA,CAAQ,MAAM,GAAG,CAAA;AAEzE,EAAI,IAAA,IAAA,CAAK,WAAW,CAAG,EAAA;AACrB,IAAA,OAAO,EAAC;AAAA;AAGV,EAAA,IAAI,CAAC,gBAAkB,EAAA;AACrB,IAAA,IAAA,GAAO,IAAK,CAAA,MAAA;AAAA,MACV,CAAA,GAAA,KACE,WAAY,CAAA,QAAA,CAAS,GAAG,CAAA,IAAK,YAAa,CAAA,IAAA,CAAK,IAAK,CAAA,CAAA,CAAA,KAAK,CAAE,CAAA,GAAA,KAAQ,GAAG;AAAA,KAC1E;AAAA;AAGF,EAAO,OAAA,IAAA,CAAK,KAAM,CAAA,CAAA,EAAG,OAAO,CAAA;AAC9B;AAEa,MAAA,WAAA,GAAc,CAAC,OAAA,EAAkB,MAA6B,KAAA;AACzE,EAAI,IAAA,CAAC,OAAQ,CAAA,IAAA,CAAK,QAAU,EAAA;AAC1B,IAAA,OAAO,EAAC;AAAA;AAEV,EAAA,MAAM,WAAc,GAAA,MAAA,CAAO,iBAAkB,CAAA,mBAAmB,CAAK,IAAA,CAAA;AACrE,EAAI,IAAA,QAAA,GAAW,QAAQ,IAAK,CAAA,QAAA;AAC5B,EAAI,IAAA,KAAA,CAAM,OAAQ,CAAA,QAAQ,CAAG,EAAA;AAC3B,IAAW,QAAA,GAAA,QAAA,CAAS,KAAM,CAAA,CAAA,EAAG,WAAW,CAAA;AAAA;AAE1C,EAAO,OAAA,QAAA;AACT;;;;;"}
1
+ {"version":3,"file":"routeUtil.cjs.js","sources":["../../../src/service/routes/routeUtil.ts"],"sourcesContent":["import { Config } from '@backstage/config';\nimport { Request } from 'express';\nimport { CacheService, LoggerService } from '@backstage/backend-plugin-api';\nimport {\n qetaCreateTagPermission,\n TagsResponse,\n} from '@drodil/backstage-plugin-qeta-common';\nimport { RouteOptions } from '../types';\n\nexport const getTags = async (\n request: Request,\n options: RouteOptions,\n existingTags: TagsResponse,\n) => {\n if (!request.body.tags) {\n return [];\n }\n\n const maxTags = options.config.getOptionalNumber('qeta.tags.max') ?? 5;\n const allowedTags =\n options.config.getOptionalStringArray('qeta.tags.allowedTags') ?? [];\n const allowTagCreationResults = await options.permissionMgr.authorizeBoolean(\n request,\n [{ permission: qetaCreateTagPermission }],\n );\n const allowTagCreation = allowTagCreationResults[0] ?? false;\n\n const rawTags = request.body.tags;\n let tags: string[] = Array.isArray(rawTags) ? rawTags : rawTags.split(',');\n\n if (tags.length === 0) {\n return [];\n }\n\n if (!allowTagCreation) {\n tags = tags.filter(\n tag =>\n allowedTags.includes(tag) || existingTags.tags.some(t => t.tag === tag),\n );\n }\n\n return tags.slice(0, maxTags);\n};\n\nexport const getEntities = (request: Request, config: Config): string[] => {\n if (!request.body.entities) {\n return [];\n }\n const maxEntities = config.getOptionalNumber('qeta.entities.max') ?? 3;\n let entities = request.body.entities;\n if (Array.isArray(entities)) {\n entities = entities.slice(0, maxEntities);\n }\n return entities;\n};\n\nexport const getCachedData = async <T>(\n cache: CacheService | undefined,\n key: string,\n ttl: number,\n fetchFn: () => Promise<T>,\n logger?: LoggerService,\n): Promise<T> => {\n if (!cache) {\n return fetchFn();\n }\n\n const cached = await cache.get(key);\n let data: T | undefined;\n if (cached) {\n try {\n data = JSON.parse(cached as string);\n } catch (_e) {\n // NOOP\n }\n }\n\n const refresh = async () => {\n try {\n const fresh = await fetchFn();\n await cache.set(key, JSON.stringify(fresh), { ttl });\n return fresh;\n } catch (error) {\n logger?.warn(`Failed to refresh cache for ${key}: ${error}`);\n return undefined;\n }\n };\n\n if (data) {\n refresh();\n return data;\n }\n\n const fresh = await refresh();\n if (!fresh) {\n throw new Error('Failed to fetch data');\n }\n return fresh;\n};\n"],"names":["qetaCreateTagPermission","fresh"],"mappings":"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"}
@@ -1,6 +1,7 @@
1
1
  'use strict';
2
2
 
3
3
  var util = require('../util.cjs.js');
4
+ var routeUtil = require('./routeUtil.cjs.js');
4
5
 
5
6
  const statisticRoutes = (router, options) => {
6
7
  const { database, permissionMgr } = options;
@@ -31,82 +32,119 @@ const statisticRoutes = (router, options) => {
31
32
  };
32
33
  router.get("/statistics/user/impact", async (request, response) => {
33
34
  const userRef = await permissionMgr.getUsername(request);
34
- const userImpact = await database.getTotalViews(userRef, void 0, true);
35
- const userImpactWeek = await database.getTotalViews(userRef, 7, true);
36
- return response.status(200).json({ impact: userImpact, lastWeekImpact: userImpactWeek });
35
+ const key = `qeta:statistics:impact:${userRef}`;
36
+ const ttl = 300 * 1e3;
37
+ const impact = await routeUtil.getCachedData(
38
+ options.cache,
39
+ key,
40
+ ttl,
41
+ async () => {
42
+ const userImpact = await database.getTotalViews(
43
+ userRef,
44
+ void 0,
45
+ true
46
+ );
47
+ const userImpactWeek = await database.getTotalViews(userRef, 7, true);
48
+ return { impact: userImpact, lastWeekImpact: userImpactWeek };
49
+ },
50
+ options.logger
51
+ );
52
+ return response.status(200).json(impact);
37
53
  });
38
54
  router.get("/statistics/global", async (_req, response) => {
39
- const globalStats = await database.getGlobalStats();
40
- let summary = await getSummary();
41
- if (!summary) {
42
- summary = {
43
- totalAnswers: 0,
44
- totalArticles: 0,
45
- totalLinks: 0,
46
- totalComments: 0,
47
- totalQuestions: 0,
48
- totalViews: 0,
49
- totalVotes: 0,
50
- totalTags: 0,
51
- totalUsers: 0
52
- };
53
- }
54
- let todayStatsAdded = false;
55
- const statistics = globalStats.map((g) => {
56
- const d = new Date(g.date);
57
- if (d.toDateString() !== (/* @__PURE__ */ new Date()).toDateString()) {
58
- return g;
59
- }
60
- todayStatsAdded = true;
61
- return {
62
- date: d,
63
- ...summary
64
- };
65
- });
66
- if (!todayStatsAdded) {
67
- statistics.push({ date: /* @__PURE__ */ new Date(), ...summary });
68
- }
69
- return response.status(200).json({ statistics, summary });
55
+ const key = "qeta:statistics:global";
56
+ const ttl = 3600 * 1e3;
57
+ const data = await routeUtil.getCachedData(
58
+ options.cache,
59
+ key,
60
+ ttl,
61
+ async () => {
62
+ const globalStats = await database.getGlobalStats();
63
+ let summary = await getSummary();
64
+ if (!summary) {
65
+ summary = {
66
+ totalAnswers: 0,
67
+ totalArticles: 0,
68
+ totalLinks: 0,
69
+ totalComments: 0,
70
+ totalQuestions: 0,
71
+ totalViews: 0,
72
+ totalVotes: 0,
73
+ totalTags: 0,
74
+ totalUsers: 0
75
+ };
76
+ }
77
+ let todayStatsAdded = false;
78
+ const statistics = globalStats.map((g) => {
79
+ const d = new Date(g.date);
80
+ if (d.toDateString() !== (/* @__PURE__ */ new Date()).toDateString()) {
81
+ return g;
82
+ }
83
+ todayStatsAdded = true;
84
+ return {
85
+ date: d,
86
+ ...summary
87
+ };
88
+ });
89
+ if (!todayStatsAdded) {
90
+ statistics.push({ date: /* @__PURE__ */ new Date(), ...summary });
91
+ }
92
+ return { statistics, summary };
93
+ },
94
+ options.logger
95
+ );
96
+ return response.json(data);
70
97
  });
71
98
  router.get("/statistics/user/:userRef(*)", async (req, response) => {
72
99
  const userRef = req.params.userRef;
73
- const userStats = await database.getUserStats(userRef);
74
- let summary = await database.getUser(userRef);
75
- let todayStatsAdded = false;
76
- const statistics = userStats.map((g) => {
77
- const d = new Date(g.date);
78
- if (!summary || d.toDateString() !== (/* @__PURE__ */ new Date()).toDateString()) {
79
- return g;
80
- }
81
- todayStatsAdded = true;
82
- return {
83
- date: d,
84
- ...summary
85
- };
86
- });
87
- if (!todayStatsAdded) {
88
- if (!summary) {
89
- summary = {
90
- userRef,
91
- totalViews: 0,
92
- totalQuestions: 0,
93
- totalAnswers: 0,
94
- totalComments: 0,
95
- totalVotes: 0,
96
- totalArticles: 0,
97
- totalFollowers: 0,
98
- totalLinks: 0,
99
- reputation: 0,
100
- answerScore: 0,
101
- postScore: 0,
102
- correctAnswers: 0
103
- };
104
- }
105
- if (userStats.length === 0) {
106
- statistics.push({ date: /* @__PURE__ */ new Date(), ...summary });
107
- }
108
- }
109
- return response.status(200).json({ statistics, summary });
100
+ const key = `qeta:statistics:user:${userRef}`;
101
+ const ttl = 3600 * 1e3;
102
+ const data = await routeUtil.getCachedData(
103
+ options.cache,
104
+ key,
105
+ ttl,
106
+ async () => {
107
+ const userStats = await database.getUserStats(userRef);
108
+ let summary = await database.getUser(userRef);
109
+ let todayStatsAdded = false;
110
+ const statistics = userStats.map((g) => {
111
+ const d = new Date(g.date);
112
+ if (!summary || d.toDateString() !== (/* @__PURE__ */ new Date()).toDateString()) {
113
+ return g;
114
+ }
115
+ todayStatsAdded = true;
116
+ return {
117
+ date: d,
118
+ ...summary
119
+ };
120
+ });
121
+ if (!todayStatsAdded) {
122
+ if (!summary) {
123
+ summary = {
124
+ userRef,
125
+ totalViews: 0,
126
+ totalQuestions: 0,
127
+ totalAnswers: 0,
128
+ totalComments: 0,
129
+ totalVotes: 0,
130
+ totalArticles: 0,
131
+ totalFollowers: 0,
132
+ totalLinks: 0,
133
+ reputation: 0,
134
+ answerScore: 0,
135
+ postScore: 0,
136
+ correctAnswers: 0
137
+ };
138
+ }
139
+ if (userStats.length === 0) {
140
+ statistics.push({ date: /* @__PURE__ */ new Date(), ...summary });
141
+ }
142
+ }
143
+ return { statistics, summary };
144
+ },
145
+ options.logger
146
+ );
147
+ return response.status(200).json(data);
110
148
  });
111
149
  router.get(
112
150
  "/statistics/posts/top-upvoted-users",