detail.js 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186
  1. var express = require('express');
  2. var router = express.Router();
  3. const redis = require('../../libs/redis');
  4. const models = require('../../models');
  5. const utils = require('../../libs/utils');
  6. const common = require('./common');
  7. const { getListBuilder } = require('../../libs/pager');
  8. const categories = require('../../config/category');
  9. const { tags } = require('../../config/tag');
  10. const { coloringList } = require('./config');
  11. const { getSimilarArts, getSimilarArtsSimple } = require('./similarArts');
  12. const CACHE_PREFIX = "art_v2";
  13. // const CACHE_EXPIRES = 60; // 60s刷新一次
  14. const CACHE_EXPIRES = 600;
  15. const artSelect = 'name title desc seoTitle seoDescription width height date publishTime tags lastMod mystery hasSpecial useSpecialThumb publishVersion totalStartCount totalDoneCount completionRate';
  16. // 详情页路由
  17. router.get('/:id', function (req, res, next) {
  18. (async function () {
  19. let id = req.params.id;
  20. utils.validators.validateId(id);
  21. // let cacheKey = `${CACHE_PREFIX}_detail_${id}`;
  22. // let htmlData = await redis.getAsync(cacheKey);
  23. let htmlData = null;
  24. if (!htmlData) {
  25. // 详情
  26. let doc = await models.Art
  27. .findById(id)
  28. .select(artSelect)
  29. .populate('user', 'username name')
  30. .lean()
  31. .exec();
  32. if (!doc) throw createError(404, 'Art Not Found!');
  33. common.organizeDetail(doc);
  34. // 关联推荐
  35. // let mytags = [...doc.tags];
  36. // let cates = categories.map(e => e.id);
  37. // let alltags = tags.map(e => e.tag);
  38. // mytags = mytags.filter(e => !cates.includes(e) && alltags.includes(e));
  39. // if (mytags.length == 0) mytags = [...doc.tags];
  40. // let query = {
  41. // page: req.query.page,
  42. // length: req.query.length,
  43. // search: req.query.search,
  44. // orderBy: 'publishTime',
  45. // order: 'desc',
  46. // base: { open: true, status: 9000 },
  47. // filters: { tags: mytags },
  48. // }
  49. // let result = await getListBuilder(query, models.Art, [{ path: 'user', select: 'username' }]);
  50. // common.organizeData(result.data);
  51. // 用新的算法
  52. console.time('getSimilarArts');
  53. let result = await getSimilarArts(id, {limit: 30, candidateLimit: 1000, fields : artSelect });
  54. console.timeEnd('getSimilarArts')
  55. common.organizeData(result);
  56. // 填色页合集推荐
  57. const recmCollections = recommendColoringPages(doc, coloringList);
  58. // 评论
  59. const comments = await models.Comment.find({ approved: true, page: doc._id }).sort({ createdAt: -1 });
  60. // deeplink 相关
  61. let applink = `https://art.pcoloring.com${req.originalUrl}`;
  62. let downlink = `https://art.pcoloring.com/app`;
  63. let openapplink = applink;
  64. if (!req.originalUrl.includes('check')) {
  65. openapplink = applink.concat(req.originalUrl.includes('?') ? '&check=1' : '?check=1');
  66. }
  67. const userAgent = req.headers['user-agent'];
  68. console.log('User-Agent:', userAgent);
  69. if (userAgent) {
  70. const ua = userAgent.toLowerCase();
  71. if (ua.includes('iphone') || ua.includes('ipad') || ua.includes('ipod')) {
  72. downlink = 'itms-apps://itunes.apple.com/app/id1575480118?utm_source=share';
  73. } else if (ua.includes('android')) {
  74. downlink = 'https://play.google.com/store/apps/details?id=com.pcoloring.art.puzzle.color.by.number&utm_source=share';
  75. }
  76. }
  77. let data = {
  78. title: `${doc.seoTitle}`,
  79. description: `${doc.seoDescription}`,
  80. detail: doc,
  81. // data: result.data,
  82. page: result.page,
  83. pageId: doc._id,
  84. // length: result.length,
  85. // recordsFiltered: result.recordsFiltered,
  86. // recordsTotal: result.recordsTotal,
  87. // relates: result.data,
  88. relates: result,
  89. uri: `/coloring-page/${doc._id}`,
  90. imageUrl: doc.thumb,
  91. pageUri: common.replaceUriParams,
  92. comments,
  93. dateFormat: common.dateFormat,
  94. collections: recmCollections,
  95. applink,
  96. downlink,
  97. openapplink,
  98. };
  99. // 渲染EJS模板到内存中
  100. res.render('v2/detail', data, async (err, html) => {
  101. if (err) {
  102. // 如果渲染出错,调用next()传递错误
  103. return next(err);
  104. }
  105. // 渲染成功,存redis, 发送数据到客户端
  106. htmlData = html;
  107. // try {
  108. // await redis.set(cacheKey, htmlData, 'EX', CACHE_EXPIRES);
  109. // } catch (e) {
  110. // console.error(e);
  111. // }
  112. res.send(htmlData);
  113. });
  114. } else {
  115. // 缓存命中, 直接发送缓存数据
  116. res.set({ 'X-From-Cache': 'true' });
  117. res.send(htmlData);
  118. }
  119. })().catch(next);
  120. });
  121. /**
  122. * 填色页推荐系统 - 根据标签相似度推荐相关填色页合集
  123. * @param {Object} currentPage - 当前填色页对象,包含tags属性
  124. * @param {Array} collections - 所有填色页合集列表
  125. * @returns {Array} - 推荐的3个填色页合集
  126. */
  127. function recommendColoringPages(currentPage, collections) {
  128. // 计算每个候选页与当前页的标签匹配度
  129. const scoredPages = collections.map(page => {
  130. const matchedTags = page.tags.filter(tag =>
  131. currentPage.tags.some(currentTag =>
  132. currentTag.toLowerCase() === tag.toLowerCase()
  133. )
  134. );
  135. return {
  136. ...page,
  137. score: matchedTags.length,
  138. matchedTags
  139. };
  140. });
  141. // 按匹配度降序排序
  142. scoredPages.sort((a, b) => b.score - a.score);
  143. // 收集匹配度大于0的推荐
  144. const tagMatchedRecommendations = scoredPages.filter(page => page.score > 0);
  145. // 如果标签匹配的推荐不足3个,从剩余的合集中随机选取补足
  146. if (tagMatchedRecommendations.length < 3) {
  147. const remainingPages = scoredPages.filter(page => page.score === 0);
  148. // 打乱剩余页面顺序
  149. const shuffledRemaining = remainingPages.sort(() => 0.5 - Math.random());
  150. // 补足到3个推荐
  151. return [...tagMatchedRecommendations, ...shuffledRemaining].slice(0, 3);
  152. }
  153. // 如果标签匹配的推荐超过3个,取前3个
  154. return tagMatchedRecommendations.slice(0, 3);
  155. }
  156. module.exports = router;