server.ts 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  1. // @deno-types="https://unpkg.com/pocketbase@0.8.3/dist/pocketbase.es.d.ts"
  2. import PocketBase from "https://unpkg.com/pocketbase@0.8.3/dist/pocketbase.es.mjs";
  3. import { serve } from "https://deno.land/std/http/server.ts";
  4. import { Md5 } from "https://deno.land/std@0.160.0/hash/md5.ts";
  5. import "https://deno.land/std/dotenv/load.ts";
  6. const allowOrigin = Deno.env.get("ALLOW_ORIGIN")?.split(",");
  7. const webhookUrl = Deno.env.get("WEBHOOK_URL");
  8. let allowedOrigin = "*";
  9. const pb = new PocketBase(Deno.env.get("PB_URL"));
  10. const _authData = await pb.collection("users").authWithPassword(
  11. Deno.env.get("PB_USER"),
  12. Deno.env.get("PB_PASSWORD"),
  13. );
  14. // Validate Url
  15. const isValidUrl = (url: string) => {
  16. if (url === "") {
  17. return true; // "website" filed is optional
  18. } else {
  19. try {
  20. new URL(url);
  21. } catch (e) {
  22. console.error(e);
  23. return false;
  24. }
  25. return true;
  26. }
  27. };
  28. async function handler(req: Request): Promise<Response> {
  29. const url = new URL(req.url);
  30. // console.log(req.method, url.pathname, "uri:", url.searchParams.get("uri"));
  31. const reqestOrigin = req.headers.get("origin");
  32. if (reqestOrigin === null || !allowOrigin.includes(reqestOrigin)) {
  33. return new Response("Request is rejected due to CORS policy.");
  34. } else {
  35. allowedOrigin = reqestOrigin;
  36. }
  37. // List comments for a given page uri
  38. if (req.method === "GET" && url.searchParams.get("uri") !== null) {
  39. const resultList = await pb.collection("comments").getFullList(0, {
  40. filter: `uri='${url.searchParams.get("uri")}'`,
  41. sort: "created,-parent",
  42. });
  43. const commentlist: {
  44. id: string;
  45. author: string;
  46. avatar: string;
  47. website: string;
  48. content: string;
  49. created: string;
  50. reply: unknown[];
  51. }[] = [];
  52. resultList.forEach((item) => {
  53. if (item.parent === "") {
  54. commentlist.push({
  55. id: item.id,
  56. author: item.author,
  57. avatar: new Md5().update(item.email).toString(),
  58. website: item.website,
  59. content: item.content,
  60. created: item.created,
  61. reply: [],
  62. });
  63. } else {
  64. const index = commentlist.findIndex((e) => e.id === item.parent);
  65. commentlist[index].reply.push({
  66. id: item.id,
  67. author: item.author,
  68. avatar: new Md5().update(item.email).toString(),
  69. website: item.website,
  70. content: item.content,
  71. created: item.created,
  72. });
  73. }
  74. });
  75. const body = JSON.stringify({
  76. count: resultList.length,
  77. list: commentlist.reverse(),
  78. });
  79. return new Response(body, {
  80. status: 200,
  81. headers: {
  82. "Content-Type": "application/json; charset=UTF-8",
  83. "Access-Control-Allow-Origin": allowedOrigin,
  84. },
  85. });
  86. }
  87. // handle new comments
  88. if (req.method === "POST") {
  89. const newComment = await req.json();
  90. if (!newComment.author || !newComment.email || !newComment.content) {
  91. return new Response("名字、邮箱、评论内容不能为空");
  92. } else if (!isValidUrl(newComment.website)) {
  93. return new Response("网址格式错误");
  94. } else {
  95. const record = await pb.collection("comments").create({
  96. "uri": newComment.uri,
  97. "author": newComment.author,
  98. "email": newComment.email,
  99. "website": newComment.website,
  100. "content": newComment.content,
  101. "parent": newComment.parent,
  102. });
  103. const body = JSON.stringify({
  104. id: record.id,
  105. author: record.author,
  106. avatar: new Md5().update(record.email).toString(),
  107. website: record.website,
  108. content: record.content,
  109. created: record.created,
  110. });
  111. if (webhookUrl) {
  112. fetch(webhookUrl, {
  113. method: "POST",
  114. body: body,
  115. });
  116. }
  117. return new Response(body, {
  118. status: 200,
  119. headers: {
  120. "Content-Type": "application/json; charset=UTF-8",
  121. "Access-Control-Allow-Origin": allowedOrigin,
  122. },
  123. });
  124. }
  125. }
  126. if (req.method === "OPTIONS") {
  127. return new Response(null, {
  128. status: 204,
  129. headers: {
  130. "Access-Control-Allow-Methods": "GET, POST",
  131. "Access-Control-Allow-Origin": allowedOrigin,
  132. "Access-Control-Allow-Headers": "Origin, Referer, Content-Type",
  133. },
  134. });
  135. }
  136. return new Response("Bad request!");
  137. }
  138. serve(handler);