comment.ts 4.1 KB

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