{"id":3495,"date":"2025-05-17T17:52:46","date_gmt":"2025-05-17T08:52:46","guid":{"rendered":"https:\/\/bravesoft.co.jp\/seblog\/?p=3495"},"modified":"2025-05-18T23:00:46","modified_gmt":"2025-05-18T14:00:46","slug":"%e7%94%9f%e6%88%90ai%e3%81%a71%e8%a1%8c%e3%82%82%e3%82%b3%e3%83%bc%e3%83%89%e3%82%92%e6%9b%b8%e3%81%8b%e3%81%9a1%e4%b8%87%e8%a1%8c%e3%81%ae%e3%82%a2%e3%83%97%e3%83%aa%e3%82%92%e3%81%a4%e3%81%8f","status":"publish","type":"post","link":"https:\/\/bravesoft.co.jp\/seblog\/archives\/3495","title":{"rendered":"\u751f\u6210AI\u30671\u884c\u3082\u30b3\u30fc\u30c9\u3092\u66f8\u304b\u305a1\u4e07\u884c\u306e\u30a2\u30d7\u30ea\u3092\u3064\u304f\u3063\u305f\u8a71\uff08Cursor,Claude3.7,Node.js)  #36"},"content":{"rendered":"\n<p>\u751f\u6210AI\u306e\u9032\u5316\u304c\u6b62\u307e\u3089\u306a\u3044\uff01<\/p>\n\n\n\n<p>\u65b0\u6280\u8853\u3092\u77e5\u308b\u306b\u306f\u624b\u3092\u52d5\u304b\u3059\u306e\u304c\u4e00\u756a\u3002<br>\u3068\u3044\u3046\u3053\u3068\u3067\u3001GW\u306b1\u9031\u9593\u304b\u3051\u3066\u751f\u6210AI\u3067\u30a2\u30d7\u30ea\u3092\u3064\u304f\u3063\u3066\u307f\u305f\u3002\u6700\u8fd1\u306fvibe coding\uff08\u30d0\u30a4\u30d6\u30fb\u30b3\u30fc\u30c7\u30a3\u30f3\u30b0\uff09\u3068\u3044\u3046\u3089\u3057\u3044\u3002<\/p>\n\n\n\n<p>\u305d\u306e\u7d50\u679c\u3001AI\u3068\u306e\u5bfe\u8a71\u3067\u3001\uff11\u884c\u3082\u30b3\u30fc\u30c9\u3092\u66f8\u304b\u305a\u306b\u30011\u4e07\u884c\u306eWEB\u30a2\u30d7\u30ea\u304c\u51fa\u6765\u305f\u30fb\u30fb\uff01<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u3069\u3093\u306a\u30a2\u30d7\u30ea\uff1f<\/h2>\n\n\n\n<p>bravesoft\u3067\u8d77\u304d\u3066\u3044\u308b\u3053\u3068\u3092\u53ef\u8996\u5316\u3057\u3066\u5171\u6709\u3059\u308b\u30a2\u30d7\u30ea<br>\u305d\u306e\u540d\u3082\u300cBRAVE FAN BOARD\u300d\u3002\u30aa\u30d5\u30a3\u30b9\u3067\u30c7\u30a3\u30b9\u30d7\u30ec\u30a4\u306b\u8868\u793a\u3002<br>bravesoft\u5fdc\u63f4\u56e3\u304cbraver\u3092\u79f0\u3048\u305f\u308a\u3001Slack\u7b49\u3067\u306e\u30a2\u30af\u30c6\u30a3\u30d6\u5ea6\u3092\u898b\u3084\u3059\u304f\u5206\u6790\u3002<\/p>\n\n\n\n<p>\u30a2\u30d7\u30ea\u3092\u4f7f\u3063\u3066\u3044\u308b\u52d5\u753b\u304c\u3053\u3061\u3089\u3002\u3053\u3093\u306a\u30a2\u30d7\u30ea\u3092AI\u30681\u9031\u9593\u3067\u3064\u304f\u3063\u305f\u3002<\/p>\n\n\n\n<figure class=\"wp-block-video\"><video controls src=\"https:\/\/bravesoft.co.jp\/seblog\/wp-content\/uploads\/2025\/05\/bravefanboard_movie_fix.mp4\"><\/video><\/figure>\n\n\n<p>\n\n<p>\u50d5\u3082\u30a4\u30c1\u63a8\u3057braver\u306b\u767b\u5834\u3002AI\u306b\u8912\u3081\u3089\u308c\u3066\u306a\u3093\u304b\u3046\u308c\u3057\u3044\u7b11\u3002<\/p>\n<p><a href=\"https:\/\/i0.wp.com\/bravesoft.co.jp\/seblog\/wp-content\/uploads\/2025\/05\/ichi_suga.png?ssl=1\"><img loading=\"lazy\" decoding=\"async\" data-attachment-id=\"3505\" data-permalink=\"https:\/\/bravesoft.co.jp\/seblog\/archives\/3495\/ichi_suga\" data-orig-file=\"https:\/\/i0.wp.com\/bravesoft.co.jp\/seblog\/wp-content\/uploads\/2025\/05\/ichi_suga.png?fit=2154%2C1346&amp;ssl=1\" data-orig-size=\"2154,1346\" data-comments-opened=\"0\" data-image-meta=\"{&quot;aperture&quot;:&quot;0&quot;,&quot;credit&quot;:&quot;&quot;,&quot;camera&quot;:&quot;&quot;,&quot;caption&quot;:&quot;&quot;,&quot;created_timestamp&quot;:&quot;0&quot;,&quot;copyright&quot;:&quot;&quot;,&quot;focal_length&quot;:&quot;0&quot;,&quot;iso&quot;:&quot;0&quot;,&quot;shutter_speed&quot;:&quot;0&quot;,&quot;title&quot;:&quot;&quot;,&quot;orientation&quot;:&quot;0&quot;}\" data-image-title=\"ichi_suga\" data-image-description=\"\" data-image-caption=\"\" data-medium-file=\"https:\/\/i0.wp.com\/bravesoft.co.jp\/seblog\/wp-content\/uploads\/2025\/05\/ichi_suga.png?fit=300%2C187&amp;ssl=1\" data-large-file=\"https:\/\/i0.wp.com\/bravesoft.co.jp\/seblog\/wp-content\/uploads\/2025\/05\/ichi_suga.png?fit=700%2C437&amp;ssl=1\" tabindex=\"0\" role=\"button\" src=\"https:\/\/i0.wp.com\/bravesoft.co.jp\/seblog\/wp-content\/uploads\/2025\/05\/ichi_suga.png?resize=507%2C317&#038;ssl=1\" alt=\"\" width=\"507\" height=\"317\" class=\"alignnone  wp-image-3505\" srcset=\"https:\/\/i0.wp.com\/bravesoft.co.jp\/seblog\/wp-content\/uploads\/2025\/05\/ichi_suga.png?resize=700%2C437&amp;ssl=1 700w, https:\/\/i0.wp.com\/bravesoft.co.jp\/seblog\/wp-content\/uploads\/2025\/05\/ichi_suga.png?resize=300%2C187&amp;ssl=1 300w, https:\/\/i0.wp.com\/bravesoft.co.jp\/seblog\/wp-content\/uploads\/2025\/05\/ichi_suga.png?resize=768%2C480&amp;ssl=1 768w, https:\/\/i0.wp.com\/bravesoft.co.jp\/seblog\/wp-content\/uploads\/2025\/05\/ichi_suga.png?resize=1536%2C960&amp;ssl=1 1536w, https:\/\/i0.wp.com\/bravesoft.co.jp\/seblog\/wp-content\/uploads\/2025\/05\/ichi_suga.png?resize=2048%2C1280&amp;ssl=1 2048w, https:\/\/i0.wp.com\/bravesoft.co.jp\/seblog\/wp-content\/uploads\/2025\/05\/ichi_suga.png?resize=800%2C500&amp;ssl=1 800w, https:\/\/i0.wp.com\/bravesoft.co.jp\/seblog\/wp-content\/uploads\/2025\/05\/ichi_suga.png?resize=482%2C300&amp;ssl=1 482w, https:\/\/i0.wp.com\/bravesoft.co.jp\/seblog\/wp-content\/uploads\/2025\/05\/ichi_suga.png?w=1400&amp;ssl=1 1400w, https:\/\/i0.wp.com\/bravesoft.co.jp\/seblog\/wp-content\/uploads\/2025\/05\/ichi_suga.png?w=2100&amp;ssl=1 2100w\" sizes=\"(max-width: 507px) 100vw, 507px\" data-recalc-dims=\"1\" \/><\/a><\/p>\n<h2><b>\u958b\u767a\u306e\u9032\u3081\u65b9\uff1eCursor+Claude3.7\u3092\u5c0e\u5165<\/b><\/h2>\n<p>Cursor\u306fAI\u30a8\u30fc\u30b8\u30a7\u30f3\u30c8\u3068\u5bfe\u8a71\u3057\u306a\u304c\u3089\u3001\u74b0\u5883\u69cb\u7bc9\u30fb\u958b\u767a\u30fb\u5b9f\u884c\u30fb\u691c\u8a3c\u30fb\u30d0\u30b0\u306e\u4fee\u6b63\u307e\u3067\u5168\u90e8\u3084\u3063\u3066\u304f\u308c\u3066\u304b\u306a\u308a\u4fbf\u5229\u3002<b>AI\u3068\u4e00\u7dd2\u306bNode.js\u3001TailwindCSS\u3001GitHub\u74b0\u5883\u3092\u69cb\u7bc9\u3002<\/b>\u3053\u308c\u307e\u3067\u306fChatGPT\u306a\u308aAI\u30c4\u30fc\u30eb\u306b\u3044\u308d\u3044\u308d\u30b3\u30d4\u30da\u3059\u308b\u5fc5\u8981\u304c\u3042\u3063\u305f\u3002Cursor\u306a\u3089\u74b0\u5883\u5168\u4f53\u3092AI\u304c\u628a\u63e1\u30fb\u7406\u89e3\u3057\u3066\u64cd\u4f5c\u30fb\u7de8\u96c6\u307e\u3067\u3057\u3066\u304f\u308c\u308b\u3002\u5f37\u529b\u306a\u30d0\u30c7\u30a3\u3068\u3044\u3063\u305f\u611f\u3058\u3002<\/p>\n<p>AI\u306b\u306fChatGPT\u3084Gemini\u3082\u6307\u5b9a\u3067\u304d\u308b\u304c\u3001\u3084\u3063\u3071\u308aClaude3.7\u304c\u512a\u79c0\u3068\u611f\u3058\u305f\u3002<\/p>\n<h2><b>\uff11\u884c\u305f\u308a\u3068\u3082\u30bd\u30fc\u30b9\u30b3\u30fc\u30c9\u3092\u89e6\u3089\u306a\u3044\u3068\u8a93\u3046<\/b><\/h2>\n<p>\u3084\u308b\u304b\u3089\u306b\u306f\u5b8c\u5168\u306bAI\u306b\u958b\u767a\u3057\u3066\u3082\u3089\u304a\u3046\u3002\u3068\u3044\u3046\u3053\u3068\u3067\uff11\u884c\u3082\u30b3\u30fc\u30c7\u30a3\u30f3\u30b0\u305b\u305a\u3002\u300c\u3053\u3046\u3044\u3046\u30bd\u30fc\u30b9\u3092\u66f8\u3044\u3066\u3001\u3068\u304b\u30c7\u30fc\u30bf\u306f\u3053\u306e\u3088\u3046\u306b\u4fdd\u5b58\u3057\u3066\u3002\u30c7\u30b6\u30a4\u30f3\u306f\u3082\u3046\u3061\u3087\u3063\u3068\u30b3\u30b3\u3092\u5927\u304d\u304f\u300d\u306a\u3069\u3068\u4eba\u9593\u306b\u53e3\u982d\u3067\u4f1d\u3048\u308b\u5f62\u3067\u5b8c\u6210\u307e\u3067\u3044\u3051\u305f\u3002\uff08\u305f\u3060\u3001\u30b3\u30fc\u30c9\u3084\u30ed\u30b0\u3092\u3042\u308b\u7a0b\u5ea6\u7406\u89e3\u3057\u305f\u3046\u3048\u3067\u6307\u793a\u3092\u5224\u65ad\u3059\u308b\u5c40\u9762\u3082\u591a\u3005\u3042\u308a\u3001\u30b3\u30fc\u30c7\u30a3\u30f3\u30b0\u306e\u77e5\u8b58\u304c\u8584\u3044\u4eba\u306b\u306f\u307e\u3060\u307e\u3060\u96e3\u3057\u3044\u306e\u304c\u73fe\u72b6\uff09<\/p>\n<h2><b>\u30c7\u30fc\u30bf\u3067\u307f\u308b\u958b\u767a<\/b><\/h2>\n<p>\u6240\u8981\u6642\u9593\uff1a50\u6642\u9593\u7a0b\u5ea6<br \/>\u753b\u9762\u6570\uff1a30\u753b\u9762<br \/>AI\u3068\u306e\u4f1a\u8a71\u6570\uff1a\u3060\u3044\u305f\u3044500\u56de<br \/>AI\u304c\u66f8\u3044\u305f\u30b3\u30fc\u30c9\u6570\uff1a\u7d041\u4e07\u884c<br \/>\u5fc5\u8981\u7d4c\u8cbb\uff1aAI\u30c4\u30fc\u30eb\u8cbb1\u4e07\u5186\u672a\u6e80\u3001\u4f1a\u793e\u306b\u3042\u3063\u305f\u53e4\u3044MAC,\u30c7\u30a3\u30b9\u30d7\u30ec\u30a4<\/p>\n<h2><b>\u958b\u767a\u30b7\u30fc\u30f3\u52d5\u753b<\/b><\/h2>\n<p>\u5b9f\u969b\u306bCursor\u3067\u958b\u767a\u3057\u3066\u3044\u308b\u30b7\u30fc\u30f3\u304c\u3053\u3061\u3089\u3002\u30a8\u30e9\u30fc\u304c\u8d77\u304d\u3066\u305f\u306e\u3067\u3001\u30a8\u30e9\u30fc\u6587\u3092\u8cbc\u308a\u4ed8\u3051\u3066\u300c\u3053\u306e\u30a8\u30e9\u30fc\u3092\u76f4\u3057\u3066\u300d\u3068\u3044\u3046\u6307\u793a\u3092\u3060\u3057\u305f\u3002\u305f\u3063\u305f\u305d\u308c\u3060\u3051\u3002\u3042\u3068\u306f\u5b8c\u5168\u81ea\u52d5\u3002\u5b9f\u884c\u30dc\u30bf\u30f3\u3092\u62bc\u3057\u7d9a\u3051\u3066\u3044\u308b\u3068\u30a8\u30e9\u30fc\u304c\u76f4\u3063\u3066\u308b\u3002\uff08\u3046\u307e\u304f\u3044\u304b\u306a\u304f\u3066\u3084\u308a\u76f4\u3059\u6642\u3082\u591a\u3005\u3042\u308b\uff09<\/p>\n\n\n<figure class=\"wp-block-video\"><video controls loop muted src=\"https:\/\/bravesoft.co.jp\/seblog\/wp-content\/uploads\/2025\/05\/cursor\u3066\u3099\u81ea\u52d5\u30cf\u3099\u30af\u3099\u4fee\u6b63.mp4\" playsinline><\/video><\/figure>\n\n\n\n<h2 class=\"wp-block-heading\"><strong>\u611f\u60f3\uff0975\u70b9\u307e\u3067\u306f\u4e00\u77ac<\/strong><\/h2>\n\n\n\n<p>\u3056\u3063\u3068\u4f1d\u3048\u3066\u52d5\u304f\u3082\u306e\u3092\u3064\u304f\u308b\u306e\u304c\u30db\u30f3\u30c8\u306b\u65e9\u3044\u3002\u3056\u3063\u304f\u308a\u3057\u305f\u6307\u793a\u3067OK\u3002\u5fc5\u8981\u306a\u95a2\u9023\u6a5f\u80fd\u3082\u63d0\u6848\u3057\u3066\u304f\u308c\u3066\u3059\u3050\u5b9f\u88c5\u5b8c\u4e86\u3002\u4f8b\u3048\u3070\u30e6\u30fc\u30b6\u4e00\u89a7\u3084\u30b0\u30e9\u30d5\u8868\u793a\u306a\u3069\u3001\u30db\u30f3\u30c8\u306b\u4e00\u77ac\u3067\u51fa\u6765\u305f\u3002\u3055\u304d\u306b\u304b\u3063\u3061\u308a\u4ed5\u69d8\u3092\u6c7a\u3081\u308b\u306e\u3067\u306f\u306a\u304f\u3001AI\u3068\u3064\u304f\u308a\u306a\u304c\u3089\u4ed5\u69d8\u3092\u8003\u3048\u3066\u3044\u304f\u611f\u3058\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\"><strong>100\u70b9\u306b\u3082\u3063\u3066\u3044\u304f\u306e\u306f\u5927\u5909<\/strong><\/h2>\n\n\n\n<p>\u672c\u756a\u3067\u4f7f\u3048\u308b\u3088\u3046\u306b\u7d30\u304b\u306a\u6700\u7d42\u8abf\u6574\u3092\u3059\u308b\u306e\u304c\u3068\u3066\u3082\u5927\u5909\u3060\u3063\u305f\u3002\u4f8b\u3048\u3070\u30c7\u30fc\u30bf\u53d6\u5f97\u51e6\u7406\u3092\u5b9f\u884c\u3059\u308b\u30bf\u30a4\u30df\u30f3\u30b0\u306e\u5909\u66f4\u3002\u753b\u50cf\u8868\u793a\u306e\u7d30\u304b\u306a\u8abf\u6574\u3002\u3061\u3087\u3063\u3068\u6c17\u306e\u5229\u3044\u305f\u6a5f\u80fd\u3092\u8ffd\u52a0\u3057\u305f\u3044\u3002\u306a\u3069\u304c\u5927\u5909\u3002\u3042\u3063\u3061\u3092\u5909\u3048\u305f\u3089\u3053\u3063\u3061\u304c\u30d0\u30b0\u308b\u73fe\u8c61\u3082\u8d77\u304d\u305f\u308a\u3002\u601d\u3044\u901a\u308a\u306b\u3044\u304b\u305a\u3001\u6700\u7d42\u8abf\u6574\u306b\u304b\u306a\u308a\u6642\u9593\u3092\u53d6\u3089\u308c\u305f\u3002<\/p>\n\n\n\n<p>75\u70b9\u307e\u3067\u306f10\u6642\u9593\u3002\u305d\u3053\u304b\u3089100\u70b9\u306b\u3082\u3063\u3066\u304f\u307e\u3067\u306b40\u6642\u9593\u304b\u304b\u308b\u3088\u3046\u306a\u611f\u3058\u3060\u3002\u305f\u3068\u3048\u3070AI\u306b\u7d75\u3092\u66f8\u3044\u3066\u3082\u3089\u3046\u6642\u3082\u4f3c\u305f\u3088\u3046\u306a\u3053\u3068\u304c\u8d77\u3053\u308b\u3002\u3056\u3063\u304f\u308a\u3053\u3046\u3044\u3046\u7d75\u3092\u66f8\u3044\u3066\uff01\u3000\u306f\u3059\u3050\u306b\u51fa\u6765\u308b\u3051\u3069\u3002\u3053\u306e\u7d75\u306e\u3053\u306e\u8fba\u306b\u3053\u3046\u3044\u3046\u7269\u4f53\u3092\u7f6e\u3044\u3066\u3001\u3068\u3044\u3046\u7d30\u304b\u3044\u6307\u793a\u3092\u3059\u308b\u3068\u4e00\u6c17\u306b\u7cbe\u5ea6\u304c\u60aa\u304f\u306a\u308a\u8a66\u884c\u56de\u6570\u304c\u5897\u3048\u308b\u3042\u306e\u611f\u3058\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\"><strong>\u3069\u3093\u3069\u3093\u91cd\u304f\u306a\u308a\u7cbe\u5ea6\u306f\u60aa\u304f\u306a\u308b<\/strong><\/h2>\n\n\n\n<p>\u30bd\u30fc\u30b9\u304c\u6570\u5343\u884c\u306b\u306a\u3063\u3066\u304f\u308b\u3068AI\u306e\u601d\u8003\u6642\u9593\u304c\u3069\u3093\u3069\u3093\u9577\u304f\u306a\u308a\u3001\u4fee\u6b63\u7cbe\u5ea6\u3082\u843d\u3061\u308b\u3002<br>\u5f8c\u534a\u6226\u306e\u65b9\u306f\u4eba\u9593\u304c\u7de8\u96c6\u3057\u305f\u307b\u3046\u304c\u65e9\u3044\u3068\u611f\u3058\u308b\u3053\u3068\u304c\u591a\u304b\u3063\u305f\u3002\u8907\u96d1\u306a\u30b7\u30b9\u30c6\u30e0\u306e\u958b\u767a\u306f\u307e\u3060\u96e3\u3057\u3044\u3002\u30d7\u30ed\u30c8\u3084\u30c7\u30fc\u30bf\u8868\u793a\u304c\u30e1\u30a4\u30f3\u306a\u30c0\u30c3\u30b7\u30e5\u30dc\u30fc\u30c9\u306a\u3069\u306b\u306f\u5341\u5206\u4f7f\u3048\u308b\u304c\u3001\u8a2d\u8a08\u3084\u30a2\u30fc\u30ad\u30c6\u30af\u30c1\u30e3\u306a\u3069\u3061\u3083\u3093\u3068\u69cb\u7bc9\u3067\u304d\u308b\u308f\u3051\u3067\u306f\u306a\u3044\u306e\u3067\u3001\u305d\u308c\u4ee5\u4e0a\u8907\u96d1\u306a\u30b7\u30b9\u30c6\u30e0\u306fAI\u30d9\u30fc\u30b9\u3067\u958b\u767a\u3059\u308b\u306e\u306f\u96e3\u3057\u3044\u3002\u4eba\u9593\u304c\u8a2d\u8a08\u3057\u90e8\u5206\u7684\u306a\u30b3\u30fc\u30c7\u30a3\u30f3\u30b0\u3092\u88dc\u4f50\u3057\u3066\u3082\u3089\u3046\u306e\u304c\u73fe\u5728\u5730\u70b9\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\"><strong>\u305d\u308c\u3067\u3082\u3084\u308a\u304d\u308c\u305f<\/strong><\/h2>\n\n\n\n<p>AI\u3068\u4f1a\u8a71\u3060\u3051\u3067\u65e5\u5e38\u5229\u7528\u3067\u304d\u308b\u30a2\u30d7\u30ea\u3092\u3064\u304f\u308b\u3053\u3068\u306f\u3051\u3063\u3053\u3046\u5927\u5909\u3060\u3063\u305f\u304c\u3001\u305d\u308c\u3067\u3082\u3084\u308a\u304d\u308c\u305f\u3002<br>\u666e\u6bb5\u306f\u30b3\u30fc\u30c7\u30a3\u30f3\u30b0\u3057\u3066\u3044\u306a\u3044\u4eba(=\u50d5)\u304c\u3001\u4e00\u9031\u9593\u306e\u4f1a\u8a71\u3060\u3051\u3067\u30a2\u30d7\u30ea\u3092\u4f5c\u308c\u308b\u6642\u4ee3\u3002<br>\u8ab2\u984c\u3084\u54c1\u8cea\u554f\u984c\u306f\u3042\u308b\u3068\u3057\u3066\u3082\u3001\u307e\u3060\u307e\u3060\u3053\u308c\u304b\u3089AI\u304c\u8ce2\u304f\u306a\u308b\u3068\u60f3\u5b9a\u3059\u308b\u3068\u3001\u4eba\u9593\u304c\u30b3\u30fc\u30c7\u30a3\u30f3\u30b0\u3059\u308b\u5fc5\u8981\u6027\u306f\u7121\u304f\u306a\u3063\u3066\u3044\u304f\u3088\u3046\u306b\u601d\u3048\u308b\u3002\u6a5f\u68b0\u8a9e\u3084\u30a2\u30bb\u30f3\u30d6\u30ea\u8a00\u8a9e\u3092\u5168\u30a8\u30f3\u30b8\u30cb\u30a2\u304c\u7406\u89e3\u3059\u308b\u5fc5\u8981\u6027\u304c\u306a\u304f\u306a\u3063\u305f\u3088\u3046\u306b\u3001\u4eba\u9593\u304c\u30b3\u30fc\u30c7\u30a3\u30f3\u30b0\u3092\u3059\u308b\u5fc5\u8981\u6027\u3082\u306a\u304f\u306a\u3063\u3066\u3044\u304d\u305d\u3046\u3002\uff08\u4e00\u65b9\u3067\u307e\u3060\u30a4\u30de\u30a4\u30c1\u3068\u611f\u3058\u308b\u3053\u3068\u3082\u591a\u304f\u3069\u306e\u304f\u3089\u3044\u5148\u306a\u306e\u304b\u306f\u672a\u77e5\u6570\u3002\u7126\u3063\u3066\u7d76\u671b\u3057\u3059\u304e\u306a\u3044\u3088\u3046\u306b\uff09<\/p>\n\n\n\n<h2 class=\"wp-block-heading\"><strong>\u30a8\u30f3\u30b8\u30cb\u30a2\u306f\u4e0a\u6d41\u3078\u5411\u304b\u3048\uff01<\/strong><\/h2>\n\n\n\n<p>\u3082\u306e\u3065\u304f\u308a\u306e\u672c\u8cea\u306f\u304d\u3063\u3068\u5909\u308f\u3089\u306a\u3044\u3002\u30af\u30ea\u30a8\u30a4\u30c6\u30a3\u30d6\u3001\u30a2\u30a4\u30c7\u30a3\u30a2\u3001\u30c7\u30b6\u30a4\u30f3\u3001\u5c0e\u5165\u904b\u7528\u3001\u7d99\u7d9a\u6539\u5584\u306a\u3069\u3001\u3088\u308a\u4e0a\u6d41\u3092\u4eba\u9593\u304c\u62c5\u5f53\u3057AI\u3092\u99c6\u4f7f\u3057\u3066\u5727\u5012\u7684\u306a\u30b9\u30d4\u30fc\u30c9\u3067\u3082\u306e\u3065\u304f\u308a\u304c\u51fa\u6765\u308b\u3088\u3046\u306b\u306a\u3063\u305f\u3002<\/p>\n\n\n\n<p>\u50d5\u304c\u4eba\u751f\u3067\u4e00\u756a\u30b3\u30fc\u30c7\u30a3\u30f3\u30b0\u3057\u3066\u3044\u305f\u306e\u306f20\u5e74\u524d\u5b66\u751f\u30d0\u30a4\u30c8\u30d7\u30ed\u30b0\u30e9\u30de\u6642\u4ee3\u3002\u30b3\u30fc\u30c7\u30a3\u30f3\u30b0\u81ea\u4f53\u304c\u697d\u3057\u304b\u3063\u305f\u300222\u6b73\u3067\u8d77\u696d\u3057\u3066\u304b\u3089\u306f\u3001\u4ed5\u69d8\u3084\u30c7\u30b6\u30a4\u30f3\u3092\u8003\u3048\u305f\u308a\u3001\u30b5\u30fc\u30d3\u30b9\u3084\u30a4\u30d9\u30f3\u30c8\u3092\u7acb\u3061\u4e0a\u3052\u305f\u308a\u3001\u63a1\u7528\u3084\u30ab\u30eb\u30c1\u30e3\u30fc\u306b\u30b3\u30df\u30c3\u30c8\u3057\u305f\u308a\u3068\u3001\u30d3\u30b8\u30cd\u30b9\u306e\u4e0a\u6d41\u3067\u5fd9\u3057\u304f\u3082\u306e\u3065\u304f\u308a\u3092\u8ffd\u6c42\u3057\u3066\u304d\u305f\u3002<\/p>\n\n\n\n<p>\u30b3\u30fc\u30c7\u30a3\u30f3\u30b0\u30fb\u6280\u8853\u5168\u822c\u3092\u7406\u89e3\u3059\u308b\u4eba\u304c\u30d3\u30b8\u30cd\u30b9\u3067\u679c\u305f\u305b\u308b\u5f79\u5272\u306f\u5927\u304d\u3044\u3002AI\u306b\u904e\u5ea6\u306b\u671f\u5f85\u305b\u305a\u80fd\u529b\u306e\u30ae\u30ea\u30ae\u30ea\u6700\u5927\u9650\u3092\u5f15\u304d\u51fa\u3059\u3002\u30b9\u30de\u30db\u306e\u6d78\u900f\u306e\u3088\u3046\u306b10\u5e74\u304b\u3051\u3066\u5f90\u3005\u306b\u3001\uff08\u73fe\u5834\u611f\u3067\u306f\u4e00\u6c17\u306b\uff09\u9032\u3081\u308b\u3002<\/p>\n\n\n\n<p>\u30a8\u30f3\u30b8\u30cb\u30a2\u306f\u4e0a\u6d41\u306b\u3069\u3093\u3069\u3093\u5411\u304b\u3044\u30d3\u30b8\u30cd\u30b9\u306b\u5411\u304d\u5408\u304a\u3046\u3002\u30af\u30ea\u30a8\u30a4\u30c6\u30a3\u30d6\u3084\u30b3\u30df\u30e5\u30cb\u30b1\u30fc\u30b7\u30e7\u30f3\u3092\u5927\u5207\u306b\u3057\u3088\u3046\u3002\u3053\u308c\u304b\u3089\u3082\u306e\u3065\u304f\u308a\u304c\u3069\u306e\u3088\u3046\u306b\u5909\u5316\u3059\u308b\u304b\u306f\u8ab0\u306b\u3082\u308f\u304b\u3089\u306a\u3044\u3002\u305d\u3057\u3066\u89e3\u6c7a\u3059\u3079\u304d\u8ab2\u984c\u306f\u76db\u308a\u3060\u304f\u3055\u3093\u3002\u653f\u6cbb\u3001\u8ca7\u56f0\u3001\u6226\u4e89\u3001\u5065\u5eb7\u3001\u8001\u5f8c\u3001\u5730\u65b9\u3001\u5b64\u72ec\u3001\u6559\u80b2\u3001\u7121\u6c17\u529b\u30fb\u30fb\u7b49\u3005\u8ab2\u984c\u5c71\u7a4d\u3002AI\u3092\u4f7f\u3063\u305f\u30a8\u30f3\u30b8\u30cb\u30a2\u30ea\u30f3\u30b0\u3067\u5168\u90e8\u89e3\u6c7a\u3057\u3066\u3044\u3053\u3046\uff01\u3000\u5909\u5316\u306e\u5148\u7aef\u306b\u7acb\u3061\u7d9a\u3051\u3001\u4eba\u3005\u306e\u5e78\u798f\u306b\u8ca2\u732e\u3057\u3066\u3044\u304f\u306e\u307f\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\"><strong>\u304a\u307e\u3051\uff09\u30bd\u30fc\u30b9\u30b3\u30fc\u30c9\u3092\u5927\u516c\u958b<\/strong>\uff01<\/h2>\n\n\n\n<p>\u30bd\u30fc\u30b9\u30b3\u30fc\u30c9\u306e\u30e1\u30a4\u30f3\u90e8\u5206(index.js)\u3092\u516c\u958b\u3002\u3053\u308c\u304cAI\u304c\u66f8\u3044\u305f5000\u884c\u3060\uff01<\/p>\n\n\n\n<p>\u5197\u9577\u3060\u3063\u305f\u308a\u80a5\u5927\u5316\u3057\u3066\u308b\u3051\u3069\u3054\u611b\u5b0c\uff01<\/p>\n\n\n\n<div class=\"hcb_wrap\"><pre class=\"prism line-numbers lang-js\" data-lang=\"JavaScript\" data-show-lang=\"1\"><code>const express = require(&#39;express&#39;);\nconst dotenv = require(&#39;dotenv&#39;);\nconst path = require(&#39;path&#39;);\nconst cors = require(&#39;cors&#39;);\nconst { WebClient } = require(&#39;@slack\/web-api&#39;);\nconst axios = require(&#39;axios&#39;);\nconst { Anthropic } = require(&#39;@anthropic-ai\/sdk&#39;);\nconst OpenAI = require(&#39;openai&#39;);\nconst fs = require(&#39;fs&#39;);\nconst sqlite3 = require(&#39;sqlite3&#39;).verbose();\nconst multer = require(&#39;multer&#39;);\nconst { google } = require(&#39;googleapis&#39;);\nconst { JWT } = require(&#39;google-auth-library&#39;);\n\n\/\/ \u65e5\u672c\u6642\u9593\u306e\u73fe\u5728\u65e5\u6642\u3092\u53d6\u5f97\u3059\u308b\u30d8\u30eb\u30d1\u30fc\u95a2\u6570\nfunction getJSTDate() {\n  const now = new Date();\n  return new Date(now.getTime() + (9 * 60 * 60 * 1000)); \/\/ UTC+9\u6642\u9593\uff08\u65e5\u672c\u6642\u9593\uff09\n}\n\n\/\/ \u65e5\u672c\u6642\u9593\u306e\u73fe\u5728\u65e5\u6642\u304b\u3089YYYY-MM-DD\u5f62\u5f0f\u306e\u65e5\u4ed8\u6587\u5b57\u5217\u3092\u53d6\u5f97\nfunction getJSTDateString() {\n  return getJSTDate().toISOString().split(&#39;T&#39;)[0];\n}\n\n\/\/ \u8a2d\u5b9a\u30d5\u30a1\u30a4\u30eb\u306e\u30d1\u30b9\nconst SETTINGS_FILE_PATH = path.join(__dirname, &#39;settings.json&#39;);\n\n\/\/ \u30c7\u30d5\u30a9\u30eb\u30c8\u306e\u8a2d\u5b9a\u5024\nconst defaultSettings = {\n  channelId: process.env.SLACK_CHANNEL_ID || &#39;&#39;,\n  botToken: process.env.SLACK_BOT_TOKEN || &#39;&#39;\n};\n\n\/\/ \u30c7\u30d5\u30a9\u30eb\u30c8AI\u5206\u6790\u30d7\u30ed\u30f3\u30d7\u30c8\nconst defaultAiAnalysisPrompt = &quot;\u3042\u306a\u305f\u306f\u512a\u308c\u305f\u4eba\u6750\u5206\u6790\u306e\u30d7\u30ed\u30d5\u30a7\u30c3\u30b7\u30e7\u30ca\u30eb\u3067\u3059\u3002Slack\u306e\u4f1a\u8a71\u30c7\u30fc\u30bf\u304b\u3089\u30e6\u30fc\u30b6\u30fc\u306e\u624d\u80fd\u3084\u5f37\u307f\u3001\u72ec\u81ea\u6027\u3092\u898b\u51fa\u3057\u3001\u5177\u4f53\u7684\u306a\u30a8\u30d4\u30bd\u30fc\u30c9\u3092\u4ea4\u3048\u306a\u304c\u3089\u89e3\u8aac\u3059\u308b\u80fd\u529b\u306b\u512a\u308c\u3066\u3044\u307e\u3059\u3002\u5206\u6790\u3067\u306f\u3001\u30e6\u30fc\u30b6\u30fc\u56fa\u6709\u306e\u7279\u5fb4\u3092\u5177\u4f53\u7684\u306b\u6307\u6458\u3057\u3001\u5b9f\u969b\u306e\u4f1a\u8a71\u304b\u3089\u306e\u4f8b\u3092\u5f15\u7528\u3057\u306a\u304c\u3089\u3001\u305d\u306e\u4eba\u3060\u3051\u304c\u6301\u3064\u4fa1\u5024\u3092\u660e\u3089\u304b\u306b\u3057\u3066\u304f\u3060\u3055\u3044\u3002\u6279\u5224\u7684\u306a\u5185\u5bb9\u306f\u4e00\u5207\u542b\u3081\u305a\u3001\u660e\u308b\u304f\u524d\u5411\u304d\u306a\u89b3\u70b9\u304b\u3089\u305d\u306e\u4eba\u306e\u9b45\u529b\u3092\u6700\u5927600\u6587\u5b57\u3067\u8868\u73fe\u3057\u3066\u304f\u3060\u3055\u3044\u3002&quot;;\n\n\/\/ \u73fe\u5728\u306eAI\u5206\u6790\u30d7\u30ed\u30f3\u30d7\u30c8\uff08\u521d\u671f\u5024\u306f\u30c7\u30d5\u30a9\u30eb\u30c8\uff09\nlet currentAiAnalysisPrompt = defaultAiAnalysisPrompt;\n\n\/\/ \u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u8d77\u52d5\u6642\u306b\u8a2d\u5b9a\u30d5\u30a1\u30a4\u30eb\u3092\u8aad\u307f\u8fbc\u3080\ntry {\n  const settings = getSettings();\n  if (settings.aiAnalysisPrompt) {\n    currentAiAnalysisPrompt = settings.aiAnalysisPrompt;\n    console.log(&#39;\u8a2d\u5b9a\u30d5\u30a1\u30a4\u30eb\u304b\u3089AI\u5206\u6790\u30d7\u30ed\u30f3\u30d7\u30c8\u3092\u8aad\u307f\u8fbc\u307f\u307e\u3057\u305f&#39;);\n  }\n} catch (error) {\n  console.error(&#39;\u8a2d\u5b9a\u30d5\u30a1\u30a4\u30eb\u306e\u8aad\u307f\u8fbc\u307f\u306b\u5931\u6557\u3057\u307e\u3057\u305f:&#39;, error);\n}\n\n\/\/ SQLite\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u3092\u521d\u671f\u5316\nconst db = new sqlite3.Database(&#39;.\/slack_data.db&#39;);\n\n\/\/ \u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u306e\u30c6\u30fc\u30d6\u30eb\u3092\u521d\u671f\u5316\nfunction initDatabase() {\n  console.log(&#39;\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u3092\u521d\u671f\u5316\u3057\u3066\u3044\u307e\u3059...&#39;);\n  \n  \/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u30c6\u30fc\u30d6\u30eb\u306e\u4f5c\u6210\n  db.run(`\n    CREATE TABLE IF NOT EXISTS slack_messages (\n      id INTEGER PRIMARY KEY AUTOINCREMENT,\n      message_id TEXT,\n      user_id TEXT,\n      channel_id TEXT,\n      channel_name TEXT,\n      text TEXT,\n      message_type TEXT,\n      timestamp TEXT,\n      thread_ts TEXT,\n      is_reply INTEGER DEFAULT 0,\n      reactions TEXT,\n      created_at DATETIME DEFAULT CURRENT_TIMESTAMP\n    )\n  `);\n  \n  \/\/ \u6700\u7d42\u540c\u671f\u65e5\u30c6\u30fc\u30d6\u30eb\u306e\u4f5c\u6210\n  db.run(`\n    CREATE TABLE IF NOT EXISTS last_sync_date (\n      id INTEGER PRIMARY KEY AUTOINCREMENT,\n      last_sync_date DATE UNIQUE,\n      last_sync_timestamp INTEGER,\n      created_at DATETIME DEFAULT CURRENT_TIMESTAMP\n    )\n  `);\n  \n  \/\/ \u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u30c6\u30fc\u30d6\u30eb\u306e\u4f5c\u6210\n  db.run(`\n    CREATE TABLE IF NOT EXISTS slack_thread_replies (\n      id INTEGER PRIMARY KEY AUTOINCREMENT,\n      message_id TEXT,\n      parent_message_id TEXT,\n      user_id TEXT,\n      channel_id TEXT,\n      text TEXT,\n      timestamp TEXT,\n      reply_id TEXT,\n      thread_ts TEXT,\n      channel_name TEXT,\n      created_at DATETIME DEFAULT CURRENT_TIMESTAMP\n    )\n  `);\n  \n  \/\/ \u30b9\u30bf\u30f3\u30d7\uff08\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\uff09\u30c6\u30fc\u30d6\u30eb\u306e\u4f5c\u6210\n  db.run(`\n    CREATE TABLE IF NOT EXISTS slack_reactions (\n      id INTEGER PRIMARY KEY AUTOINCREMENT,\n      message_id TEXT,\n      channel_id TEXT,\n      user_id TEXT,\n      name TEXT,\n      timestamp TEXT,\n      created_at DATETIME DEFAULT CURRENT_TIMESTAMP\n    )\n  `);\n  \n  \/\/ \u30e6\u30fc\u30b6\u30fc\u30c6\u30fc\u30d6\u30eb\u306e\u4f5c\u6210\n  db.run(`\n    CREATE TABLE IF NOT EXISTS slack_users (\n      id INTEGER PRIMARY KEY AUTOINCREMENT,\n      user_id TEXT UNIQUE,\n      name TEXT,\n      real_name TEXT,\n      display_name TEXT,\n      avatar TEXT,\n      is_bot INTEGER DEFAULT 0,\n      created_at DATETIME DEFAULT CURRENT_TIMESTAMP\n    )\n  `);\n  \n  \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u30c6\u30fc\u30d6\u30eb\u306e\u4f5c\u6210\n  db.run(`\n    CREATE TABLE IF NOT EXISTS slack_channels (\n      id INTEGER PRIMARY KEY AUTOINCREMENT,\n      channel_id TEXT UNIQUE,\n      name TEXT,\n      created_at DATETIME DEFAULT CURRENT_TIMESTAMP\n    )\n  `);\n  \n  \/\/ \u30e6\u30fc\u30b6\u30fc\u5206\u6790\u7d50\u679c\u30c6\u30fc\u30d6\u30eb\u306e\u4f5c\u6210\n  db.run(`\n    CREATE TABLE IF NOT EXISTS user_analysis (\n      id INTEGER PRIMARY KEY AUTOINCREMENT,\n      user_id TEXT,\n      analysis_json TEXT,\n      analyzed_at DATETIME DEFAULT CURRENT_TIMESTAMP\n    )\n  `);\n  \n  \/\/ \u4eca\u65e5\u306e\u30a4\u30c1\u63a8\u3057braver\u30c6\u30fc\u30d6\u30eb\u306e\u4f5c\u6210\n  db.run(`\n    CREATE TABLE IF NOT EXISTS daily_pickup_user (\n      id INTEGER PRIMARY KEY AUTOINCREMENT,\n      user_id TEXT,\n      name TEXT,\n      real_name TEXT,\n      display_name TEXT,\n      avatar TEXT,\n      message_count INTEGER,\n      reply_count INTEGER,\n      attendance_count INTEGER,\n      office_count INTEGER,\n      report_count INTEGER,\n      sent_reaction_count INTEGER,\n      received_reaction_count INTEGER,\n      analysis TEXT,\n      pickup_date DATE DEFAULT CURRENT_DATE,\n      created_at DATETIME DEFAULT CURRENT_TIMESTAMP\n    )\n  `);\n  \n  console.log(&#39;\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u521d\u671f\u5316\u304c\u5b8c\u4e86\u3057\u307e\u3057\u305f&#39;);\n}\n\n\/\/ \u30a2\u30d7\u30ea\u8d77\u52d5\u6642\u306b\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u521d\u671f\u5316\ninitDatabase();\n\n\/\/ \u74b0\u5883\u5909\u6570\u306e\u8aad\u307f\u8fbc\u307f\uff08\u5fc5\u305a\u6700\u521d\u306b\u5b9f\u884c\uff09\ndotenv.config();\n\n\/\/ .renv\u30d5\u30a1\u30a4\u30eb\u304b\u3089\u3082\u74b0\u5883\u5909\u6570\u3092\u8aad\u307f\u8fbc\u307f\ntry {\n  if (fs.existsSync(&#39;.renv&#39;)) {\n    const renvContent = fs.readFileSync(&#39;.renv&#39;, &#39;utf8&#39;);\n    const renvLines = renvContent.split(&#39;\\n&#39;);\n    \n    renvLines.forEach(line =&gt; {\n      const trimmedLine = line.trim();\n      if (trimmedLine && !trimmedLine.startsWith(&#39;#&#39;) && trimmedLine.includes(&#39;=&#39;)) {\n        const [key, ...valueParts] = trimmedLine.split(&#39;=&#39;);\n        const value = valueParts.join(&#39;=&#39;); \/\/ API\u30ad\u30fc\u306b\u306f\u3057\u3070\u3057\u3070 = \u304c\u542b\u307e\u308c\u308b\u3053\u3068\u304c\u3042\u308b\u305f\u3081\n        if (key && value) {\n          process.env[key.trim()] = value.trim();\n          console.log(`\u74b0\u5883\u5909\u6570\u3092.renv\u304b\u3089\u8aad\u307f\u8fbc\u307f: ${key.trim()} = \u8a2d\u5b9a\u6e08\u307f`);\n        }\n      }\n    });\n    console.log(&#39;.renv\u30d5\u30a1\u30a4\u30eb\u304b\u3089\u306e\u74b0\u5883\u5909\u6570\u8aad\u307f\u8fbc\u307f\u304c\u5b8c\u4e86\u3057\u307e\u3057\u305f&#39;);\n  }\n} catch (error) {\n  console.error(&#39;.renv\u30d5\u30a1\u30a4\u30eb\u306e\u8aad\u307f\u8fbc\u307f\u306b\u5931\u6557\u3057\u307e\u3057\u305f:&#39;, error.message);\n}\n\n\/\/ Express\u30a2\u30d7\u30ea\u306e\u521d\u671f\u5316\nconst app = express();\nconst port = process.env.PORT || 3000;\n\n\/\/ Claude API\u8a2d\u5b9a\u3092\u53d6\u5f97\nconst anthropic = new Anthropic({\n  apiKey: process.env.ANTHROPIC_API_KEY\n});\n\n\/\/ OpenAI API\u8a2d\u5b9a\u3092\u53d6\u5f97\nlet openai = null;\nif (process.env.OPENAI_API_KEY) {\n  openai = new OpenAI({\n    apiKey: process.env.OPENAI_API_KEY,\n  });\n  console.log(&#39;OpenAI API\u30ad\u30fc\u304c\u8a2d\u5b9a\u3055\u308c\u307e\u3057\u305f&#39;);\n} else {\n  console.log(&#39;OpenAI API\u30ad\u30fc\u304c\u672a\u8a2d\u5b9a\u3067\u3059&#39;);\n}\n\n\/\/ CORS\u3092\u6709\u52b9\u5316\napp.use(cors({\n  origin: true,\n  methods: [&#39;GET&#39;, &#39;POST&#39;, &#39;PUT&#39;, &#39;DELETE&#39;, &#39;OPTIONS&#39;],\n  allowedHeaders: [&#39;Content-Type&#39;, &#39;Authorization&#39;],\n  credentials: true\n}));\n\n\/\/ JSON\u30dc\u30c7\u30a3\u30d1\u30fc\u30b5\u30fc\u3092\u4f7f\u7528\napp.use(express.json());\napp.use(express.urlencoded({ extended: true }));\n\n\/\/ \u9759\u7684\u30d5\u30a1\u30a4\u30eb\u306e\u63d0\u4f9b\napp.use(express.static(path.join(__dirname, &#39;public&#39;)));\n\n\/\/ Slack API\u8a2d\u5b9a\u3092\u74b0\u5883\u5909\u6570\u304b\u3089\u53d6\u5f97\nconst slackToken = process.env.SLACK_BOT_TOKEN;\n\n\/\/ \u8907\u6570\u306e\u30c1\u30e3\u30f3\u30cd\u30ebID\u3092\u30b5\u30dd\u30fc\u30c8\nlet slackChannelIds = [];\n\n\/\/ \u74b0\u5883\u5909\u6570\u304b\u3089\u30c1\u30e3\u30f3\u30cd\u30ebID\u3092\u53d6\u5f97\nif (process.env.SLACK_CHANNEL_IDS) {\n  \/\/ \u30ab\u30f3\u30de\u533a\u5207\u308a\u306e\u8907\u6570\u30c1\u30e3\u30f3\u30cd\u30ebID\u3092\u30b5\u30dd\u30fc\u30c8\n  slackChannelIds = process.env.SLACK_CHANNEL_IDS.split(&#39;,&#39;).map(id =&gt; id.trim()).filter(id =&gt; id);\n  console.log(`\u74b0\u5883\u5909\u6570\u304b\u3089\u8907\u6570\u30c1\u30e3\u30f3\u30cd\u30ebID\u3092\u8aad\u307f\u8fbc\u307f\u307e\u3057\u305f: ${slackChannelIds.length}\u4ef6`);\n} else if (process.env.SLACK_CHANNEL_ID) {\n  \/\/ \u5f8c\u65b9\u4e92\u63db\u6027\u306e\u305f\u3081\u306b\u5358\u4e00\u30c1\u30e3\u30f3\u30cd\u30ebID\u3082\u30b5\u30dd\u30fc\u30c8\n  slackChannelIds = [process.env.SLACK_CHANNEL_ID];\n  console.log(`\u74b0\u5883\u5909\u6570\u304b\u3089\u5358\u4e00\u30c1\u30e3\u30f3\u30cd\u30ebID\u3092\u8aad\u307f\u8fbc\u307f\u307e\u3057\u305f: ${process.env.SLACK_CHANNEL_ID}`);\n  \n  \/\/ SLACK_CHANNEL_IDS\u304c\u306a\u3051\u308c\u3070\u8a2d\u5b9a\uff08\u4e00\u8cab\u6027\u306e\u305f\u3081\uff09\n  process.env.SLACK_CHANNEL_IDS = process.env.SLACK_CHANNEL_ID;\n  console.log(&#39;\u74b0\u5883\u5909\u6570SLACK_CHANNEL_IDS\u3092\u8a2d\u5b9a\u3057\u307e\u3057\u305f:&#39;, process.env.SLACK_CHANNEL_IDS);\n}\n\n\/\/ \u30b0\u30ed\u30fc\u30d0\u30eb\u5909\u6570\u3068\u3057\u3066\u3082\u8a2d\u5b9a\nglobal.slackChannelIds = slackChannelIds;\n\n\/\/ \u8a2d\u5b9a\u306e\u691c\u8a3c\nconsole.log(&#39;\u74b0\u5883\u5909\u6570: SLACK_BOT_TOKEN&#39;, slackToken ? &#39;\u8a2d\u5b9a\u6e08\u307f&#39; : &#39;\u672a\u8a2d\u5b9a&#39;);\nconsole.log(&#39;\u74b0\u5883\u5909\u6570: SLACK_CHANNEL_IDS&#39;, slackChannelIds.length &gt; 0 ? `\u8a2d\u5b9a\u6e08\u307f (${slackChannelIds.join(&#39;, &#39;)})` : &#39;\u672a\u8a2d\u5b9a&#39;);\nconsole.log(&#39;\u30b0\u30ed\u30fc\u30d0\u30eb\u5909\u6570: slackChannelIds&#39;, global.slackChannelIds ? `\u8a2d\u5b9a\u6e08\u307f (${global.slackChannelIds.join(&#39;, &#39;)})` : &#39;\u672a\u8a2d\u5b9a&#39;);\n\n\/\/ WebClient\u306e\u521d\u671f\u5316\nconst slackClient = new WebClient(slackToken);\n\n\/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u5206\u6790\u3059\u308b\u95a2\u6570 (\u65e7\u540d:analyzeTextWithGPT)\nasync function analyzeTextWithAI(messages, userMap) {\n  try {\n    console.log(&#39;\u5206\u6790\u958b\u59cb: \u5143\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u6570:&#39;, messages.length);\n    \n    \/\/ 400\u6587\u5b57\u4ee5\u4e0a\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u306e\u307f\u3092\u30d5\u30a3\u30eb\u30bf\u30ea\u30f3\u30b0\n    const longMessages = messages.filter(msg =&gt; {\n      return msg.text && msg.text.length &gt;= 400;\n    });\n    \n    console.log(`400\u6587\u5b57\u4ee5\u4e0a\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u30d5\u30a3\u30eb\u30bf\u30ea\u30f3\u30b0: ${longMessages.length}\u4ef6\/${messages.length}\u4ef6`);\n    \n    \/\/ \u5206\u6790\u5bfe\u8c61\u304c\u30bc\u30ed\u306e\u5834\u5408\u306f\u65e9\u671f\u30ea\u30bf\u30fc\u30f3\n    if (longMessages.length === 0) {\n      console.log(&#39;400\u6587\u5b57\u4ee5\u4e0a\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u304c\u898b\u3064\u304b\u308a\u307e\u305b\u3093\u3002\u5206\u6790\u3092\u30b9\u30ad\u30c3\u30d7\u3057\u307e\u3059\u3002&#39;);\n      return {\n        summary: { mainTopics: [], overallSummary: &quot;400\u6587\u5b57\u4ee5\u4e0a\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u304c\u898b\u3064\u304b\u308a\u307e\u305b\u3093\u3067\u3057\u305f\u3002&quot; },\n        trendAnalysis: { keywords: [], topicChanges: &quot;\u5206\u6790\u306a\u3057&quot;, communicationPattern: &quot;\u5206\u6790\u306a\u3057&quot; },\n        userAnalysis: {},\n        insightsAndRecommendations: [&quot;400\u6587\u5b57\u4ee5\u4e0a\u306e\u8a73\u7d30\u306a\u6295\u7a3f\u304c\u3042\u308a\u307e\u305b\u3093\u3002&quot;]\n      };\n    }\n    \n    \/\/ \u4ee5\u964d\u306e\u51e6\u7406\u3067\u306f messages \u306e\u4ee3\u308f\u308a\u306b longMessages \u3092\u4f7f\u7528\n    \n    \/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u30e6\u30fc\u30b6\u30fc\u3054\u3068\u306b\u30b0\u30eb\u30fc\u30d7\u5316\n    const messagesByUser = {};\n    longMessages.forEach(msg =&gt; {\n      const userId = msg.user;\n      const userName = userMap[userId]?.name || &#39;Unknown User&#39;;\n      \n      if (!messagesByUser[userName]) {\n        messagesByUser[userName] = [];\n      }\n      \n      messagesByUser[userName].push({\n        text: msg.text,\n        ts: msg.ts,\n        timestamp: new Date(parseFloat(msg.ts) * 1000).toLocaleString()\n      });\n    });\n    \n    \/\/ \u6642\u7cfb\u5217\u9806\u306b\u30bd\u30fc\u30c8\u3055\u308c\u305f\u30e1\u30c3\u30bb\u30fc\u30b8\n    const sortedMessages = longMessages\n      .map(msg =&gt; ({\n        text: msg.text,\n        user: userMap[msg.user]?.name || &#39;Unknown User&#39;,\n        timestamp: new Date(parseFloat(msg.ts) * 1000)\n      }))\n      .sort((a, b) =&gt; a.timestamp - b.timestamp);\n    \n    \/\/ \u5206\u6790\u7528\u306e\u30d7\u30ed\u30f3\u30d7\u30c8\u3092\u4f5c\u6210\uff08\u57fa\u672c\u60c5\u5831\uff09\n    const basePrompt = `\n\u3042\u306a\u305f\u306fSlack\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u5206\u6790\u3059\u308b\u5c02\u9580\u5bb6\u3067\u3059\u3002\u4ee5\u4e0b\u306eSlack\u30c1\u30e3\u30f3\u30cd\u30eb\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u306e\u3046\u3061\u3001400\u6587\u5b57\u4ee5\u4e0a\u306e\u8a73\u7d30\u306a\u6295\u7a3f\u306e\u307f\u3092\u5206\u6790\u3057\u3066\u3001\u6b21\u306e\u70b9\u3092JSON\u5f62\u5f0f\u3067\u8a73\u7d30\u306b\u56de\u7b54\u3057\u3066\u304f\u3060\u3055\u3044\uff1a\n\n1. \u5168\u4f53\u30b5\u30de\u30ea\u30fc: \u9577\u6587\u6295\u7a3f\u306e\u5185\u5bb9\u3068\u50be\u5411\uff08\u4e3b\u8981\u30c8\u30d4\u30c3\u30af\u3001\u8b70\u8ad6\u306e\u6d41\u308c\u3001\u4e3b\u306a\u7d50\u8ad6\u306a\u3069\uff09\n2. \u9577\u6587\u6295\u7a3f\u306e\u50be\u5411\u5206\u6790: \u30ad\u30fc\u30ef\u30fc\u30c9\u3001\u30c8\u30d4\u30c3\u30af\u306e\u7279\u5fb4\u3001\u8868\u73fe\u306e\u7279\u5fb4\u306a\u3069\n3. \u30e6\u30fc\u30b6\u30fc\u5206\u6790: \u9577\u6587\u3092\u6295\u7a3f\u3059\u308b\u30e6\u30fc\u30b6\u30fc\u306e\u30b3\u30df\u30e5\u30cb\u30b1\u30fc\u30b7\u30e7\u30f3\u30b9\u30bf\u30a4\u30eb\u3001\u8208\u5473\u95a2\u5fc3\u3001\u5f79\u5272\u306a\u3069\n\n\u30c7\u30fc\u30bf\u60c5\u5831:\n- \u9577\u6587\u30e1\u30c3\u30bb\u30fc\u30b8\u6570: ${longMessages.length}\u4ef6\uff08\u5168${messages.length}\u4ef6\u4e2d\uff09\n- \u9577\u6587\u6295\u7a3f\u30e6\u30fc\u30b6\u30fc\u6570: ${Object.keys(messagesByUser).length}\u540d\n- \u671f\u9593: ${sortedMessages.length &gt; 0 ? \n    `${sortedMessages[0].timestamp.toLocaleString()} \u304b\u3089 ${sortedMessages[sortedMessages.length-1].timestamp.toLocaleString()}` : \n    &#39;\u4e0d\u660e&#39;}\n\n\u4ee5\u4e0b\u306f\u5404\u30e6\u30fc\u30b6\u30fc\u306e\u9577\u6587\u30e1\u30c3\u30bb\u30fc\u30b8\u4f8b\u3067\u3059\uff1a\n`;\n    \n    \/\/ \u30e6\u30fc\u30b6\u30fc\u3054\u3068\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u4f8b\u3092\u8ffd\u52a0\uff08\u5404\u30e6\u30fc\u30b6\u30fc\u6700\u59275\u3064\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\uff09\n    let userMessagesPrompt = &#39;&#39;;\n    Object.entries(messagesByUser).forEach(([userName, messages]) =&gt; {\n      userMessagesPrompt += `\u3010${userName}\u3011\\n`;\n      messages\n        .slice(0, 5) \/\/ \u6700\u59275\u3064\u307e\u3067\n        .forEach((msg, i) =&gt; {\n          userMessagesPrompt += `${i+1}. ${msg.timestamp}: ${msg.text}\\n`;\n        });\n      userMessagesPrompt += &#39;\\n&#39;;\n    });\n    \n    \/\/ \u6642\u7cfb\u5217\u5206\u6790\u7528\u306e\u30d7\u30ed\u30f3\u30d7\u30c8\uff08\u6700\u521d\u304b\u3089\u6700\u5f8c\u307e\u306710\u4ef6\u7a0b\u5ea6\u30d4\u30c3\u30af\u30a2\u30c3\u30d7\uff09\n    let timelinePrompt = &#39;\\n\u3010\u6642\u7cfb\u5217\u3067\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u30b5\u30f3\u30d7\u30eb\u3011\\n&#39;;\n    const timelineSamples = [];\n    \n    if (sortedMessages.length &lt;= 10) {\n      \/\/ 10\u4ef6\u4ee5\u4e0b\u306a\u3089\u5168\u3066\n      timelineSamples.push(...sortedMessages);\n    } else {\n      \/\/ \u6700\u521d2\u4ef6\n      timelineSamples.push(sortedMessages[0], sortedMessages[1]);\n      \n      \/\/ \u4e2d\u9593\u90e8\u5206\u304b\u3089\u6570\u4ef6\n      const middleStartIdx = Math.floor(sortedMessages.length * 0.3);\n      const middleEndIdx = Math.floor(sortedMessages.length * 0.7);\n      const step = Math.floor((middleEndIdx - middleStartIdx) \/ 4);\n      \n      for (let i = middleStartIdx; i &lt;= middleEndIdx; i += step) {\n        if (timelineSamples.length &lt; 8) {\n          timelineSamples.push(sortedMessages[i]);\n        }\n      }\n      \n      \/\/ \u6700\u5f8c2\u4ef6\n      timelineSamples.push(\n        sortedMessages[sortedMessages.length - 2],\n        sortedMessages[sortedMessages.length - 1]\n      );\n    }\n    \n    timelineSamples.forEach((msg, i) =&gt; {\n      timelinePrompt += `${msg.timestamp.toLocaleString()}: ${msg.user}: ${msg.text}\\n`;\n    });\n    \n    \/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u5168\u4f53\u304b\u3089\u4e3b\u8981\u30ad\u30fc\u30ef\u30fc\u30c9\u3092\u62bd\u51fa\n    const allMessages = longMessages.map(msg =&gt; msg.text).join(&#39; &#39;).toLowerCase();\n    const keywords = extractKeywords(allMessages);\n    const keywordsText = keywords.join(&#39;, &#39;);\n    \n    \/\/ \u30ec\u30b9\u30dd\u30f3\u30b9\u5f62\u5f0f\u306e\u30ac\u30a4\u30c9\u30e9\u30a4\u30f3\n    const responseFormatPrompt = `\n\u5206\u6790\u7d50\u679c\u306f\u4ee5\u4e0b\u306eJSON\u5f62\u5f0f\u3067\u8fd4\u3057\u3066\u304f\u3060\u3055\u3044\uff1a\n\n{\n  &quot;summary&quot;: {\n    &quot;mainTopics&quot;: [&quot;\u30c8\u30d4\u30c3\u30af1&quot;, &quot;\u30c8\u30d4\u30c3\u30af2&quot;],\n    &quot;overallSummary&quot;: &quot;\u9577\u6587\u6295\u7a3f\u5168\u4f53\u306e\u8981\u7d04...&quot;,\n    &quot;keyConclusions&quot;: [&quot;\u7d50\u8ad61&quot;, &quot;\u7d50\u8ad62&quot;]\n  },\n  &quot;trendAnalysis&quot;: {\n    &quot;keywords&quot;: [&quot;\u30ad\u30fc\u30ef\u30fc\u30c91&quot;, &quot;\u30ad\u30fc\u30ef\u30fc\u30c92&quot;],\n    &quot;topicChanges&quot;: &quot;\u30c8\u30d4\u30c3\u30af\u306e\u5909\u5316\u306b\u3064\u3044\u3066\u306e\u8aac\u660e...&quot;,\n    &quot;communicationPattern&quot;: &quot;\u9577\u6587\u6295\u7a3f\u306e\u30d1\u30bf\u30fc\u30f3\u306e\u8aac\u660e...&quot;\n  },\n  &quot;userAnalysis&quot;: {\n    &quot;\u30e6\u30fc\u30b6\u30fc\u540d1&quot;: {\n      &quot;communicationStyle&quot;: &quot;\u9577\u6587\u6295\u7a3f\u306e\u30b9\u30bf\u30a4\u30eb\u306e\u8aac\u660e...&quot;,\n      &quot;topics&quot;: [&quot;\u8208\u5473\u306e\u3042\u308b\u30c8\u30d4\u30c3\u30af1&quot;, &quot;\u8208\u5473\u306e\u3042\u308b\u30c8\u30d4\u30c3\u30af2&quot;],\n      &quot;role&quot;: &quot;\u4f1a\u8a71\u5185\u3067\u306e\u5f79\u5272...&quot;,\n      &quot;personality&quot;: &quot;\u6027\u683c\u3084\u30b3\u30df\u30e5\u30cb\u30b1\u30fc\u30b7\u30e7\u30f3\u7279\u6027...&quot;\n    },\n    &quot;\u30e6\u30fc\u30b6\u30fc\u540d2&quot;: {\n      &quot;communicationStyle&quot;: &quot;\u9577\u6587\u6295\u7a3f\u306e\u30b9\u30bf\u30a4\u30eb\u306e\u8aac\u660e...&quot;,\n      &quot;topics&quot;: [&quot;\u8208\u5473\u306e\u3042\u308b\u30c8\u30d4\u30c3\u30af1&quot;, &quot;\u8208\u5473\u306e\u3042\u308b\u30c8\u30d4\u30c3\u30af2&quot;],\n      &quot;role&quot;: &quot;\u4f1a\u8a71\u5185\u3067\u306e\u5f79\u5272...&quot;,\n      &quot;personality&quot;: &quot;\u6027\u683c\u3084\u30b3\u30df\u30e5\u30cb\u30b1\u30fc\u30b7\u30e7\u30f3\u7279\u6027...&quot;\n    }\n  },\n  &quot;insightsAndRecommendations&quot;: [\n    &quot;\u6d1e\u5bdf1...&quot;,\n    &quot;\u6d1e\u5bdf2...&quot;\n  ]\n}\n\n\u30ad\u30fc\u30ef\u30fc\u30c9\u62bd\u51fa\u7d50\u679c: ${keywordsText}\n\u203b\u3053\u306e\u5206\u6790\u3067\u306f\u3001400\u6587\u5b57\u4ee5\u4e0a\u306e\u9577\u6587\u6295\u7a3f\u306e\u307f\u3092\u5bfe\u8c61\u3068\u3057\u3066\u3044\u307e\u3059\u3002\n`;\n\n    \/\/ \u30c8\u30d4\u30c3\u30af\u306e\u5909\u5316\u3092\u691c\u51fa\n    const topicChanges = detectTopicChanges(sortedMessages);\n    \n    \/\/ \u30a2\u30af\u30c6\u30a3\u30d3\u30c6\u30a3\u30d1\u30bf\u30fc\u30f3\u3092\u5206\u6790\n    const activityPattern = analyzeActivityPattern(sortedMessages);\n    \n    \/\/ \u9001\u4fe1\u524d\u306b\u30c8\u30fc\u30af\u30f3\u6570\u3092\u30c1\u30a7\u30c3\u30af\u3057\u3001\u5fc5\u8981\u306b\u5fdc\u3058\u3066\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u5727\u7e2e\n    const compressForTokenLimit = (userMessagesPrompt, timelinePrompt, maxTotalTokens = 100000) =&gt; {\n      \/\/ \u7c21\u6613\u7684\u306a\u30c8\u30fc\u30af\u30f3\u6570\u8a08\u7b97\u95a2\u6570\n      const estimateTokens = (text) =&gt; {\n        const japaneseChars = text.match(\/[\\u3000-\\u303f\\u3040-\\u309f\\u30a0-\\u30ff\\uff00-\\uff9f\\u4e00-\\u9faf]\/g) || [];\n        const otherChars = text.length - japaneseChars.length;\n        return Math.ceil(japaneseChars.length \/ 2 + otherChars \/ 4);\n      };\n      \n      \/\/ \u5404\u90e8\u5206\u306e\u30c8\u30fc\u30af\u30f3\u6570\u3092\u8a08\u7b97\n      const userMsgTokens = estimateTokens(userMessagesPrompt);\n      const timelineTokens = estimateTokens(timelinePrompt);\n      const baseAndFormatTokens = 3000; \/\/ \u30b7\u30b9\u30c6\u30e0\u30d7\u30ed\u30f3\u30d7\u30c8\u3068\u69cb\u9020\u90e8\u5206\u306e\u304a\u304a\u3088\u3069\u306e\u30c8\u30fc\u30af\u30f3\u6570\n      \n      const totalEstimatedTokens = userMsgTokens + timelineTokens + baseAndFormatTokens;\n      console.log(`\u63a8\u5b9a\u30c8\u30fc\u30af\u30f3\u6570: \u30e6\u30fc\u30b6\u30fc\u30e1\u30c3\u30bb\u30fc\u30b8=${userMsgTokens}, \u30bf\u30a4\u30e0\u30e9\u30a4\u30f3=${timelineTokens}, \u5408\u8a08=${totalEstimatedTokens}`);\n      \n      \/\/ \u30c8\u30fc\u30af\u30f3\u6570\u304c\u4e0a\u9650\u3092\u8d85\u3048\u3066\u3044\u306a\u3051\u308c\u3070\u305d\u306e\u307e\u307e\u8fd4\u3059\n      if (totalEstimatedTokens &lt;= maxTotalTokens) {\n        return { userMessagesPrompt, timelinePrompt, totalEstimatedTokens };\n      }\n      \n      console.log(`\u30c8\u30fc\u30af\u30f3\u6570\u304c\u4e0a\u9650(${maxTotalTokens})\u3092\u8d85\u3048\u3066\u3044\u307e\u3059\u3002\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u5727\u7e2e\u3057\u307e\u3059...`);\n      \n      \/\/ \u524a\u6e1b\u304c\u5fc5\u8981\u306a\u91cf\u3092\u8a08\u7b97\n      const excessTokens = totalEstimatedTokens - maxTotalTokens;\n      const reduceRatio = 1 - (excessTokens \/ (userMsgTokens + timelineTokens));\n      \n      \/\/ \u30e6\u30fc\u30b6\u30fc\u30e1\u30c3\u30bb\u30fc\u30b8\u3068\u30bf\u30a4\u30e0\u30e9\u30a4\u30f3\u3092\u6bd4\u4f8b\u914d\u5206\u3067\u524a\u6e1b\n      let compressedUserMsgs = userMessagesPrompt;\n      let compressedTimeline = timelinePrompt;\n      \n      \/\/ 1. \u307e\u305a\u30bf\u30a4\u30e0\u30e9\u30a4\u30f3\u3092\u524a\u6e1b\uff08\u6700\u3082\u524a\u6e1b\u3057\u3084\u3059\u3044\uff09\n      if (timelineTokens &gt; 1000) {\n        \/\/ \u30bf\u30a4\u30e0\u30e9\u30a4\u30f3\u3092\u5927\u5e45\u306b\u524a\u6e1b\n        const timelineLines = timelinePrompt.split(&#39;\\n&#39;);\n        const keptLines = Math.max(5, Math.floor(timelineLines.length * reduceRatio));\n        \n        \/\/ \u91cd\u8981\u306a\u30dd\u30a4\u30f3\u30c8\u306e\u307f\u3092\u6b8b\u3059\uff08\u5148\u982d\u3001\u4e2d\u9593\u3001\u672b\u5c3e\uff09\n        const selectedLines = [];\n        selectedLines.push(timelineLines[0]); \/\/ \u30d8\u30c3\u30c0\u30fc\n        \n        if (timelineLines.length &gt; 2) {\n          selectedLines.push(timelineLines[1]); \/\/ \u6700\u521d\u306e\u30b5\u30f3\u30d7\u30eb\n          \n          \/\/ \u4e2d\u9593\u306e\u30b5\u30f3\u30d7\u30eb\u304b\u3089\u6570\u500b\uff08\u5747\u7b49\u9593\u9694\uff09\n          const step = Math.floor((timelineLines.length - 3) \/ (keptLines - 3));\n          for (let i = 0; i &lt; keptLines - 3; i++) {\n            const idx = 1 + step * (i + 1);\n            if (idx &lt; timelineLines.length - 1) {\n              selectedLines.push(timelineLines[idx]);\n            }\n          }\n          \n          selectedLines.push(timelineLines[timelineLines.length - 1]); \/\/ \u6700\u5f8c\u306e\u30b5\u30f3\u30d7\u30eb\n        }\n        \n        compressedTimeline = selectedLines.join(&#39;\\n&#39;);\n        console.log(`\u30bf\u30a4\u30e0\u30e9\u30a4\u30f3\u3092 ${timelineLines.length} \u884c\u304b\u3089 ${selectedLines.length} \u884c\u306b\u524a\u6e1b\u3057\u307e\u3057\u305f`);\n      }\n      \n      \/\/ 2. \u6b21\u306b\u30e6\u30fc\u30b6\u30fc\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u524a\u6e1b\uff08\u5404\u30e6\u30fc\u30b6\u30fc\u306e\u30b5\u30f3\u30d7\u30eb\u6570\u3092\u6e1b\u3089\u3059\uff09\n      if (estimateTokens(compressedTimeline) + estimateTokens(compressedUserMsgs) + baseAndFormatTokens &gt; maxTotalTokens) {\n        \/\/ \u30e6\u30fc\u30b6\u30fc\u3054\u3068\u306e\u30d6\u30ed\u30c3\u30af\u306b\u5206\u5272\n        const userBlocks = compressedUserMsgs.split(\/\u3010(.+?)\u3011\/).filter(block =&gt; block.trim());\n        const userSections = [];\n        \n        for (let i = 0; i &lt; userBlocks.length; i += 2) {\n          if (i + 1 &lt; userBlocks.length) {\n            userSections.push({\n              user: userBlocks[i],\n              content: userBlocks[i+1]\n            });\n          }\n        }\n        \n        console.log(`\u30e6\u30fc\u30b6\u30fc\u30bb\u30af\u30b7\u30e7\u30f3\u6570: ${userSections.length}`);\n        \n        \/\/ \u5404\u30e6\u30fc\u30b6\u30fc\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u30b5\u30f3\u30d7\u30eb\u3092\u524a\u6e1b\n        const compressedSections = userSections.map(section =&gt; {\n          const messages = section.content.split(\/\\d+\\. \/).filter(msg =&gt; msg.trim());\n          \n          if (messages.length &lt;= 2) return `\u3010${section.user}\u3011\\n${section.content}`;\n          \n          \/\/ \u30b5\u30f3\u30d7\u30eb\u6570\u30922\u3064\u3060\u3051\u306b\u524a\u6e1b\n          const firstSample = messages[1]; \/\/ 0\u756a\u76ee\u306f\u7a7a\u304b\u524d\u306e\u30bb\u30d1\u30ec\u30fc\u30bf\u306e\u53ef\u80fd\u6027\n          const lastSample = messages[messages.length - 1];\n          \n          return `\u3010${section.user}\u3011\\n1. ${firstSample}\\n2. ${lastSample}\\n`;\n        });\n        \n        compressedUserMsgs = compressedSections.join(&#39;\\n&#39;);\n        console.log(&#39;\u5404\u30e6\u30fc\u30b6\u30fc\u306e\u30b5\u30f3\u30d7\u30eb\u6570\u3092\u524a\u6e1b\u3057\u307e\u3057\u305f&#39;);\n      }\n      \n      \/\/ 3. \u305d\u308c\u3067\u3082\u591a\u3059\u304e\u308b\u5834\u5408\u306f\u30e6\u30fc\u30b6\u30fc\u6570\u3092\u524a\u6e1b\n      if (estimateTokens(compressedTimeline) + estimateTokens(compressedUserMsgs) + baseAndFormatTokens &gt; maxTotalTokens) {\n        \/\/ \u30e6\u30fc\u30b6\u30fc\u3054\u3068\u306e\u30d6\u30ed\u30c3\u30af\u306b\u5206\u5272\n        const userBlocks = compressedUserMsgs.split(\/\u3010(.+?)\u3011\/).filter(block =&gt; block.trim());\n        const userSections = [];\n        \n        for (let i = 0; i &lt; userBlocks.length; i += 2) {\n          if (i + 1 &lt; userBlocks.length) {\n            userSections.push({\n              user: userBlocks[i],\n              content: userBlocks[i+1]\n            });\n          }\n        }\n        \n        \/\/ \u30e6\u30fc\u30b6\u30fc\u6570\u3092\u524a\u6e1b\uff08\u30e1\u30c3\u30bb\u30fc\u30b8\u6570\u306e\u591a\u3044\u4e0a\u4f4d\u306e\u307f\u3092\u6b8b\u3059\uff09\n        const maxUsers = Math.max(5, Math.floor(userSections.length * 0.5));\n        const keptSections = userSections.slice(0, maxUsers);\n        \n        compressedUserMsgs = keptSections.map(section =&gt; `\u3010${section.user}\u3011\\n${section.content}`).join(&#39;\\n&#39;);\n        console.log(`\u30e6\u30fc\u30b6\u30fc\u6570\u3092 ${userSections.length} \u304b\u3089 ${keptSections.length} \u306b\u524a\u6e1b\u3057\u307e\u3057\u305f`);\n      }\n      \n      \/\/ \u6700\u7d42\u7684\u306a\u30c8\u30fc\u30af\u30f3\u6570\u3092\u518d\u8a08\u7b97\n      const finalTokens = estimateTokens(compressedTimeline) + estimateTokens(compressedUserMsgs) + baseAndFormatTokens;\n      console.log(`\u5727\u7e2e\u5f8c\u306e\u63a8\u5b9a\u30c8\u30fc\u30af\u30f3\u6570: ${finalTokens}`);\n      \n      return { \n        userMessagesPrompt: compressedUserMsgs, \n        timelinePrompt: compressedTimeline,\n        totalEstimatedTokens: finalTokens\n      };\n    };\n\n    \/\/ OpenAI\u3092\u4f7f\u3063\u3066\u5206\u6790\n    if (openai && process.env.OPENAI_API_KEY) {\n      try {\n        console.log(&#39;OpenAI\u3092\u4f7f\u7528\u3057\u3066\u5206\u6790\u3092\u958b\u59cb\u3057\u307e\u3059...&#39;);\n        console.log(&#39;OpenAI API\u30ad\u30fc\u78ba\u8a8d:&#39;, process.env.OPENAI_API_KEY ? &#39;API\u30ad\u30fc\u8a2d\u5b9a\u6e08\u307f&#39; : &#39;API\u30ad\u30fc\u672a\u8a2d\u5b9a&#39;);\n        console.log(&#39;\u30e1\u30c3\u30bb\u30fc\u30b8\u6570:&#39;, longMessages.length);\n        console.log(&#39;\u30e6\u30fc\u30b6\u30fc\u6570:&#39;, Object.keys(messagesByUser).length);\n        \n        if (!process.env.OPENAI_API_KEY) {\n          throw new Error(&#39;OpenAI API\u30ad\u30fc\u304c\u8a2d\u5b9a\u3055\u308c\u3066\u3044\u307e\u305b\u3093&#39;);\n        }\n        \n        \/\/ \u30b7\u30b9\u30c6\u30e0\u30d7\u30ed\u30f3\u30d7\u30c8\u3068\u30e6\u30fc\u30b6\u30fc\u30d7\u30ed\u30f3\u30d7\u30c8\u3092\u4f5c\u6210\n        const systemPrompt = &quot;\u3042\u306a\u305f\u306f\u512a\u79c0\u306a\u4eba\u6750\u5206\u6790\u306e\u30d7\u30ed\u3067\u3059\u3002Slack\u306e\u4f1a\u8a71\u304b\u3089\u5177\u4f53\u7684\u306a\u30a8\u30d4\u30bd\u30fc\u30c9\u3001\u6848\u4ef6\u540d\u3001\u4eba\u540d\u3092\u4f7f\u3063\u3066600\u6587\u5b57\u7a0b\u5ea6\u3067\u7c21\u6f54\u306b\u5206\u6790\u3057\u3066\u304f\u3060\u3055\u3044\u30021) \u95a2\u308f\u3063\u305f\u5177\u4f53\u7684\u306a\u30d7\u30ed\u30b8\u30a7\u30af\u30c8\u30012) \u767a\u8a00\u304b\u3089\u898b\u3048\u308b\u5c02\u9580\u6027\u30013) \u30c1\u30fc\u30e0\u8ca2\u732e\u306e\u5b9f\u4f8b\u30014) \u30b3\u30df\u30e5\u30cb\u30b1\u30fc\u30b7\u30e7\u30f3\u306e\u7279\u5fb4\u7684\u306a\u30d1\u30bf\u30fc\u30f3\u30015) \u5177\u4f53\u7684\u306a\u5f37\u307f\u3092\u5fc5\u305a\u542b\u3081\u3066\u304f\u3060\u3055\u3044\u3002\u7c21\u6f54\u306a\u6587\u4f53\u3067\u3001\u5177\u4f53\u7684\u306a\u30a8\u30d4\u30bd\u30fc\u30c9\u3092\u6839\u62e0\u306b\u3057\u305f\u5206\u6790\u3092\u884c\u3063\u3066\u304f\u3060\u3055\u3044\u3002&quot;;\n        \n        \/\/ \u30c8\u30fc\u30af\u30f3\u6570\u306e\u524a\u6e1b\u51e6\u7406\u3092\u9069\u7528\n        const { userMessagesPrompt: compressedUserMsgs, timelinePrompt: compressedTimeline, totalEstimatedTokens } = \n          compressForTokenLimit(userMessagesPrompt, timelinePrompt, 8000); \/\/ GPT-4\u306f\u5165\u529b\u30c8\u30fc\u30af\u30f3\u5236\u9650\u304c\u4f4e\u3044\u305f\u3081\u8abf\u6574\n        \n        \/\/ \u6700\u7d42\u7684\u306a\u30d7\u30ed\u30f3\u30d7\u30c8\u3092\u4f5c\u6210\uff08\u5727\u7e2e\u30d0\u30fc\u30b8\u30e7\u30f3\u3092\u4f7f\u7528\uff09\n        const finalPrompt = basePrompt + compressedUserMsgs + compressedTimeline + responseFormatPrompt;\n        \n        console.log(&#39;OpenAI\u30ea\u30af\u30a8\u30b9\u30c8\u3092\u9001\u4fe1\u3057\u307e\u3059...&#39;);\n        console.log(`\u63a8\u5b9a\u30c8\u30fc\u30af\u30f3\u6570: ${totalEstimatedTokens}`);\n        \n        try {\n          \/\/ OpenAI API\u3092\u547c\u3073\u51fa\u3059\uff08\u30c8\u30fc\u30af\u30f3\u5236\u9650\u306b\u914d\u616e\uff09\n          const response = await openai.chat.completions.create({\n            model: &quot;gpt-4o&quot;,\n            temperature: 0.2,\n            max_tokens: 3000,\n            messages: [\n              {\n                role: &quot;system&quot;,\n                content: &quot;\u3042\u306a\u305f\u306f\u512a\u79c0\u306a\u4eba\u6750\u5206\u6790\u306e\u30d7\u30ed\u3067\u3059\u3002Slack\u306e\u4f1a\u8a71\u304b\u3089\u5177\u4f53\u7684\u306a\u30a8\u30d4\u30bd\u30fc\u30c9\u3001\u6848\u4ef6\u540d\u3001\u4eba\u540d\u3092\u4f7f\u3063\u3066600\u6587\u5b57\u7a0b\u5ea6\u3067\u7c21\u6f54\u306b\u5206\u6790\u3057\u3066\u304f\u3060\u3055\u3044\u30021) \u95a2\u308f\u3063\u305f\u5177\u4f53\u7684\u306a\u30d7\u30ed\u30b8\u30a7\u30af\u30c8\u30012) \u767a\u8a00\u304b\u3089\u898b\u3048\u308b\u5c02\u9580\u6027\u30013) \u30c1\u30fc\u30e0\u8ca2\u732e\u306e\u5b9f\u4f8b\u30014) \u30b3\u30df\u30e5\u30cb\u30b1\u30fc\u30b7\u30e7\u30f3\u306e\u7279\u5fb4\u7684\u306a\u30d1\u30bf\u30fc\u30f3\u30015) \u5177\u4f53\u7684\u306a\u5f37\u307f\u3092\u5fc5\u305a\u542b\u3081\u3066\u304f\u3060\u3055\u3044\u3002\u7c21\u6f54\u306a\u6587\u4f53\u3067\u3001\u5177\u4f53\u7684\u306a\u30a8\u30d4\u30bd\u30fc\u30c9\u3092\u6839\u62e0\u306b\u3057\u305f\u5206\u6790\u3092\u884c\u3063\u3066\u304f\u3060\u3055\u3044\u3002&quot;\n              },\n              {\n                role: &quot;user&quot;,\n                content: finalPrompt\n              }\n            ]\n          });\n          \n          console.log(&#39;OpenAI\u304b\u3089\u306e\u5fdc\u7b54\u3092\u53d7\u4fe1\u3057\u307e\u3057\u305f&#39;);\n          \n          \/\/ OpenAI API\u304b\u3089\u306e\u5fdc\u7b54\u3092\u51e6\u7406\n          if (response && response.choices && response.choices.length &gt; 0) {\n            const aiResponse = response.choices[0].message.content;\n            console.log(&#39;\u5fdc\u7b54\u5185\u5bb9\u306e\u4e00\u90e8:&#39;, aiResponse.substring(0, 100) + &#39;...&#39;);\n            \n            \/\/ JSON\u3092\u30d1\u30fc\u30b9\n            let analysisResult;\n            try {\n              \/\/ \u524d\u5f8c\u306e\u4f59\u5206\u306a\u30c6\u30ad\u30b9\u30c8\u3092\u524a\u9664\u3057\u3066\u7d14\u7c8b\u306aJSON\u3060\u3051\u3092\u62bd\u51fa\n              let jsonString = aiResponse.trim();\n              \n              \/\/ \u3082\u3057```json```\u3067\u56f2\u307e\u308c\u3066\u3044\u305f\u3089\u3001\u305d\u306e\u90e8\u5206\u3060\u3051\u3092\u62bd\u51fa\n              const jsonBlockMatch = jsonString.match(\/```(?:json)?\\s*([\\s\\S]*?)\\s*```\/);\n              if (jsonBlockMatch && jsonBlockMatch[1]) {\n                jsonString = jsonBlockMatch[1].trim();\n                console.log(&#39;\u30de\u30fc\u30af\u30c0\u30a6\u30f3\u30b3\u30fc\u30c9\u30d6\u30ed\u30c3\u30af\u304b\u3089\u7d14\u7c8b\u306aJSON\u3092\u62bd\u51fa\u3057\u307e\u3057\u305f&#39;);\n              }\n              \n              \/\/ JSON\u89e3\u6790\n              analysisResult = JSON.parse(jsonString);\n              console.log(&#39;JSON\u89e3\u6790\u6210\u529f&#39;);\n              \n              return analysisResult;\n            } catch (parseError) {\n              console.error(&#39;JSON\u89e3\u6790\u30a8\u30e9\u30fc:&#39;, parseError.message);\n              \n              \/\/ JSON\u4fee\u6b63\u3092\u8a66\u307f\u308b\n              try {\n                const fixedJsonString = cleanupAllProblematicCharacters(aiResponse);\n                analysisResult = JSON.parse(fixedJsonString);\n                console.log(&#39;JSON\u4fee\u6b63\u5f8c\u306e\u89e3\u6790\u6210\u529f&#39;);\n                return analysisResult;\n              } catch (fixError) {\n                console.error(&#39;JSON\u4fee\u6b63\u5f8c\u3082\u89e3\u6790\u5931\u6557:&#39;, fixError.message);\n                \/\/ \u7c21\u6613\u5206\u6790\u306b\u5207\u308a\u66ff\u3048\n                return createSimpleAnalysis(longMessages, messagesByUser, keywords, topicChanges, activityPattern);\n              }\n            }\n          } else {\n            console.warn(&#39;OpenAI\u306e\u5fdc\u7b54\u5f62\u5f0f\u304c\u60f3\u5b9a\u5916\u3067\u3059&#39;);\n            return createSimpleAnalysis(longMessages, messagesByUser, keywords, topicChanges, activityPattern);\n          }\n        } catch (apiError) {\n          console.error(&#39;OpenAI API\u547c\u3073\u51fa\u3057\u30a8\u30e9\u30fc:&#39;, apiError);\n          return createSimpleAnalysis(longMessages, messagesByUser, keywords, topicChanges, activityPattern);\n        }\n      } catch (error) {\n        console.warn(&#39;OpenAI API\u30a8\u30e9\u30fc:&#39;, error.message);\n        return createSimpleAnalysis(longMessages, messagesByUser, keywords, topicChanges, activityPattern);\n      }\n    }\n    \n    \/\/ OpenAI\u304c\u4f7f\u3048\u306a\u3044\u5834\u5408\u306e\u30d5\u30a9\u30fc\u30eb\u30d0\u30c3\u30af\n    console.warn(&#39;OpenAI\u304c\u4f7f\u7528\u3067\u304d\u306a\u3044\u305f\u3081\u3001\u7c21\u6613\u5206\u6790\u3092\u4f7f\u7528\u3057\u307e\u3059&#39;);\n    return createSimpleAnalysis(longMessages, messagesByUser, keywords, topicChanges, activityPattern);\n  } catch (error) {\n    console.error(&#39;\u30c6\u30ad\u30b9\u30c8\u5206\u6790\u4e2d\u306b\u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u307e\u3057\u305f:&#39;, error);\n    return {\n      error: &#39;\u30c6\u30ad\u30b9\u30c8\u5206\u6790\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39;,\n      details: error.message\n    };\n  }\n}\n\n\/\/ Claude\u304c\u4f7f\u3048\u306a\u3044\u5834\u5408\u307e\u305f\u306f\u5931\u6557\u3057\u305f\u5834\u5408\u306e\u30d5\u30a9\u30fc\u30eb\u30d0\u30c3\u30af\u5206\u6790\u95a2\u6570\nfunction createSimpleAnalysis(messages, messagesByUser, keywords, topicChanges, activityPattern) {\n  console.warn(&#39;\u7c21\u6613\u5206\u6790\u3092\u4f7f\u7528\u3057\u307e\u3059\uff08Claude\u306f\u4f7f\u7528\u3057\u307e\u305b\u3093\uff09&#39;);\n  const analysisResult = {\n    channelInsights: &quot;\u3053\u306e\u30c1\u30e3\u30f3\u30cd\u30eb\u306f\u4e3b\u306b\u30d7\u30ed\u30b8\u30a7\u30af\u30c8\u95a2\u9023\u306e\u6280\u8853\u7684\u306a\u8b70\u8ad6\u3068\u66f4\u65b0\u60c5\u5831\u306e\u5171\u6709\u306b\u4f7f\u7528\u3055\u308c\u3066\u3044\u307e\u3059\u3002\u4f1a\u8a71\u306f\u30d7\u30ed\u30d5\u30a7\u30c3\u30b7\u30e7\u30ca\u30eb\u3067\u3042\u308a\u306a\u304c\u3089\u3082\u30ab\u30b8\u30e5\u30a2\u30eb\u306a\u30c8\u30fc\u30f3\u3067\u3059\u3002&quot;,\n    channelSummary: {\n      mainKeywords: keywords,\n      topicChanges: topicChanges,\n      activityPattern: activityPattern\n    },\n    userAnalysis: {}\n  };\n  \n  \/\/ \u5404\u30e6\u30fc\u30b6\u30fc\u306e\u30c6\u30ad\u30b9\u30c8\u3092\u7c21\u6613\u5206\u6790\n  Object.entries(messagesByUser).forEach(([userName, messages]) =&gt; {\n    const joinedText = messages.map(m =&gt; m.text).join(&#39; &#39;).toLowerCase();\n    let style = &quot;&quot;;\n    let interests = &quot;&quot;;\n    let personality = &quot;&quot;;\n    \n    \/\/ \u7c21\u6613\u7684\u306a\u30ad\u30fc\u30ef\u30fc\u30c9\u5206\u6790\n    if (joinedText.includes(&#39;\u554f\u984c&#39;) || joinedText.includes(&#39;\u30a8\u30e9\u30fc&#39;)) {\n      style += &quot;\u554f\u984c\u89e3\u6c7a\u6307\u5411\u3001&quot;;\n      personality += &quot;\u5206\u6790\u7684\u3001&quot;;\n    }\n    if (joinedText.includes(&#39;\u601d\u3046&#39;) || joinedText.includes(&#39;\u8003\u3048&#39;)) {\n      style += &quot;\u5185\u7701\u7684\u3001&quot;;\n      personality += &quot;\u601d\u616e\u6df1\u3044\u3001&quot;;\n    }\n    if (joinedText.includes(&#39;!&#39;) || joinedText.includes(&#39;\uff1f&#39;)) {\n      style += &quot;\u8cea\u554f\u304c\u591a\u3044\u3001&quot;;\n      personality += &quot;\u597d\u5947\u5fc3\u65fa\u76db\u3001&quot;;\n    }\n    if (joinedText.includes(&#39;api&#39;) || joinedText.includes(&#39;\u30b3\u30fc\u30c9&#39;)) {\n      interests += &quot;\u30d7\u30ed\u30b0\u30e9\u30df\u30f3\u30b0\u3001&quot;;\n    }\n    if (joinedText.includes(&#39;slack&#39;) || joinedText.includes(&#39;\u30e1\u30c3\u30bb\u30fc\u30b8&#39;)) {\n      interests += &quot;\u30b3\u30df\u30e5\u30cb\u30b1\u30fc\u30b7\u30e7\u30f3\u30c4\u30fc\u30eb\u3001&quot;;\n    }\n    \n    analysisResult.userAnalysis[userName] = {\n      communicationStyle: style ? style.slice(0, -1) : &quot;\u30c7\u30fc\u30bf\u4e0d\u8db3\u3067\u5224\u65ad\u3067\u304d\u307e\u305b\u3093&quot;,\n      topics: interests ? interests.slice(0, -1).split(&#39;\u3001&#39;) : [&quot;\u30c7\u30fc\u30bf\u4e0d\u8db3&quot;],\n      role: messages.length &gt; (messages.length \/ Object.keys(messagesByUser).length) * 1.5 \n          ? &quot;\u4e3b\u8981\u306a\u767a\u8a00\u8005&quot; \n          : &quot;\u901a\u5e38\u306e\u53c2\u52a0\u8005&quot;,\n      personality: personality ? personality.slice(0, -1) : &quot;\u30c7\u30fc\u30bf\u4e0d\u8db3\u3067\u5224\u65ad\u3067\u304d\u307e\u305b\u3093&quot;,\n      messageCount: messages.length,\n      averageLength: Math.round(messages.map(m =&gt; m.text).join(&#39; &#39;).length \/ messages.length)\n    };\n  });\n  \n  return analysisResult;\n}\n\n\/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u304b\u3089\u30ad\u30fc\u30ef\u30fc\u30c9\u3092\u62bd\u51fa\u3059\u308b\u88dc\u52a9\u95a2\u6570\nfunction extractKeywords(text) {\n  \/\/ \u7c21\u6613\u7684\u306a\u30ad\u30fc\u30ef\u30fc\u30c9\u62bd\u51fa\uff08\u5b9f\u969b\u306e\u5b9f\u88c5\u3067\u306f\u3082\u3063\u3068\u6d17\u7df4\u3055\u308c\u305f\u65b9\u6cd5\u3092\u4f7f\u7528\uff09\n  const commonWords = [&#39;\u3066&#39;, &#39;\u306b&#39;, &#39;\u306f&#39;, &#39;\u3092&#39;, &#39;\u306e&#39;, &#39;\u304c&#39;, &#39;\u3068&#39;, &#39;\u3067\u3059&#39;, &#39;\u307e\u3059&#39;, &#39;\u3057\u305f&#39;, &#39;\u304b\u3089&#39;, &#39;\u306a\u3044&#39;, &#39;\u3044\u308b&#39;, &#39;\u3042\u308b&#39;, &#39;\u308c\u308b&#39;];\n  const words = text.replace(\/[.,\\\/#!$%\\^&\\*;:{}=\\-_`~()]\/g, &#39;&#39;).toLowerCase().split(\/\\s+\/);\n  \n  const wordCounts = {};\n  words.forEach(word =&gt; {\n    if (word.length &gt; 1 && !commonWords.includes(word)) {\n      wordCounts[word] = (wordCounts[word] || 0) + 1;\n    }\n  });\n  \n  return Object.entries(wordCounts)\n    .sort((a, b) =&gt; b[1] - a[1])\n    .slice(0, 5)\n    .map(entry =&gt; entry[0]);\n}\n\n\/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u304b\u3089\u30c8\u30d4\u30c3\u30af\u306e\u5909\u5316\u3092\u691c\u51fa\u3059\u308b\u88dc\u52a9\u95a2\u6570\nfunction detectTopicChanges(messages) {\n  if (messages.length &lt; 3) return &quot;\u30e1\u30c3\u30bb\u30fc\u30b8\u304c\u5c11\u306a\u3044\u305f\u3081\u3001\u30c8\u30d4\u30c3\u30af\u306e\u5909\u5316\u3092\u691c\u51fa\u3067\u304d\u307e\u305b\u3093&quot;;\n  \n  \/\/ \u7c21\u6613\u7684\u306a\u30c8\u30d4\u30c3\u30af\u5909\u5316\u691c\u51fa\n  const firstThirdKeywords = extractKeywords(messages.slice(0, Math.floor(messages.length \/ 3)).map(m =&gt; m.text).join(&#39; &#39;));\n  const lastThirdKeywords = extractKeywords(messages.slice(-Math.floor(messages.length \/ 3)).map(m =&gt; m.text).join(&#39; &#39;));\n  \n  \/\/ \u30ad\u30fc\u30ef\u30fc\u30c9\u306e\u9055\u3044\u3092\u78ba\u8a8d\n  const differentKeywords = lastThirdKeywords.filter(kw =&gt; !firstThirdKeywords.includes(kw));\n  \n  if (differentKeywords.length &gt; 0) {\n    return `\u4f1a\u8a71\u306e\u9014\u4e2d\u3067\u30c8\u30d4\u30c3\u30af\u304c\u5909\u5316\u3057\u305f\u53ef\u80fd\u6027\u304c\u3042\u308a\u307e\u3059\u3002\u6700\u8fd1\u306e\u30ad\u30fc\u30ef\u30fc\u30c9: ${differentKeywords.join(&#39;, &#39;)}`;\n  } else {\n    return &quot;\u4f1a\u8a71\u5168\u4f53\u3092\u901a\u3057\u3066\u4e00\u8cab\u3057\u305f\u30c8\u30d4\u30c3\u30af\u304c\u7dad\u6301\u3055\u308c\u3066\u3044\u307e\u3059&quot;;\n  }\n}\n\n\/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u306e\u30a2\u30af\u30c6\u30a3\u30d3\u30c6\u30a3\u30d1\u30bf\u30fc\u30f3\u3092\u5206\u6790\u3059\u308b\u88dc\u52a9\u95a2\u6570\nfunction analyzeActivityPattern(messages) {\n  if (messages.length &lt; 3) return &quot;\u30e1\u30c3\u30bb\u30fc\u30b8\u304c\u5c11\u306a\u3044\u305f\u3081\u3001\u30a2\u30af\u30c6\u30a3\u30d3\u30c6\u30a3\u30d1\u30bf\u30fc\u30f3\u3092\u5206\u6790\u3067\u304d\u307e\u305b\u3093&quot;;\n  \n  \/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u306e\u6642\u9593\u9593\u9694\u3092\u8a08\u7b97\n  const intervals = [];\n  for (let i = 1; i &lt; messages.length; i++) {\n    intervals.push(messages[i].timestamp - messages[i-1].timestamp);\n  }\n  \n  const avgInterval = intervals.reduce((sum, val) =&gt; sum + val, 0) \/ intervals.length;\n  \n  \/\/ \u6700\u3082\u6d3b\u767a\u306a\u53c2\u52a0\u8005\u3092\u7279\u5b9a\n  const userCounts = {};\n  messages.forEach(msg =&gt; {\n    userCounts[msg.user] = (userCounts[msg.user] || 0) + 1;\n  });\n  \n  const mostActiveUser = Object.entries(userCounts)\n    .sort((a, b) =&gt; b[1] - a[1])[0][0];\n  \n  \/\/ \u8fd4\u5374\u3059\u308b\u30d1\u30bf\u30fc\u30f3\u60c5\u5831\n  return {\n    averageTimeBetweenMessages: Math.round(avgInterval) + &quot;\u79d2&quot;,\n    mostActiveUser: mostActiveUser,\n    conversationPace: avgInterval &lt; 60 ? &quot;\u975e\u5e38\u306b\u6d3b\u767a\u306a\u4f1a\u8a71&quot; : \n                     avgInterval &lt; 300 ? &quot;\u6d3b\u767a\u306a\u4f1a\u8a71&quot; : \n                     avgInterval &lt; 1800 ? &quot;\u901a\u5e38\u306e\u30da\u30fc\u30b9\u306e\u4f1a\u8a71&quot; : &quot;\u9593\u9694\u306e\u7a7a\u3044\u305f\u4f1a\u8a71&quot;\n  };\n}\n\n\n\/\/ \u30b7\u30f3\u30d7\u30eb\u30c6\u30b9\u30c8\u30da\u30fc\u30b8\uff08\u30c7\u30d0\u30c3\u30b0\u7528\uff09\napp.get(&#39;\/simple&#39;, (req, res) =&gt; {\n  res.send(`\n    &lt;!DOCTYPE html&gt;\n    &lt;html&gt;\n    &lt;head&gt;\n      &lt;title&gt;Slack \u30c6\u30b9\u30c8&lt;\/title&gt;\n      &lt;style&gt;\n        body { font-family: Arial; max-width: 800px; margin: 0 auto; padding: 20px; background: #f4f4f4; }\n        button { padding: 10px; background: #4CAF50; color: white; border: none; cursor: pointer; margin: 5px; }\n        #messages { margin-top: 20px; border: 1px solid #ddd; padding: 15px; min-height: 200px; background: white; }\n      &lt;\/style&gt;\n    &lt;\/head&gt;\n    &lt;body&gt;\n      &lt;h1&gt;Slack \u30c6\u30b9\u30c8&lt;\/h1&gt;\n      &lt;button id=&quot;getSlackMessages&quot;&gt;Slack\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u53d6\u5f97&lt;\/button&gt;\n      &lt;div id=&quot;messages&quot;&gt;\u3053\u3053\u306b\u7d50\u679c\u304c\u8868\u793a\u3055\u308c\u307e\u3059...&lt;\/div&gt;\n\n      &lt;script&gt;\n        document.getElementById(&#39;getSlackMessages&#39;).addEventListener(&#39;click&#39;, async () =&gt; {\n          const messagesDiv = document.getElementById(&#39;messages&#39;);\n          messagesDiv.innerHTML = &#39;Slack\u30e1\u30c3\u30bb\u30fc\u30b8\u8aad\u307f\u8fbc\u307f\u4e2d...&#39;;\n          \n          try {\n            const response = await fetch(&#39;\/api\/slack-messages&#39;);\n            const data = await response.json();\n            \n            if (data.error) {\n              messagesDiv.innerHTML = &#39;&lt;p&gt;\u30a8\u30e9\u30fc: &#39; + data.error + &#39;&lt;\/p&gt;&#39;;\n              return;\n            }\n            \n            let html = &#39;&lt;h3&gt;Slack\u30e1\u30c3\u30bb\u30fc\u30b8\u53d6\u5f97\u6210\u529f!&lt;\/h3&gt;&#39;;\n            html += &#39;&lt;ul&gt;&#39;;\n            data.forEach(msg =&gt; {\n              html += &#39;&lt;li&gt;&#39;;\n              html += &#39;&lt;strong&gt;&#39; + (msg.userName || &#39;Unknown User&#39;) + &#39;&lt;\/strong&gt; &#39;;\n              html += &#39;&lt;span style=&quot;color: #888;&quot;&gt;(&#39; + msg.timestamp + &#39;)&lt;\/span&gt;: &#39;;\n              html += msg.text;\n              html += &#39;&lt;\/li&gt;&#39;;\n            });\n            html += &#39;&lt;\/ul&gt;&#39;;\n            messagesDiv.innerHTML = html;\n          } catch (error) {\n            messagesDiv.innerHTML = &#39;&lt;p&gt;\u30a8\u30e9\u30fc: &#39; + error.message + &#39;&lt;\/p&gt;&#39;;\n          }\n        });\n      &lt;\/script&gt;\n    &lt;\/body&gt;\n    &lt;\/html&gt;\n  `);\n});\n\n\/\/ Slack\u304b\u3089\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u53d6\u5f97\u3059\u308bAPI\napp.get(&#39;\/api\/slack-messages&#39;, async (req, res) =&gt; {\n  try {\n    const requestedChannelId = req.query.channelId;\n    \/\/ \u30c1\u30e3\u30f3\u30cd\u30ebID\u304c\u6307\u5b9a\u3055\u308c\u3066\u3044\u308b\u304b\u78ba\u8a8d\n    let targetChannelId;\n    \n    if (requestedChannelId) {\n      \/\/ \u30ea\u30af\u30a8\u30b9\u30c8\u3067\u6307\u5b9a\u3055\u308c\u305f\u30c1\u30e3\u30f3\u30cd\u30ebID\u3092\u4f7f\u7528\n      targetChannelId = requestedChannelId;\n      console.log(`\u30ea\u30af\u30a8\u30b9\u30c8\u3067\u6307\u5b9a\u3055\u308c\u305f\u30c1\u30e3\u30f3\u30cd\u30ebID: ${targetChannelId}`);\n    } else if (global.slackChannelIds && global.slackChannelIds.length &gt; 0) {\n      \/\/ \u30b0\u30ed\u30fc\u30d0\u30eb\u5909\u6570\u304b\u3089\u30c1\u30e3\u30f3\u30cd\u30ebID\u3092\u53d6\u5f97\n      targetChannelId = global.slackChannelIds[0];\n      console.log(`\u30b0\u30ed\u30fc\u30d0\u30eb\u5909\u6570\u304b\u3089\u306e\u30c7\u30d5\u30a9\u30eb\u30c8\u30c1\u30e3\u30f3\u30cd\u30ebID: ${targetChannelId}`);\n    } else if (slackChannelIds.length &gt; 0) {\n      \/\/ \u30c7\u30d5\u30a9\u30eb\u30c8\u306f\u6700\u521d\u306e\u30c1\u30e3\u30f3\u30cd\u30eb\u3092\u4f7f\u7528\n      targetChannelId = slackChannelIds[0];\n      console.log(`\u30c7\u30d5\u30a9\u30eb\u30c8\u30c1\u30e3\u30f3\u30cd\u30ebID: ${targetChannelId}`);\n    } else {\n      console.error(&#39;Slack API\u8a2d\u5b9a\u304c\u4e0d\u8db3\u3057\u3066\u3044\u307e\u3059&#39;);\n      return res.status(500).json({ \n        error: &#39;Slack API\u306e\u8a2d\u5b9a\u304c\u4e0d\u8db3\u3057\u3066\u3044\u307e\u3059\u3002\u74b0\u5883\u5909\u6570\u3092\u78ba\u8a8d\u3057\u3066\u304f\u3060\u3055\u3044\u3002&#39; \n      });\n    }\n\n    console.log(`\u30c1\u30e3\u30f3\u30cd\u30eb ${targetChannelId} \u304b\u3089Slack\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u53d6\u5f97\u3057\u307e\u3059...`);\n    \n    \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u60c5\u5831\u3092\u53d6\u5f97\n    let channelName = &#39;Slack Channel&#39;;\n    try {\n      console.log(`\u30c1\u30e3\u30f3\u30cd\u30eb ${targetChannelId} \u304b\u3089Slack\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u53d6\u5f97\u3057\u307e\u3059...`);\n      \n      \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u60c5\u5831\u3092\u53d6\u5f97\n      let channelName = &#39;Slack Channel&#39;;\n      try {\n        const channelInfo = await slackClient.conversations.info({\n          channel: targetChannelId\n        });\n        channelName = channelInfo.channel.name || &#39;Slack Channel&#39;;\n        console.log(`\u30c1\u30e3\u30f3\u30cd\u30eb\u540d: ${channelName}`);\n      } catch (channelError) {\n        console.error(&#39;\u30c1\u30e3\u30f3\u30cd\u30eb\u60c5\u5831\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, channelError);\n      }\n      \n      \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u5c65\u6b74\u3092\u53d6\u5f97\n      const result = await slackClient.conversations.history({\n        channel: targetChannelId,\n        cursor: cursor,\n        limit: 200, \/\/ 1\u56de\u306e\u30ea\u30af\u30a8\u30b9\u30c8\u3067\u6700\u5927200\u4ef6\n        oldest: Math.max(oldestTimestamp, latestMsg || 0), \/\/ 3\u30f6\u6708\u524d\u304b\u6700\u65b0\u306e\u4fdd\u5b58\u6e08\u307f\u30e1\u30c3\u30bb\u30fc\u30b8\u306e\u65b0\u3057\u3044\u65b9\u3092\u57fa\u6e96\u306b\u3059\u308b\n        include_all_metadata: true  \/\/ \u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u60c5\u5831\u306a\u3069\u3059\u3079\u3066\u306e\u30e1\u30bf\u30c7\u30fc\u30bf\u3092\u542b\u3081\u308b\n      });\n      \n      \/\/ API\u30ec\u30b9\u30dd\u30f3\u30b9\u306e\u30b5\u30f3\u30d7\u30eb\u3092\u30ed\u30b0\u306b\u51fa\u529b\uff08\u30c7\u30d0\u30c3\u30b0\u7528\uff09\n      if (result.messages && result.messages.length &gt; 0) {\n        console.log(`API\u30ec\u30b9\u30dd\u30f3\u30b9\u30b5\u30f3\u30d7\u30eb: \u6700\u521d\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u306e\u30d5\u30a3\u30fc\u30eb\u30c9\u4e00\u89a7:`, Object.keys(result.messages[0]).join(&#39;, &#39;));\n        \n        \/\/ \u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u3092\u542b\u3080\u30e1\u30c3\u30bb\u30fc\u30b8\u304c\u3042\u308b\u304b\u78ba\u8a8d\n        const msgWithReactions = result.messages.find(msg =&gt; msg.reactions && msg.reactions.length &gt; 0);\n        if (msgWithReactions) {\n          console.log(`\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u3092\u542b\u3080\u30e1\u30c3\u30bb\u30fc\u30b8\u4f8b:`, JSON.stringify(msgWithReactions.reactions, null, 2));\n        } else {\n          console.log(`\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u3092\u542b\u3080\u30e1\u30c3\u30bb\u30fc\u30b8\u304c\u898b\u3064\u304b\u308a\u307e\u305b\u3093\u3067\u3057\u305f`);\n        }\n      }\n      \n      console.log(`${result.messages.length}\u4ef6\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u53d6\u5f97\u3057\u307e\u3057\u305f`);\n\n      \/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u6574\u5f62\n      const messages = result.messages.map(msg =&gt; {\n        return {\n          text: msg.text,\n          user: msg.user,\n          ts: msg.ts,\n          timestamp: new Date(parseFloat(msg.ts) * 1000).toLocaleString(),\n          channelId: targetChannelId,\n          channelName: channelName\n        };\n      });\n\n      \/\/ \u30e6\u30fc\u30b6\u30fc\u60c5\u5831\u3092\u53d6\u5f97\u3057\u3066\u30e1\u30c3\u30bb\u30fc\u30b8\u306b\u8ffd\u52a0\n      const userIds = [...new Set(messages.map(msg =&gt; msg.user).filter(id =&gt; id && typeof id === &#39;string&#39; && id !== &#39;undefined&#39; && id !== &#39;null&#39;))];\n      const userPromises = userIds.map(async userId =&gt; {\n        try {\n          \/\/ null\u307e\u305f\u306f\u7121\u52b9\u306a\u30e6\u30fc\u30b6\u30fcID\u3092\u30c1\u30a7\u30c3\u30af\n          if (!userId || userId === &#39;null&#39; || userId === &#39;undefined&#39;) {\n            console.warn(`\u7121\u52b9\u306a\u30e6\u30fc\u30b6\u30fcID &#39;${userId}&#39; \u3092\u30b9\u30ad\u30c3\u30d7\u3057\u307e\u3059`);\n            return { id: userId || &#39;unknown&#39;, name: &#39;Unknown User&#39;, avatar: &#39;&#39; };\n          }\n          \n          const userInfo = await slackClient.users.info({ user: userId });\n          if (!userInfo || !userInfo.user) {\n            console.warn(`\u30e6\u30fc\u30b6\u30fc\u60c5\u5831\u304c\u898b\u3064\u304b\u308a\u307e\u305b\u3093: ${userId}`);\n            return { id: userId, name: &#39;Unknown User&#39;, avatar: &#39;&#39; };\n          }\n          \n          return {\n            id: userId,\n            name: userInfo.user.real_name || userInfo.user.name || &#39;Unknown User&#39;,\n            avatar: userInfo.user.profile?.image_72 || &#39;&#39;\n          };\n        } catch (error) {\n          console.error(`\u30e6\u30fc\u30b6\u30fc\u60c5\u5831\u53d6\u5f97\u30a8\u30e9\u30fc (${userId}):`, error);\n          return { id: userId || &#39;unknown&#39;, name: &#39;Unknown User&#39;, avatar: &#39;&#39; };\n        }\n      });\n\n      const users = await Promise.all(userPromises);\n      const userMap = {};\n      users.forEach(user =&gt; {\n        if (user && user.id) {\n          userMap[user.id] = user;\n        }\n      });\n\n      \/\/ \u30e6\u30fc\u30b6\u30fc\u60c5\u5831\u3092\u30e1\u30c3\u30bb\u30fc\u30b8\u306b\u8ffd\u52a0\n      const messagesWithUserInfo = messages.map(msg =&gt; {\n        const userId = msg.user || &#39;unknown&#39;;\n        return {\n          ...msg,\n          userName: userMap[userId]?.name || &#39;Unknown User&#39;,\n          userAvatar: userMap[userId]?.avatar || &#39;&#39;\n        };\n      });\n\n      res.json(messagesWithUserInfo);\n    } catch (error) {\n      console.error(&#39;Slack\u30e1\u30c3\u30bb\u30fc\u30b8\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, error);\n      \n      if (error.data) {\n        console.error(&#39;\u30a8\u30e9\u30fc\u8a73\u7d30:&#39;, JSON.stringify(error.data));\n      }\n      \n      res.status(500).json({ error: &#39;Slack\u30e1\u30c3\u30bb\u30fc\u30b8\u306e\u53d6\u5f97\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n    }\n  } catch (error) {\n    console.error(&#39;Slack\u30e1\u30c3\u30bb\u30fc\u30b8\u53d6\u5f97\u306e\u5916\u90e8\u30a8\u30e9\u30fc:&#39;, error);\n    res.status(500).json({ error: &#39;Slack\u30e1\u30c3\u30bb\u30fc\u30b8\u306e\u53d6\u5f97\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n  }\n});\n\n\/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u5206\u6790\u306eAPI\napp.get(&#39;\/api\/analyze-messages&#39;, async (req, res) =&gt; {\n  try {\n    const { channelId, userId } = req.query;\n    console.log(`\u5206\u6790\u30ea\u30af\u30a8\u30b9\u30c8: \u30c1\u30e3\u30f3\u30cd\u30eb=${channelId}, \u30e6\u30fc\u30b6\u30fc=${userId || &#39;\u672a\u6307\u5b9a\uff08\u5168\u30e6\u30fc\u30b6\u30fc\uff09&#39;}`);\n    \n    if (!channelId) {\n      return res.status(400).json({ error: &#39;\u30c1\u30e3\u30f3\u30cd\u30ebID\u304c\u6307\u5b9a\u3055\u308c\u3066\u3044\u307e\u305b\u3093&#39; });\n    }\n    \n    \/\/ \u7c21\u6613\u5206\u6790\u7d50\u679c\u3092\u8fd4\u3059\n    return res.json({\n      success: true,\n      source: &quot;simple&quot;,\n      messageCount: 0,\n      filteredMessageCount: 0,\n      userCount: 0,\n      channelName: &quot;\u5206\u6790\u6a5f\u80fd\u306f\u73fe\u5728\u7121\u52b9\u5316\u3055\u308c\u3066\u3044\u307e\u3059&quot;,\n      summary: {\n        mainTopics: [&quot;\u6a5f\u80fd\u7121\u52b9\u5316\u4e2d&quot;],\n        overallSummary: &quot;AI\u5206\u6790\u6a5f\u80fd\u306f\u73fe\u5728\u7121\u52b9\u5316\u3055\u308c\u3066\u3044\u307e\u3059\u3002&quot;\n      },\n      trendAnalysis: {\n        keywords: [&quot;\u7121\u52b9\u5316&quot;, &quot;\u30e1\u30f3\u30c6\u30ca\u30f3\u30b9\u4e2d&quot;],\n        topicChanges: &quot;\u5206\u6790\u306a\u3057&quot;,\n        communicationPattern: &quot;\u5206\u6790\u306a\u3057&quot;\n      },\n      userAnalysis: {},\n      insightsAndRecommendations: [&quot;\u73fe\u5728\u3001\u5206\u6790\u6a5f\u80fd\u306f\u30e1\u30f3\u30c6\u30ca\u30f3\u30b9\u4e2d\u306e\u305f\u3081\u5229\u7528\u3067\u304d\u307e\u305b\u3093\u3002&quot;]\n    });\n    \n  } catch (error) {\n    console.error(&#39;\u30e1\u30c3\u30bb\u30fc\u30b8\u5206\u6790\u4e2d\u306b\u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u307e\u3057\u305f:&#39;, error);\n    res.status(500).json({ error: error.message });\n  }\n});\n\n\/\/ API\u30ad\u30fc\u8a2d\u5b9a\u753b\u9762\napp.get(&#39;\/settings&#39;, (req, res) =&gt; {\n  res.sendFile(path.join(__dirname, &#39;public&#39;, &#39;settings.html&#39;));\n});\n\n\/\/ API\u30ad\u30fc\u8a2d\u5b9a\u3092\u4fdd\u5b58\u3059\u308b\u30a8\u30f3\u30c9\u30dd\u30a4\u30f3\u30c8\napp.post(&#39;\/api\/settings&#39;, express.json(), (req, res) =&gt; {\n  try {\n    const { anthropicApiKey } = req.body;\n    \n    if (!anthropicApiKey) {\n      return res.status(400).json({ error: &#39;API\u30ad\u30fc\u306f\u5fc5\u9808\u3067\u3059&#39; });\n    }\n    \n    \/\/ .renv\u30d5\u30a1\u30a4\u30eb\u3092\u8aad\u307f\u8fbc\u3080\n    let envContent = &#39;&#39;;\n    try {\n      envContent = fs.readFileSync(&#39;.renv&#39;, &#39;utf8&#39;);\n    } catch (error) {\n      console.error(&#39;.renv\u30d5\u30a1\u30a4\u30eb\u306e\u8aad\u307f\u8fbc\u307f\u306b\u5931\u6557\u3057\u307e\u3057\u305f:&#39;, error.message);\n      envContent = &#39;SLACK_BOT_TOKEN=\\nSLACK_CHANNEL_ID=\\nOPENAI_API_KEY=\\nANTHROPIC_API_KEY=\\n&#39;;\n    }\n    \n    \/\/ ANTHROPIC_API_KEY\u306e\u884c\u3092\u7f6e\u304d\u63db\u3048\u308b\n    const lines = envContent.split(&#39;\\n&#39;);\n    let found = false;\n    for (let i = 0; i &lt; lines.length; i++) {\n      if (lines[i].startsWith(&#39;ANTHROPIC_API_KEY=&#39;)) {\n        lines[i] = `ANTHROPIC_API_KEY=${anthropicApiKey}`;\n        found = true;\n        break;\n      }\n    }\n    \n    \/\/ API\u30ad\u30fc\u306e\u884c\u304c\u898b\u3064\u304b\u3089\u306a\u304b\u3063\u305f\u5834\u5408\u306f\u8ffd\u52a0\n    if (!found) {\n      lines.push(`ANTHROPIC_API_KEY=${anthropicApiKey}`);\n    }\n    \n    \/\/ .renv\u30d5\u30a1\u30a4\u30eb\u306b\u66f8\u304d\u8fbc\u3080\n    fs.writeFileSync(&#39;.renv&#39;, lines.join(&#39;\\n&#39;));\n    \n    \/\/ \u4fdd\u5b58\u3057\u305f\u5185\u5bb9\u3092\u30ed\u30b0\u306b\u51fa\u529b\uff08\u30c7\u30d0\u30c3\u30b0\u7528\uff09\n    console.log(&#39;\u4fdd\u5b58\u3057\u305f.renv\u30d5\u30a1\u30a4\u30eb\u306e\u5185\u5bb9:&#39;);\n    console.log(lines.join(&#39;\\n&#39;));\n    \n    \/\/ \u74b0\u5883\u5909\u6570\u3092\u66f4\u65b0\n    process.env.ANTHROPIC_API_KEY = anthropicApiKey;\n    \n    \/\/ Anthropic client\u3092\u518d\u521d\u671f\u5316\n    try {\n      anthropic.apiKey = anthropicApiKey;\n      console.log(&#39;Anthropic API\u30ad\u30fc\u3092\u66f4\u65b0\u3057\u307e\u3057\u305f&#39;);\n    } catch (error) {\n      console.error(&#39;Anthropic \u30af\u30e9\u30a4\u30a2\u30f3\u30c8\u306e\u518d\u521d\u671f\u5316\u306b\u5931\u6557\u3057\u307e\u3057\u305f:&#39;, error.message);\n    }\n    \n    res.json({ success: true, message: &#39;API\u30ad\u30fc\u3092\u4fdd\u5b58\u3057\u307e\u3057\u305f&#39; });\n  } catch (error) {\n    console.error(&#39;API\u30ad\u30fc\u306e\u4fdd\u5b58\u306b\u5931\u6557\u3057\u307e\u3057\u305f:&#39;, error.message);\n    res.status(500).json({ error: &#39;API\u30ad\u30fc\u306e\u4fdd\u5b58\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n  }\n});\n\n\/\/ \u8a2d\u5b9a\u60c5\u5831\u3092\u53d6\u5f97\u3059\u308bAPI\napp.get(&#39;\/api\/settings\/info&#39;, (req, res) =&gt; {\n  try {\n    \/\/ \u8a2d\u5b9a\u30d5\u30a1\u30a4\u30eb\u304b\u3089\u60c5\u5831\u3092\u53d6\u5f97\n    const settings = getSettings();\n    \n    \/\/ Slack\u30c8\u30fc\u30af\u30f3\u306e\u5b58\u5728\u78ba\u8a8d\n    const slackBotTokenSet = !!settings.slackBotToken;\n    \n    \/\/ Slack\u30c1\u30e3\u30f3\u30cd\u30eb\u306e\u5b58\u5728\u78ba\u8a8d\n    const slackChannelIdsSet = settings.slackChannelIds && settings.slackChannelIds.length &gt; 0;\n    \n    \/\/ Anthropic API\u30ad\u30fc\u306e\u5b58\u5728\u78ba\u8a8d\n    const anthropicApiKeySet = !!settings.anthropicApiKey;\n    \n    \/\/ OpenAI API\u30ad\u30fc\u306e\u5b58\u5728\u78ba\u8a8d\n    const openaiApiKeySet = !!settings.openaiApiKey;\n    \n    \/\/ \u30bb\u30f3\u30b7\u30c6\u30a3\u30d6\u306a\u60c5\u5831\u306f\u30de\u30b9\u30af\u3057\u3066\u8fd4\u3059\n    const slackBotTokenMasked = slackBotTokenSet ? \n      maskValue(settings.slackBotToken) : null;\n      \n    const anthropicApiKeyMasked = anthropicApiKeySet ? \n      maskValue(settings.anthropicApiKey) : null;\n      \n    const openaiApiKeyMasked = openaiApiKeySet ? \n      maskValue(settings.openaiApiKey) : null;\n      \n    \/\/ \u30ad\u30e3\u30e9\u30af\u30bf\u30fc\u753b\u50cf\u30d1\u30b9\u3092\u8ffd\u52a0\n    const characterImage = settings.characterImage || &#39;\/images\/brave-character.png&#39;;\n      \n    res.json({\n      slackBotTokenSet,\n      slackChannelIdsSet,\n      anthropicApiKeySet,\n      openaiApiKeySet,\n      slackBotTokenMasked,\n      anthropicApiKeyMasked,\n      openaiApiKeyMasked,\n      slackChannelIds: settings.slackChannelIds || defaultSettings.slackChannelIds || process.env.SLACK_CHANNEL_IDS || &#39;&#39;,\n      slackBotToken: settings.slackBotToken || defaultSettings.botToken || process.env.SLACK_BOT_TOKEN || &#39;&#39;,\n      slackSigningSecret: settings.slackSigningSecret || process.env.SLACK_SIGNING_SECRET || &#39;&#39;,\n      anthropicApiKey: settings.anthropicApiKey || process.env.ANTHROPIC_API_KEY || &#39;&#39;,\n      openaiApiKey: settings.openaiApiKey || process.env.OPENAI_API_KEY || &#39;&#39;,\n      aiAnalysisPrompt: settings.aiAnalysisPrompt || defaultAiAnalysisPrompt,\n      characterImage\n    });\n  } catch (error) {\n    console.error(&#39;\u8a2d\u5b9a\u60c5\u5831\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, error);\n    res.status(500).json({ error: &#39;\u8a2d\u5b9a\u60c5\u5831\u306e\u53d6\u5f97\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n  }\n});\n\n\/\/ Slack\u8a2d\u5b9a\u3092\u66f4\u65b0\u3059\u308bAPI\napp.post(&#39;\/api\/settings\/slack&#39;, express.json(), (req, res) =&gt; {\n  try {\n    const { slackBotToken, slackChannelIds } = req.body;\n    \n    \/\/ \u5fc5\u9808\u9805\u76ee\u30c1\u30a7\u30c3\u30af\n    if (!slackChannelIds || !Array.isArray(slackChannelIds) || slackChannelIds.length === 0) {\n      return res.status(400).json({ error: &#39;\u5c11\u306a\u304f\u3068\u30821\u3064\u306e\u30c1\u30e3\u30f3\u30cd\u30ebID\u304c\u5fc5\u8981\u3067\u3059&#39; });\n    }\n    \n    \/\/ .renv\u30d5\u30a1\u30a4\u30eb\u3092\u8aad\u307f\u8fbc\u3080\n    let envContent;\n    try {\n      envContent = fs.readFileSync(&#39;.renv&#39;, &#39;utf8&#39;);\n    } catch (error) {\n      console.error(&#39;.renv\u30d5\u30a1\u30a4\u30eb\u306e\u8aad\u307f\u8fbc\u307f\u306b\u5931\u6557\u3057\u307e\u3057\u305f:&#39;, error.message);\n      envContent = &#39;&#39;; \/\/ \u30d5\u30a1\u30a4\u30eb\u304c\u5b58\u5728\u3057\u306a\u3044\u5834\u5408\u306f\u7a7a\u6587\u5b57\u3067\u521d\u671f\u5316\n    }\n    \n    const lines = envContent.split(&#39;\\n&#39;).filter(line =&gt; line.trim() !== &#39;&#39;);\n    \n    \/\/ \u65e2\u5b58\u306e\u8a2d\u5b9a\u3092\u66f4\u65b0\u307e\u305f\u306f\u65b0\u898f\u8ffd\u52a0\n    let slackBotTokenFound = false;\n    let slackChannelIdsFound = false;\n    \n    for (let i = 0; i &lt; lines.length; i++) {\n      if (lines[i].startsWith(&#39;SLACK_BOT_TOKEN=&#39;)) {\n        if (slackBotToken) {\n          lines[i] = `SLACK_BOT_TOKEN=${slackBotToken}`;\n        }\n        slackBotTokenFound = true;\n      }\n      \n      \/\/ \u65b0\u5f62\u5f0f\u306e\u8907\u6570\u30c1\u30e3\u30f3\u30cd\u30ebID\u306e\u8a2d\u5b9a\n      if (lines[i].startsWith(&#39;SLACK_CHANNEL_IDS=&#39;)) {\n        lines[i] = `SLACK_CHANNEL_IDS=${slackChannelIds.join(&#39;,&#39;)}`;\n        slackChannelIdsFound = true;\n      }\n      \n      \/\/ \u65e7\u5f62\u5f0f\u306e\u5358\u4e00\u30c1\u30e3\u30f3\u30cd\u30ebID\u3092\u524a\u9664\uff08\u307e\u305f\u306f\u66f4\u65b0\uff09\n      if (lines[i].startsWith(&#39;SLACK_CHANNEL_ID=&#39;)) {\n        if (slackChannelIdsFound) {\n          \/\/ \u8907\u6570\u30c1\u30e3\u30f3\u30cd\u30ebID\u304c\u3059\u3067\u306b\u8a2d\u5b9a\u3055\u308c\u3066\u3044\u308c\u3070\u3001\u3053\u306e\u884c\u3092\u524a\u9664\n          lines.splice(i, 1);\n          i--; \/\/ \u30a4\u30f3\u30c7\u30c3\u30af\u30b9\u3092\u8abf\u6574\n        } else {\n          \/\/ \u8907\u6570\u30c1\u30e3\u30f3\u30cd\u30ebID\u304c\u8a2d\u5b9a\u3055\u308c\u3066\u3044\u306a\u3051\u308c\u3070\u3001\u30c7\u30d5\u30a9\u30eb\u30c8\u30c1\u30e3\u30f3\u30cd\u30eb\u3068\u3057\u3066\u8a2d\u5b9a\n          lines[i] = `SLACK_CHANNEL_ID=${slackChannelIds[0]}`;\n        }\n      }\n    }\n    \n    \/\/ \u8a2d\u5b9a\u304c\u5b58\u5728\u3057\u306a\u3044\u5834\u5408\u306f\u65b0\u898f\u8ffd\u52a0\n    if (!slackBotTokenFound && slackBotToken) {\n      lines.push(`SLACK_BOT_TOKEN=${slackBotToken}`);\n    }\n    \n    if (!slackChannelIdsFound) {\n      lines.push(`SLACK_CHANNEL_IDS=${slackChannelIds.join(&#39;,&#39;)}`);\n      \/\/ \u5f8c\u65b9\u4e92\u63db\u6027\u306e\u305f\u3081\u306b\u6700\u521d\u306e\u30c1\u30e3\u30f3\u30cd\u30ebID\u3092\u30c7\u30d5\u30a9\u30eb\u30c8\u3068\u3057\u3066\u8a2d\u5b9a\n      lines.push(`SLACK_CHANNEL_ID=${slackChannelIds[0]}`);\n    }\n    \n    \/\/ .renv\u30d5\u30a1\u30a4\u30eb\u306b\u66f8\u304d\u8fbc\u3080\n    fs.writeFileSync(&#39;.renv&#39;, lines.join(&#39;\\n&#39;));\n    \n    \/\/ \u74b0\u5883\u5909\u6570\u306b\u3082\u76f4\u63a5\u8a2d\u5b9a\n    if (slackBotToken) {\n      process.env.SLACK_BOT_TOKEN = slackBotToken;\n    }\n    process.env.SLACK_CHANNEL_IDS = slackChannelIds.join(&#39;,&#39;);\n    process.env.SLACK_CHANNEL_ID = slackChannelIds[0]; \/\/ \u5f8c\u65b9\u4e92\u63db\u6027\u306e\u305f\u3081\u306b\u30c7\u30d5\u30a9\u30eb\u30c8\u3068\u3057\u3066\u8a2d\u5b9a\n    \n    \/\/ \u30b0\u30ed\u30fc\u30d0\u30eb\u5909\u6570\u3092\u66f4\u65b0\n    global.slackChannelIds = slackChannelIds;\n    \n    \/\/ \u30b0\u30ed\u30fc\u30d0\u30eb\u5909\u6570\u306e\u66f4\u65b0\u78ba\u8a8d\uff08\u30c7\u30d0\u30c3\u30b0\u7528\uff09\n    console.log(&#39;\u30b0\u30ed\u30fc\u30d0\u30eb\u5909\u6570slackChannelIds\u3092\u66f4\u65b0\u3057\u307e\u3057\u305f:&#39;, JSON.stringify(global.slackChannelIds));\n    \n    \/\/ \u6210\u529f\u30ec\u30b9\u30dd\u30f3\u30b9\n    res.json({ \n      success: true, \n      message: &#39;Slack\u8a2d\u5b9a\u3092\u66f4\u65b0\u3057\u307e\u3057\u305f&#39;,\n      channelIds: slackChannelIds\n    });\n  } catch (error) {\n    console.error(&#39;Slack\u8a2d\u5b9a\u66f4\u65b0\u30a8\u30e9\u30fc:&#39;, error);\n    res.status(500).json({ error: &#39;Slack\u8a2d\u5b9a\u306e\u66f4\u65b0\u306b\u5931\u6557\u3057\u307e\u3057\u305f: &#39; + error.message });\n  }\n});\n\n\/\/ Anthropic API\u30ad\u30fc\u3092\u66f4\u65b0\u3059\u308bAPI\napp.post(&#39;\/api\/settings\/anthropic&#39;, express.json(), (req, res) =&gt; {\n  try {\n    const { anthropicApiKey } = req.body;\n    \n    if (!anthropicApiKey) {\n      return res.status(400).json({ error: &#39;API\u30ad\u30fc\u306f\u5fc5\u9808\u3067\u3059&#39; });\n    }\n    \n    \/\/ .renv\u30d5\u30a1\u30a4\u30eb\u3092\u8aad\u307f\u8fbc\u3080\n    let envContent = &#39;&#39;;\n    try {\n      envContent = fs.readFileSync(&#39;.renv&#39;, &#39;utf8&#39;);\n    } catch (error) {\n      console.error(&#39;.renv\u30d5\u30a1\u30a4\u30eb\u306e\u8aad\u307f\u8fbc\u307f\u306b\u5931\u6557\u3057\u307e\u3057\u305f:&#39;, error.message);\n      envContent = &#39;SLACK_BOT_TOKEN=\\nSLACK_CHANNEL_ID=\\nOPENAI_API_KEY=\\nANTHROPIC_API_KEY=\\n&#39;;\n    }\n    \n    \/\/ ANTHROPIC_API_KEY\u306e\u884c\u3092\u7f6e\u304d\u63db\u3048\u308b\n    const lines = envContent.split(&#39;\\n&#39;);\n    let found = false;\n    for (let i = 0; i &lt; lines.length; i++) {\n      if (lines[i].startsWith(&#39;ANTHROPIC_API_KEY=&#39;)) {\n        lines[i] = `ANTHROPIC_API_KEY=${anthropicApiKey}`;\n        found = true;\n        break;\n      }\n    }\n    \n    \/\/ API\u30ad\u30fc\u306e\u884c\u304c\u898b\u3064\u304b\u3089\u306a\u304b\u3063\u305f\u5834\u5408\u306f\u8ffd\u52a0\n    if (!found) {\n      lines.push(`ANTHROPIC_API_KEY=${anthropicApiKey}`);\n    }\n    \n    \/\/ .renv\u30d5\u30a1\u30a4\u30eb\u306b\u66f8\u304d\u8fbc\u3080\n    fs.writeFileSync(&#39;.renv&#39;, lines.join(&#39;\\n&#39;));\n    \n    \/\/ \u4fdd\u5b58\u3057\u305f\u5185\u5bb9\u3092\u30ed\u30b0\u306b\u51fa\u529b\uff08\u30c7\u30d0\u30c3\u30b0\u7528\uff09\n    console.log(&#39;\u4fdd\u5b58\u3057\u305f.renv\u30d5\u30a1\u30a4\u30eb\u306e\u5185\u5bb9:&#39;);\n    console.log(lines.join(&#39;\\n&#39;));\n    \n    \/\/ \u74b0\u5883\u5909\u6570\u3092\u66f4\u65b0\n    process.env.ANTHROPIC_API_KEY = anthropicApiKey;\n    \n    \/\/ Anthropic client\u3092\u518d\u521d\u671f\u5316\n    try {\n      anthropic.apiKey = anthropicApiKey;\n      console.log(&#39;Anthropic API\u30ad\u30fc\u3092\u66f4\u65b0\u3057\u307e\u3057\u305f&#39;);\n    } catch (error) {\n      console.error(&#39;Anthropic \u30af\u30e9\u30a4\u30a2\u30f3\u30c8\u306e\u518d\u521d\u671f\u5316\u306b\u5931\u6557\u3057\u307e\u3057\u305f:&#39;, error.message);\n    }\n    \n    res.json({ success: true, message: &#39;Anthropic API\u30ad\u30fc\u3092\u66f4\u65b0\u3057\u307e\u3057\u305f&#39; });\n  } catch (error) {\n    console.error(&#39;Anthropic API\u30ad\u30fc\u306e\u66f4\u65b0\u306b\u5931\u6557\u3057\u307e\u3057\u305f:&#39;, error.message);\n    res.status(500).json({ error: &#39;Anthropic API\u30ad\u30fc\u306e\u66f4\u65b0\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n  }\n});\n\n\/\/ OpenAI API\u30ad\u30fc\u3092\u66f4\u65b0\u3059\u308bAPI\napp.post(&#39;\/api\/settings\/openai&#39;, express.json(), (req, res) =&gt; {\n  try {\n    const { openaiApiKey } = req.body;\n    \n    if (!openaiApiKey) {\n      return res.status(400).json({ error: &#39;API\u30ad\u30fc\u304c\u6307\u5b9a\u3055\u308c\u3066\u3044\u307e\u305b\u3093&#39; });\n    }\n    \n    \/\/ \u74b0\u5883\u5909\u6570\u3092\u66f4\u65b0\n    process.env.OPENAI_API_KEY = openaiApiKey;\n    \n    \/\/ .env \u30d5\u30a1\u30a4\u30eb\u304c\u3042\u308b\u5834\u5408\u306f\u66f4\u65b0\n    const envPath = path.join(__dirname, &#39;.renv&#39;);\n    \n    if (fs.existsSync(envPath)) {\n      \/\/ \u65e2\u5b58\u306e .env \u30d5\u30a1\u30a4\u30eb\u3092\u8aad\u307f\u8fbc\u3080\n      const envFileContent = fs.readFileSync(envPath, &#39;utf8&#39;);\n      \n      \/\/ OPENAI_API_KEY \u306e\u884c\u304c\u3042\u308c\u3070\u66f4\u65b0\u3001\u306a\u3051\u308c\u3070\u8ffd\u52a0\n      let updatedContent;\n      if (envFileContent.match(\/OPENAI_API_KEY=\/)) {\n        updatedContent = envFileContent.replace(\/OPENAI_API_KEY=.*\/, `OPENAI_API_KEY=${openaiApiKey}`);\n      } else {\n        updatedContent = envFileContent + `\\nOPENAI_API_KEY=${openaiApiKey}`;\n      }\n      \n      \/\/ \u66f4\u65b0\u3057\u305f\u5185\u5bb9\u3067 .env \u30d5\u30a1\u30a4\u30eb\u3092\u4e0a\u66f8\u304d\n      fs.writeFileSync(envPath, updatedContent);\n      \n      console.log(&#39;OpenAI API\u30ad\u30fc\u304c\u8a2d\u5b9a\u3055\u308c\u307e\u3057\u305f&#39;);\n    } else {\n      console.warn(&#39;.env \u30d5\u30a1\u30a4\u30eb\u304c\u5b58\u5728\u3057\u306a\u3044\u305f\u3081\u3001\u30e1\u30e2\u30ea\u4e0a\u306e\u307f\u3067\u74b0\u5883\u5909\u6570\u3092\u66f4\u65b0\u3057\u307e\u3057\u305f&#39;);\n    }\n    \n    res.json({ success: true });\n  } catch (error) {\n    console.error(&#39;OpenAI API\u30ad\u30fc\u8a2d\u5b9a\u4e2d\u306b\u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u307e\u3057\u305f:&#39;, error);\n    res.status(500).json({ error: error.message });\n  }\n});\n\n\/\/ AI\u5206\u6790\u30d7\u30ed\u30f3\u30d7\u30c8\u8a2d\u5b9a\u30a8\u30f3\u30c9\u30dd\u30a4\u30f3\u30c8\napp.post(&#39;\/api\/settings\/ai-prompt&#39;, express.json(), (req, res) =&gt; {\n  try {\n    const { aiAnalysisPrompt } = req.body;\n    \n    \/\/ \u7a7a\u306e\u5834\u5408\u306f\u30c7\u30d5\u30a9\u30eb\u30c8\u5024\u306b\u623b\u3059\n    if (!aiAnalysisPrompt) {\n      currentAiAnalysisPrompt = defaultAiAnalysisPrompt;\n      \n      \/\/ \u8a2d\u5b9a\u3092\u66f4\u65b0\u3057\u3066\u4fdd\u5b58\n      const settings = getSettings();\n      settings.aiAnalysisPrompt = defaultAiAnalysisPrompt;\n      saveSettings(settings);\n      \n      return res.json({\n        success: true,\n        message: &#39;AI\u5206\u6790\u30d7\u30ed\u30f3\u30d7\u30c8\u304c\u30c7\u30d5\u30a9\u30eb\u30c8\u5024\u306b\u623b\u3055\u308c\u307e\u3057\u305f&#39;\n      });\n    }\n    \n    \/\/ \u30d7\u30ed\u30f3\u30d7\u30c8\u3092\u66f4\u65b0\n    currentAiAnalysisPrompt = aiAnalysisPrompt;\n    \n    \/\/ \u8a2d\u5b9a\u3092\u66f4\u65b0\u3057\u3066\u4fdd\u5b58\n    const settings = getSettings();\n    settings.aiAnalysisPrompt = aiAnalysisPrompt;\n    saveSettings(settings);\n    \n    console.log(&#39;AI\u5206\u6790\u30d7\u30ed\u30f3\u30d7\u30c8\u304c\u66f4\u65b0\u3055\u308c\u307e\u3057\u305f&#39;);\n    res.json({\n      success: true,\n      message: &#39;AI\u5206\u6790\u30d7\u30ed\u30f3\u30d7\u30c8\u304c\u6b63\u5e38\u306b\u66f4\u65b0\u3055\u308c\u307e\u3057\u305f&#39;\n    });\n  } catch (error) {\n    console.error(&#39;AI\u5206\u6790\u30d7\u30ed\u30f3\u30d7\u30c8\u8a2d\u5b9a\u4e2d\u306b\u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u307e\u3057\u305f:&#39;, error);\n    res.status(500).json({ error: error.message });\n  }\n});\n\n\/\/ AI\u5206\u6790\u30d7\u30ed\u30f3\u30d7\u30c8\u3092\u30ea\u30bb\u30c3\u30c8\u3059\u308b\u30a8\u30f3\u30c9\u30dd\u30a4\u30f3\u30c8\napp.post(&#39;\/api\/settings\/reset-ai-prompt&#39;, express.json(), (req, res) =&gt; {\n  try {\n    \/\/ \u30c7\u30d5\u30a9\u30eb\u30c8\u30d7\u30ed\u30f3\u30d7\u30c8\u306b\u623b\u3059\n    currentAiAnalysisPrompt = defaultAiAnalysisPrompt;\n    \n    \/\/ \u8a2d\u5b9a\u3092\u66f4\u65b0\u3057\u3066\u4fdd\u5b58\n    const settings = getSettings();\n    settings.aiAnalysisPrompt = defaultAiAnalysisPrompt;\n    saveSettings(settings);\n    \n    console.log(&#39;AI\u5206\u6790\u30d7\u30ed\u30f3\u30d7\u30c8\u304c\u30ea\u30bb\u30c3\u30c8\u3055\u308c\u307e\u3057\u305f&#39;);\n    \n    \/\/ \u30c7\u30d5\u30a9\u30eb\u30c8\u30d7\u30ed\u30f3\u30d7\u30c8\u3092\u30ed\u30b0\u306b\u51fa\u529b\u3057\u3066\u78ba\u8a8d\n    console.log(&#39;\u30c7\u30d5\u30a9\u30eb\u30c8\u30d7\u30ed\u30f3\u30d7\u30c8 (\u6700\u521d\u306e50\u6587\u5b57):&#39;, defaultAiAnalysisPrompt.substring(0, 50) + &#39;...&#39;);\n    \n    \/\/ \u30ec\u30b9\u30dd\u30f3\u30b9\u3092\u8fd4\u3059\n    res.json({\n      success: true,\n      message: &#39;AI\u5206\u6790\u30d7\u30ed\u30f3\u30d7\u30c8\u304c\u30c7\u30d5\u30a9\u30eb\u30c8\u5024\u306b\u6b63\u5e38\u306b\u30ea\u30bb\u30c3\u30c8\u3055\u308c\u307e\u3057\u305f&#39;,\n      aiAnalysisPrompt: defaultAiAnalysisPrompt \/\/ \u30c7\u30d5\u30a9\u30eb\u30c8\u30d7\u30ed\u30f3\u30d7\u30c8\u3092\u30ec\u30b9\u30dd\u30f3\u30b9\u306b\u542b\u3081\u308b\n    });\n  } catch (error) {\n    console.error(&#39;AI\u5206\u6790\u30d7\u30ed\u30f3\u30d7\u30c8\u30ea\u30bb\u30c3\u30c8\u4e2d\u306b\u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u307e\u3057\u305f:&#39;, error);\n    res.status(500).json({ error: error.message });\n  }\n});\n\n\/\/ \u30c7\u30fc\u30bf\u30ea\u30bb\u30c3\u30c8\u30a8\u30f3\u30c9\u30dd\u30a4\u30f3\u30c8\napp.post(&#39;\/api\/settings\/reset-data&#39;, express.json(), (req, res) =&gt; {\n  try {\n    console.log(&#39;\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u306e\u30ea\u30bb\u30c3\u30c8\u3092\u958b\u59cb\u3057\u307e\u3059...&#39;);\n    \n    \/\/ \u5404\u30c6\u30fc\u30d6\u30eb\u306e\u30c7\u30fc\u30bf\u3092\u524a\u9664\n    db.run(&#39;DELETE FROM slack_messages&#39;);\n    db.run(&#39;DELETE FROM slack_thread_replies&#39;);\n    db.run(&#39;DELETE FROM slack_reactions&#39;);\n    db.run(&#39;DELETE FROM slack_channels&#39;);\n    db.run(&#39;DELETE FROM daily_pickup_user&#39;);\n    db.run(&#39;DELETE FROM user_analysis&#39;);\n    \n    \/\/ \u30b7\u30fc\u30b1\u30f3\u30b9\u30ea\u30bb\u30c3\u30c8 (SQLite\u306e\u5834\u5408)\n    db.run(&#39;DELETE FROM sqlite_sequence WHERE name IN (\\&#39;slack_messages\\&#39;, \\&#39;slack_thread_replies\\&#39;, \\&#39;slack_reactions\\&#39;, \\&#39;user_analysis\\&#39;, \\&#39;daily_pickup_user\\&#39;)&#39;);\n    \n    console.log(&#39;\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u306e\u30ea\u30bb\u30c3\u30c8\u304c\u5b8c\u4e86\u3057\u307e\u3057\u305f&#39;);\n    \n    res.json({ \n      success: true, \n      message: &#39;\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u304c\u6b63\u5e38\u306b\u30ea\u30bb\u30c3\u30c8\u3055\u308c\u307e\u3057\u305f\u3002\u30ed\u30fc\u30ab\u30eb\u30b9\u30c8\u30ec\u30fc\u30b8\u306e\u540c\u671f\u65e5\u6642\u60c5\u5831\u3082\u524a\u9664\u3055\u308c\u307e\u3059\u3002&#39;\n    });\n  } catch (error) {\n    console.error(&#39;\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u30ea\u30bb\u30c3\u30c8\u30a8\u30e9\u30fc:&#39;, error);\n    res.status(500).json({ error: &#39;\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u306e\u30ea\u30bb\u30c3\u30c8\u306b\u5931\u6557\u3057\u307e\u3057\u305f: &#39; + error.message });\n  }\n});\n\n\/\/ \u30bb\u30f3\u30b7\u30c6\u30a3\u30d6\u306a\u5024\u3092\u30de\u30b9\u30af\u3059\u308b\u95a2\u6570\nfunction maskValue(value) {\n  if (!value) return null;\n  \n  if (value.length &gt; 12) {\n    return value.substring(0, 6) + &#39;...&#39; + value.substring(value.length - 4);\n  } else {\n    return &#39;***&#39; + value.substring(value.length - 4);\n  }\n}\n\n\/\/ \u30ad\u30e3\u30e9\u30af\u30bf\u30fc\u753b\u50cf\u4fdd\u5b58\u5148\u306e\u30c7\u30a3\u30ec\u30af\u30c8\u30ea\u8a2d\u5b9a\nconst characterStorage = multer.diskStorage({\n  destination: function(req, file, cb) {\n    const uploadDir = path.join(__dirname, &#39;public&#39;, &#39;images&#39;);\n    \/\/ \u30c7\u30a3\u30ec\u30af\u30c8\u30ea\u304c\u5b58\u5728\u3057\u306a\u3044\u5834\u5408\u306f\u4f5c\u6210\n    if (!fs.existsSync(uploadDir)) {\n      fs.mkdirSync(uploadDir, { recursive: true });\n    }\n    cb(null, uploadDir);\n  },\n  filename: function(req, file, cb) {\n    \/\/ \u30d5\u30a1\u30a4\u30eb\u540d\u3092 brave-character + \u62e1\u5f35\u5b50\u306b\u8a2d\u5b9a\uff08\u4e0a\u66f8\u304d\uff09\n    const extname = path.extname(file.originalname);\n    cb(null, &#39;brave-character-custom&#39; + extname);\n  }\n});\n\n\/\/ \u30ad\u30e3\u30e9\u30af\u30bf\u30fc\u753b\u50cf\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u7528\u306emulter\u30a4\u30f3\u30b9\u30bf\u30f3\u30b9\nconst uploadCharacter = multer({\n  storage: characterStorage,\n  limits: {\n    fileSize: 2 * 1024 * 1024 \/\/ 2MB\u5236\u9650\n  },\n  fileFilter: function(req, file, cb) {\n    \/\/ \u753b\u50cf\u30d5\u30a1\u30a4\u30eb\u306e\u307f\u8a31\u53ef\n    const filetypes = \/jpeg|jpg|png|gif|svg\/;\n    const mimetype = filetypes.test(file.mimetype);\n    const extname = filetypes.test(path.extname(file.originalname).toLowerCase());\n    \n    if (mimetype && extname) {\n      return cb(null, true);\n    }\n    cb(new Error(&#39;\u753b\u50cf\u30d5\u30a1\u30a4\u30eb\uff08jpeg, jpg, png, gif, svg\uff09\u306e\u307f\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u3067\u304d\u307e\u3059&#39;));\n  }\n});\n\n\/\/ \u30ad\u30e3\u30e9\u30af\u30bf\u30fc\u753b\u50cf\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9API\napp.post(&#39;\/api\/settings\/upload-character&#39;, uploadCharacter.single(&#39;character-image&#39;), (req, res) =&gt; {\n  try {\n    if (!req.file) {\n      return res.status(400).json({ success: false, error: &#39;\u30d5\u30a1\u30a4\u30eb\u304c\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u3055\u308c\u3066\u3044\u307e\u305b\u3093&#39; });\n    }\n    \n    console.log(`\u30ad\u30e3\u30e9\u30af\u30bf\u30fc\u753b\u50cf\u3092\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u3057\u307e\u3057\u305f: ${req.file.filename}`);\n    \n    \/\/ \u30ad\u30e3\u30e9\u30af\u30bf\u30fc\u30d5\u30a1\u30a4\u30eb\u60c5\u5831\u3092\u8a2d\u5b9a\u30d5\u30a1\u30a4\u30eb\u306b\u4fdd\u5b58\n    const settings = getSettings();\n    settings.characterImage = `\/images\/${req.file.filename}`;\n    const saved = saveSettings(settings);\n    \n    if (!saved) {\n      throw new Error(&#39;\u8a2d\u5b9a\u306e\u4fdd\u5b58\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39;);\n    }\n    \n    res.json({ \n      success: true, \n      message: &#39;\u30ad\u30e3\u30e9\u30af\u30bf\u30fc\u753b\u50cf\u304c\u6b63\u5e38\u306b\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u3055\u308c\u307e\u3057\u305f&#39;,\n      imagePath: `\/images\/${req.file.filename}`\n    });\n  } catch (error) {\n    console.error(&#39;\u30ad\u30e3\u30e9\u30af\u30bf\u30fc\u753b\u50cf\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u30a8\u30e9\u30fc:&#39;, error);\n    res.status(500).json({ success: false, error: error.message });\n  }\n});\n\n\/\/ \u30ad\u30e3\u30e9\u30af\u30bf\u30fc\u753b\u50cf\u3092\u30c7\u30d5\u30a9\u30eb\u30c8\u306b\u623b\u3059API\napp.post(&#39;\/api\/settings\/reset-character&#39;, express.json(), (req, res) =&gt; {\n  try {\n    \/\/ \u30ab\u30b9\u30bf\u30e0\u753b\u50cf\u304c\u3042\u308c\u3070\u524a\u9664\n    const customImagePath = path.join(__dirname, &#39;public&#39;, &#39;images&#39;, &#39;brave-character-custom.png&#39;);\n    const customJpgPath = path.join(__dirname, &#39;public&#39;, &#39;images&#39;, &#39;brave-character-custom.jpg&#39;);\n    const customSvgPath = path.join(__dirname, &#39;public&#39;, &#39;images&#39;, &#39;brave-character-custom.svg&#39;);\n    const customGifPath = path.join(__dirname, &#39;public&#39;, &#39;images&#39;, &#39;brave-character-custom.gif&#39;);\n    \n    if (fs.existsSync(customImagePath)) {\n      fs.unlinkSync(customImagePath);\n      console.log(&#39;\u30ab\u30b9\u30bf\u30e0\u30ad\u30e3\u30e9\u30af\u30bf\u30fc\u753b\u50cf(PNG)\u3092\u524a\u9664\u3057\u307e\u3057\u305f&#39;);\n    }\n    \n    if (fs.existsSync(customJpgPath)) {\n      fs.unlinkSync(customJpgPath);\n      console.log(&#39;\u30ab\u30b9\u30bf\u30e0\u30ad\u30e3\u30e9\u30af\u30bf\u30fc\u753b\u50cf(JPG)\u3092\u524a\u9664\u3057\u307e\u3057\u305f&#39;);\n    }\n    \n    if (fs.existsSync(customSvgPath)) {\n      fs.unlinkSync(customSvgPath);\n      console.log(&#39;\u30ab\u30b9\u30bf\u30e0\u30ad\u30e3\u30e9\u30af\u30bf\u30fc\u753b\u50cf(SVG)\u3092\u524a\u9664\u3057\u307e\u3057\u305f&#39;);\n    }\n    \n    if (fs.existsSync(customGifPath)) {\n      fs.unlinkSync(customGifPath);\n      console.log(&#39;\u30ab\u30b9\u30bf\u30e0\u30ad\u30e3\u30e9\u30af\u30bf\u30fc\u753b\u50cf(GIF)\u3092\u524a\u9664\u3057\u307e\u3057\u305f&#39;);\n    }\n    \n    \/\/ \u8a2d\u5b9a\u304b\u3089\u30ab\u30b9\u30bf\u30e0\u753b\u50cf\u8a2d\u5b9a\u3092\u524a\u9664\n    const settings = getSettings();\n    delete settings.characterImage;\n    saveSettings(settings);\n    \n    res.json({ \n      success: true, \n      message: &#39;\u30ad\u30e3\u30e9\u30af\u30bf\u30fc\u753b\u50cf\u304c\u30c7\u30d5\u30a9\u30eb\u30c8\u306b\u30ea\u30bb\u30c3\u30c8\u3055\u308c\u307e\u3057\u305f&#39;\n    });\n  } catch (error) {\n    console.error(&#39;\u30ad\u30e3\u30e9\u30af\u30bf\u30fc\u30ea\u30bb\u30c3\u30c8\u30a8\u30e9\u30fc:&#39;, error);\n    res.status(500).json({ success: false, error: error.message });\n  }\n});\n\n\/\/ \u8a2d\u5b9a\u60c5\u5831\u53d6\u5f97API\n\/\/ ... existing code ...\n\n\/\/ \u4eca\u65e5\u306e\u30a4\u30c1\u63a8\u3057Braver\u3092\u9078\u51fa\u3059\u308b\u30e1\u30a4\u30f3\u306e\u51e6\u7406\nasync function resetAndSelectDailyPickupUser() {\n  try {\n    \/\/ \u65e5\u672c\u6642\u9593\u3067\u4eca\u65e5\u306e\u65e5\u4ed8\u3092\u53d6\u5f97\n    const today = getJSTDateString();\n    console.log(`\u4eca\u65e5(${today})\u306e\u30a4\u30c1\u63a8\u3057braver\u30c7\u30fc\u30bf\u3092\u30ea\u30bb\u30c3\u30c8\u3057\u307e\u3059`);\n    \n    \/\/ \u4eca\u65e5\u306e\u30d4\u30c3\u30af\u30a2\u30c3\u30d7\u30c7\u30fc\u30bf\u3092\u524a\u9664\n    await new Promise((resolve, reject) =&gt; {\n      db.run(&#39;DELETE FROM daily_pickup_user WHERE pickup_date = ?&#39;, [today], function(err) {\n        if (err) {\n          console.error(&#39;\u30a4\u30c1\u63a8\u3057braver\u30c7\u30fc\u30bf\u306e\u30ea\u30bb\u30c3\u30c8\u306b\u5931\u6557:&#39;, err);\n          reject(err);\n          return;\n        }\n        \n        console.log(`\u4eca\u65e5\u306e\u30a4\u30c1\u63a8\u3057braver\u3092\u30ea\u30bb\u30c3\u30c8\u3057\u307e\u3057\u305f: ${this.changes}\u4ef6\u306e\u30c7\u30fc\u30bf\u3092\u524a\u9664`);\n        resolve();\n      });\n    });\n    \n    \/\/ \u30a4\u30c1\u63a8\u3057Braver\u3092\u9078\u51fa\n    await selectDailyPickupUser();\n    \n    console.log(&#39;\u30a4\u30c1\u63a8\u3057braver\u306e\u518d\u9078\u51fa\u304c\u5b8c\u4e86\u3057\u307e\u3057\u305f&#39;);\n    return true;\n  } catch (error) {\n    console.error(&#39;\u30a4\u30c1\u63a8\u3057braver\u9078\u51fa\u30a8\u30e9\u30fc:&#39;, error);\n    return false;\n  }\n}\n\n\/\/ selectDailyPickupUser\u95a2\u6570\nasync function selectDailyPickupUser() {\n  console.log(&#39;\u30a4\u30c1\u63a8\u3057braver\u3092\u9078\u51fa\u4e2d...&#39;);\n  \n  try {\n    \/\/ \u4eca\u65e5\u306e\u65e5\u4ed8 (JST)\n    const today = getJSTDateString();\n    \n    \/\/ \u30c6\u30fc\u30d6\u30eb\u69cb\u9020\u3092\u78ba\u8a8d\n    console.log(&#39;daily_pickup_user\u30c6\u30fc\u30d6\u30eb\u306e\u69cb\u9020\u3092\u78ba\u8a8d\u3057\u3066\u3044\u307e\u3059...&#39;);\n    const tableInfo = await new Promise((resolve, reject) =&gt; {\n      db.all(&#39;PRAGMA table_info(daily_pickup_user)&#39;, (err, rows) =&gt; {\n        if (err) {\n          reject(err);\n          return;\n        }\n        resolve(rows);\n      });\n    });\n    \n    \/\/ \u30ab\u30e9\u30e0\u540d\u306e\u30ea\u30b9\u30c8\u3092\u8868\u793a\n    const columns = tableInfo.map(col =&gt; col.name).join(&#39;, &#39;);\n    console.log(`daily_pickup_user\u306e\u65e2\u5b58\u30ab\u30e9\u30e0: ${columns}`);\n    \n    \/\/ \u672c\u65e5\u306e\u30c7\u30fc\u30bf\u3092\u30ea\u30bb\u30c3\u30c8\n    console.log(`\u672c\u65e5\u306e\u30a4\u30c1\u63a8\u3057braver\u30c7\u30fc\u30bf\u3092\u30ea\u30bb\u30c3\u30c8\u3057\u307e\u3059: ${today}`);\n    await new Promise((resolve, reject) =&gt; {\n      db.run(&#39;DELETE FROM daily_pickup_user WHERE pickup_date = ?&#39;, [today], function(err) {\n        if (err) {\n          console.error(&#39;\u30a4\u30c1\u63a8\u3057braver\u30c7\u30fc\u30bf\u306e\u30ea\u30bb\u30c3\u30c8\u306b\u5931\u6557:&#39;, err);\n          reject(err);\n          return;\n        }\n        console.log(`\u672c\u65e5\u306e\u30a4\u30c1\u63a8\u3057braver\u30c7\u30fc\u30bf\u3092\u30ea\u30bb\u30c3\u30c8\u3057\u307e\u3057\u305f: ${this.changes}\u4ef6\u524a\u9664`);\n        resolve();\n      });\n    });\n    \n    \/\/ \u5b9f\u969b\u306e\u30c7\u30fc\u30bf\u3092\u53d6\u5f97\u3059\u308bSQL\u30af\u30a8\u30ea\n    const query = `\n      SELECT \n        u.user_id, \n        u.name, \n        u.real_name, \n        u.display_name, \n        u.avatar,\n        (SELECT COUNT(*) FROM slack_messages WHERE user_id = u.user_id) AS message_count,\n        (SELECT COUNT(*) FROM slack_thread_replies WHERE user_id = u.user_id) AS reply_count,\n        (\n          SELECT COUNT(*) FROM slack_messages \n          WHERE user_id = u.user_id \n          AND (text LIKE &#39;\/yolo%&#39; OR text LIKE &#39;\/Yolo%&#39;)\n        ) AS attendance_count,\n        (\n          SELECT COUNT(*) FROM slack_messages \n          WHERE user_id = u.user_id \n          AND (text LIKE &#39;\/yolo_office%&#39; OR text LIKE &#39;\/Yolo_office%&#39;)\n        ) AS office_count,\n        (\n          SELECT COUNT(*) FROM slack_messages\n          WHERE user_id = u.user_id\n          AND length(text) &gt;= 50\n        ) AS report_count,\n        (\n          SELECT COUNT(*) FROM slack_reactions\n          WHERE user_id = u.user_id\n        ) AS sent_reaction_count,\n        (\n          SELECT COUNT(*) FROM slack_reactions r\n          JOIN slack_messages m ON r.message_id = m.message_id\n          WHERE m.user_id = u.user_id\n        ) AS received_reaction_count\n      FROM \n        slack_users u\n      WHERE \n        u.is_bot = 0\n        AND (\n          SELECT COUNT(*) FROM slack_reactions r\n          JOIN slack_messages m ON r.message_id = m.message_id\n          WHERE m.user_id = u.user_id\n        ) &gt;= 30\n      ORDER BY \n        RANDOM()\n      LIMIT 3\n    `;\n    \n    console.log(&#39;\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u657030\u4ee5\u4e0a\u306e\u30e6\u30fc\u30b6\u30fc\u304b\u3089\u30e9\u30f3\u30c0\u30e0\u306b3\u4eba\u9078\u51fa\u3059\u308b\u30af\u30a8\u30ea\u3092\u5b9f\u884c\u3057\u307e\u3059&#39;);\n    \n    \/\/ \u30e6\u30fc\u30b6\u30fc\u3092\u53d6\u5f97\u3057\u3066\u30a4\u30c1\u63a8\u3057braver\u306b\u767b\u9332\n    const users = await new Promise((resolve, reject) =&gt; {\n      db.all(query, [], (err, rows) =&gt; {\n        if (err) {\n          console.error(&#39;\u30e6\u30fc\u30b6\u30fc\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, err);\n          reject(err);\n          return;\n        }\n        resolve(rows);\n      });\n    });\n    \n    console.log(`\u30a4\u30c1\u63a8\u3057braver\u5019\u88dc: ${users.length}\u4eba`);\n    \n    \/\/ \u5404\u30e6\u30fc\u30b6\u30fc\u3092\u30a4\u30c1\u63a8\u3057braver\u30c6\u30fc\u30d6\u30eb\u306b\u767b\u9332\n    for (const row of users) {\n      try {\n        const userName = row.display_name || row.real_name || row.name || &#39;Unknown&#39;;\n        console.log(`\u30a4\u30c1\u63a8\u3057braver\u306b\u767b\u9332: ${userName} (${row.user_id}) - \u30e1\u30c3\u30bb\u30fc\u30b8\u6570: ${row.message_count}, \u8fd4\u4fe1\u6570: ${row.reply_count}, \u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u6570: ${row.received_reaction_count}`);\n        \n        \/\/ \u30a2\u30d0\u30bf\u30fcURL\u3092\u9ad8\u89e3\u50cf\u5ea6\u7248\u306b\u5909\u63db\n        let avatarUrl = row.avatar;\n        if (avatarUrl && avatarUrl.includes(&#39;_72&#39;)) {\n          avatarUrl = avatarUrl.replace(&#39;_72&#39;, &#39;_512&#39;);\n          console.log(`\u30a2\u30d0\u30bf\u30fc\u753b\u50cfURL\u5909\u63db: ${row.avatar} -&gt; ${avatarUrl}`);\n        }\n        \n        \/\/ AI\u5206\u6790\u30c6\u30ad\u30b9\u30c8\u3092\u751f\u6210\n        let analysis = &#39;&#39;;\n        try {\n          \/\/ \u30e6\u30fc\u30b6\u30fc\u306e\u6700\u65b0\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u53d6\u5f97\uff08\u6700\u592710\u4ef6\uff09\n          const userMessages = await new Promise((resolve, reject) =&gt; {\n            db.all(\n              `SELECT text, timestamp FROM slack_messages WHERE user_id = ? ORDER BY timestamp DESC LIMIT 10`,\n              [row.user_id],\n              (err, messages) =&gt; {\n                if (err) {\n                  console.error(`\u30e6\u30fc\u30b6\u30fc\u30e1\u30c3\u30bb\u30fc\u30b8\u53d6\u5f97\u30a8\u30e9\u30fc (${row.user_id}):`, err);\n                  reject(err);\n                } else {\n                  resolve(messages);\n                }\n              }\n            );\n          });\n          \n          \/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u304c\u3042\u308c\u3070AI\u5206\u6790\u3092\u5b9f\u884c\n          if (userMessages && userMessages.length &gt; 0) {\n            console.log(`${userName}\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u30c7\u30fc\u30bf\u3092AI\u3067\u5206\u6790\u3057\u307e\u3059 (${userMessages.length}\u4ef6)...`);\n            \n            \/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u30c6\u30ad\u30b9\u30c8\u3092\u7d50\u5408\n            const messagesText = userMessages.map(m =&gt; m.text).join(&#39;\\n\\n&#39;);\n            \n            \/\/ AI\u5206\u6790\u306e\u30d7\u30ed\u30f3\u30d7\u30c8\n            const prompt = currentAiAnalysisPrompt + `\\n\\n\u30e6\u30fc\u30b6\u30fc\u540d: ${userName}\\n\u30e1\u30c3\u30bb\u30fc\u30b8\u6570: ${row.message_count}\u4ef6\\n\u8fd4\u4fe1\u6570: ${row.reply_count}\u4ef6\\n\u51fa\u52e4\u6570: ${row.attendance_count}\u65e5\\n\u30ea\u30a2\u30eb\u51fa\u793e: ${row.office_count}\u65e5\\n\u30b9\u30bf\u30f3\u30d7\u6570: ${row.received_reaction_count}\u500b\\n\\n\u3010\u30e1\u30c3\u30bb\u30fc\u30b8\u4e00\u89a7\u3011\\n${messagesText}\\n\\n\u3053\u306e\u30e6\u30fc\u30b6\u30fc\u306e\u7279\u5fb4\u3092\u7c21\u6f54\u306b\u307e\u3068\u3081\u3066\u3001\u6539\u884c\u3092\u6700\u5c0f\u9650\u306b\u3057\u3001\u6bb5\u843d\u9593\u306e\u7a7a\u767d\u884c\u306a\u3057\u3067350\u6587\u5b57\u4ee5\u5185\u3067\u5206\u6790\u3057\u3066\u304f\u3060\u3055\u3044\u3002`;\n            \n            try {\n              \/\/ Anthropic API\u3092\u547c\u3073\u51fa\u3057\n              if (anthropic && process.env.ANTHROPIC_API_KEY) {\n                const response = await anthropic.messages.create({\n                  model: &quot;claude-3-haiku-20240307&quot;,\n                  max_tokens: 1000,\n                  messages: [\n                    { role: &quot;user&quot;, content: prompt }\n                  ],\n                  temperature: 0.7,\n                });\n                \n                \/\/ \u6539\u884c\u3084\u6bb5\u843d\u9593\u306e\u7a7a\u767d\u884c\u3092\u53d6\u308a\u9664\u3044\u3066\u5206\u6790\u30c6\u30ad\u30b9\u30c8\u3092\u6574\u5f62\n                analysis = response.content[0].text\n                  .replace(\/\\n\\s*\\n\/g, &#39;\\n&#39;) \/\/ \u7a7a\u767d\u884c\u3092\u524a\u9664\n                  .replace(\/\\n+\/g, &#39; &#39;) \/\/ \u6b8b\u308a\u306e\u6539\u884c\u3092\u7a7a\u767d\u306b\u7f6e\u63db\n                  .trim();\n                  \n                console.log(`AI\u5206\u6790\u5b8c\u4e86 (${analysis.length}\u6587\u5b57)`);\n              } else {\n                throw new Error(&#39;Anthropic API\u304c\u8a2d\u5b9a\u3055\u308c\u3066\u3044\u307e\u305b\u3093&#39;);\n              }\n            } catch (aiError) {\n              \/\/ AI\u5206\u6790\u30a8\u30e9\u30fc\u6642\u306f\u30d5\u30a9\u30fc\u30eb\u30d0\u30c3\u30af\n              console.error(`AI\u5206\u6790\u30a8\u30e9\u30fc (${row.user_id}):`, aiError);\n              const messageCount = row.message_count || 0;\n              const replyCount = row.reply_count || 0;\n              const attendanceCount = row.attendance_count || 0;\n              const officeCount = row.office_count || 0;\n              const receivedReactionCount = row.received_reaction_count || 0;\n              \n              analysis = `${userName}\u3055\u3093\u306f\u3001\u6295\u7a3f\u6570${messageCount}\u4ef6\u3001\u8fd4\u4fe1\u6570${replyCount}\u4ef6\u3068\u6d3b\u767a\u306b\u6d3b\u52d5\u3055\u308c\u3066\u3044\u307e\u3059\u3002\u51fa\u52e4\u6570\u306f${attendanceCount}\u65e5\u3001\u30ea\u30a2\u30eb\u51fa\u793e\u306f${officeCount}\u65e5\u3067\u3059\u3002\u30c1\u30fc\u30e0\u5185\u3067\u306e\u7a4d\u6975\u7684\u306a\u30b3\u30df\u30e5\u30cb\u30b1\u30fc\u30b7\u30e7\u30f3\u3068\u5354\u529b\u7684\u306a\u59ff\u52e2\u304c\u7279\u5fb4\u7684\u3067\u3059\u3002\u30b9\u30bf\u30f3\u30d7\u6570${receivedReactionCount}\u500b\u306e\u4eba\u6c17\u8005\u3067\u3059\uff01`;\n            }\n          } else {\n            \/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u304c\u306a\u3044\u5834\u5408\u306f\u30c7\u30d5\u30a9\u30eb\u30c8\u306e\u5206\u6790\n            console.log(`${userName}\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u30c7\u30fc\u30bf\u304c\u898b\u3064\u304b\u308a\u307e\u305b\u3093\u3002\u30c7\u30d5\u30a9\u30eb\u30c8\u306e\u5206\u6790\u3092\u4f7f\u7528\u3057\u307e\u3059\u3002`);\n            const messageCount = row.message_count || 0;\n            const replyCount = row.reply_count || 0;\n            const attendanceCount = row.attendance_count || 0;\n            const officeCount = row.office_count || 0;\n            const receivedReactionCount = row.received_reaction_count || 0;\n            \n            analysis = `${userName}\u3055\u3093\u306f\u3001\u6295\u7a3f\u6570${messageCount}\u4ef6\u3001\u8fd4\u4fe1\u6570${replyCount}\u4ef6\u3068\u6d3b\u767a\u306b\u6d3b\u52d5\u3055\u308c\u3066\u3044\u307e\u3059\u3002\u51fa\u52e4\u6570\u306f${attendanceCount}\u65e5\u3001\u30ea\u30a2\u30eb\u51fa\u793e\u306f${officeCount}\u65e5\u3067\u3059\u3002\u30c1\u30fc\u30e0\u5185\u3067\u306e\u7a4d\u6975\u7684\u306a\u30b3\u30df\u30e5\u30cb\u30b1\u30fc\u30b7\u30e7\u30f3\u3068\u5354\u529b\u7684\u306a\u59ff\u52e2\u304c\u7279\u5fb4\u7684\u3067\u3059\u3002\u30b9\u30bf\u30f3\u30d7\u6570${receivedReactionCount}\u500b\u306e\u4eba\u6c17\u8005\u3067\u3059\uff01`;\n          }\n        } catch (analysisError) {\n          \/\/ \u30a8\u30e9\u30fc\u6642\u306f\u30c7\u30d5\u30a9\u30eb\u30c8\u306e\u5206\u6790\u3092\u4f7f\u7528\n          console.error(`\u5206\u6790\u51e6\u7406\u30a8\u30e9\u30fc (${row.user_id}):`, analysisError);\n          const messageCount = row.message_count || 0;\n          const replyCount = row.reply_count || 0;\n          const attendanceCount = row.attendance_count || 0;\n          const officeCount = row.office_count || 0;\n          const receivedReactionCount = row.received_reaction_count || 0;\n          \n          analysis = `${userName}\u3055\u3093\u306f\u3001\u6295\u7a3f\u6570${messageCount}\u4ef6\u3001\u8fd4\u4fe1\u6570${replyCount}\u4ef6\u3068\u6d3b\u767a\u306b\u6d3b\u52d5\u3055\u308c\u3066\u3044\u307e\u3059\u3002\u51fa\u52e4\u6570\u306f${attendanceCount}\u65e5\u3001\u30ea\u30a2\u30eb\u51fa\u793e\u306f${officeCount}\u65e5\u3067\u3059\u3002\u30c1\u30fc\u30e0\u5185\u3067\u306e\u7a4d\u6975\u7684\u306a\u30b3\u30df\u30e5\u30cb\u30b1\u30fc\u30b7\u30e7\u30f3\u3068\u5354\u529b\u7684\u306a\u59ff\u52e2\u304c\u7279\u5fb4\u7684\u3067\u3059\u3002\u30b9\u30bf\u30f3\u30d7\u6570${receivedReactionCount}\u500b\u306e\u4eba\u6c17\u8005\u3067\u3059\uff01`;\n        }\n        \n        \/\/ \u30c7\u30fc\u30bf\u3092\u4fdd\u5b58\n        await new Promise((resolve, reject) =&gt; {\n          const now = new Date();\n          const currentTimestamp = now.toISOString().replace(&#39;T&#39;, &#39; &#39;).substring(0, 19);\n          \n          db.run(\n            &#39;INSERT INTO daily_pickup_user (user_id, name, real_name, display_name, avatar, message_count, reply_count, attendance_count, office_count, report_count, sent_reaction_count, received_reaction_count, analysis, pickup_date, created_at) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)&#39;,\n            [\n              row.user_id, \n              row.name, \n              row.real_name, \n              row.display_name,\n              avatarUrl,\n              row.message_count || 0,\n              row.reply_count || 0,\n              row.attendance_count || 0,\n              row.office_count || 0,\n              row.report_count || 0,\n              row.sent_reaction_count || 0,\n              row.received_reaction_count || 0,\n              analysis,\n              today,\n              currentTimestamp\n            ],\n            (err) =&gt; {\n              if (err) {\n                console.error(&#39;\u30a4\u30c1\u63a8\u3057braver\u767b\u9332\u30a8\u30e9\u30fc:&#39;, err);\n                reject(err);\n              } else {\n                resolve();\n              }\n            }\n          );\n        });\n      } catch (userError) {\n        console.error(`\u30e6\u30fc\u30b6\u30fc\u767b\u9332\u30a8\u30e9\u30fc (${row.user_id}):`, userError);\n      }\n    }\n    \n    console.log(`\u672c\u65e5\u306e\u30a4\u30c1\u63a8\u3057braver\u9078\u51fa\u5b8c\u4e86: ${users.length}\u4eba`);\n    \n  } catch (error) {\n    console.error(&#39;\u30a4\u30c1\u63a8\u3057braver\u9078\u51fa\u30a8\u30e9\u30fc:&#39;, error);\n  }\n}\n\n\/\/ Slack\u30c7\u30fc\u30bf\u3092\u540c\u671f\u3059\u308bAPI\napp.get(&#39;\/api\/sync-slack-data&#39;, async (req, res) =&gt; {\n  try {\n    if (!slackToken || !slackChannelIds.length) {\n      return res.status(500).json({ \n        error: &#39;Slack API\u306e\u8a2d\u5b9a\u304c\u4e0d\u8db3\u3057\u3066\u3044\u307e\u3059\u3002\u74b0\u5883\u5909\u6570\u3092\u78ba\u8a8d\u3057\u3066\u304f\u3060\u3055\u3044\u3002&#39; \n      });\n    }\n\n    console.log(&#39;Slack\u30c7\u30fc\u30bf\u540c\u671f\u3092\u958b\u59cb...&#39;);\n    \n    \/\/ \u540c\u671f\u3059\u308b\u30c1\u30e3\u30f3\u30cd\u30eb\u306e\u6307\u5b9a\uff08\u7279\u5b9a\u30c1\u30e3\u30f3\u30cd\u30eb\u307e\u305f\u306f\u5168\u30c1\u30e3\u30f3\u30cd\u30eb\uff09\n    const requestedChannelId = req.query.channelId;\n    let channelsToSync = [];\n    \n    if (requestedChannelId) {\n      \/\/ \u7279\u5b9a\u306e\u30c1\u30e3\u30f3\u30cd\u30eb\u304c\u6307\u5b9a\u3055\u308c\u305f\u5834\u5408\n      channelsToSync = [requestedChannelId];\n      console.log(`\u6307\u5b9a\u3055\u308c\u305f\u30c1\u30e3\u30f3\u30cd\u30eb ${requestedChannelId} \u306e\u307f\u3092\u540c\u671f\u3057\u307e\u3059`);\n    } else {\n      \/\/ \u3059\u3079\u3066\u306e\u30c1\u30e3\u30f3\u30cd\u30eb\u3092\u540c\u671f\n      channelsToSync = global.slackChannelIds || slackChannelIds;\n      console.log(`\u8a2d\u5b9a\u3055\u308c\u3066\u3044\u308b\u5168\u30c1\u30e3\u30f3\u30cd\u30eb (${channelsToSync.length}\u4ef6) \u3092\u540c\u671f\u3057\u307e\u3059: ${channelsToSync.join(&#39;, &#39;)}`);\n    }\n    \n    \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u3054\u3068\u306e\u540c\u671f\u7d50\u679c\u3092\u4fdd\u5b58\u3059\u308b\u914d\u5217\n    const syncResults = [];\n    \n    \/\/ \u5404\u30c1\u30e3\u30f3\u30cd\u30eb\u3092\u9806\u756a\u306b\u51e6\u7406\n    for (const channelId of channelsToSync) {\n      console.log(`\u30c1\u30e3\u30f3\u30cd\u30eb ${channelId} \u306e\u540c\u671f\u3092\u958b\u59cb\u3057\u307e\u3059...`);\n      \n      try {\n        \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u60c5\u5831\u3092\u53d6\u5f97\u3057\u3066\u4fdd\u5b58\n        let channelName = &#39;Unknown Channel&#39;;\n        let channelInfo;\n        \n        try {\n          channelInfo = await slackClient.conversations.info({\n            channel: channelId\n          });\n          \n          channelName = channelInfo.channel.name || &#39;Unknown Channel&#39;;\n          console.log(`\u30c1\u30e3\u30f3\u30cd\u30eb\u60c5\u5831\u53d6\u5f97: ${channelName} (${channelId})`);\n          \n          \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u60c5\u5831\u3092DB\u306b\u4fdd\u5b58\n          const channelStmt = db.prepare(`\n            INSERT OR REPLACE INTO slack_channels (channel_id, name)\n            VALUES (?, ?)\n          `);\n          \n          channelStmt.run(channelId, channelName);\n          channelStmt.finalize();\n          console.log(`\u30c1\u30e3\u30f3\u30cd\u30eb\u60c5\u5831\u3092\u4fdd\u5b58: ${channelName}`);\n        } catch (channelError) {\n          console.error(`\u30c1\u30e3\u30f3\u30cd\u30eb ${channelId} \u306e\u60c5\u5831\u53d6\u5f97\u30a8\u30e9\u30fc:`, channelError);\n          continue; \/\/ \u3053\u306e\u30c1\u30e3\u30f3\u30cd\u30eb\u3092\u30b9\u30ad\u30c3\u30d7\u3057\u3066\u6b21\u3078\n        }\n        \n        \/\/ \u904e\u53bb3\u30f6\u6708\u306e\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7\u3092\u8a08\u7b97\n        const threeMonthsAgo = new Date();\n        threeMonthsAgo.setMonth(threeMonthsAgo.getMonth() - 3); \/\/ 3\u30f6\u6708\u524d\u306b\u5909\u66f4\n        const oldestTimestamp = threeMonthsAgo.getTime() \/ 1000; \/\/ Unix\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7\u306fUNIX\u6642\u9593\uff08\u79d2\uff09\n        console.log(`3\u30f6\u6708\u524d\u306e\u65e5\u6642: ${threeMonthsAgo.toLocaleString()}, \u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7: ${oldestTimestamp}`);\n        \n        \/\/ DB\u306b\u4fdd\u5b58\u6e08\u307f\u306e\u6700\u65b0\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7\u3092\u53d6\u5f97\n        const latestMsg = await new Promise((resolve, reject) =&gt; {\n          db.get(&#39;SELECT MAX(timestamp) AS latest FROM slack_messages WHERE channel_id = ?&#39;, [channelId], (err, row) =&gt; {\n            if (err) {\n              console.error(&#39;\u6700\u65b0\u30e1\u30c3\u30bb\u30fc\u30b8\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, err);\n              resolve(null);\n            } else {\n              resolve(row?.latest || null);\n            }\n          });\n        });\n        \n        console.log(`\u65e2\u5b58\u306e\u6700\u65b0\u30e1\u30c3\u30bb\u30fc\u30b8\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7: ${latestMsg || &#39;\u306a\u3057&#39;}`);\n        \n        \/\/ \u4fdd\u5b58\u6e08\u307f\u306e\u30e1\u30c3\u30bb\u30fc\u30b8ID\u3092\u53d6\u5f97\uff08\u91cd\u8907\u30c1\u30a7\u30c3\u30af\u7528\uff09\n        const existingMessageIds = new Set();\n        await new Promise((resolve, reject) =&gt; {\n          db.each(&#39;SELECT message_id FROM slack_messages WHERE channel_id = ?&#39;, [channelId], \n            (err, row) =&gt; {\n              if (!err && row && row.message_id) {\n                existingMessageIds.add(row.message_id);\n              }\n            },\n            (err) =&gt; {\n              if (err) {\n                console.error(&#39;\u65e2\u5b58\u30e1\u30c3\u30bb\u30fc\u30b8ID\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, err);\n              } else {\n                console.log(`${existingMessageIds.size}\u4ef6\u306e\u65e2\u5b58\u30e1\u30c3\u30bb\u30fc\u30b8ID\u3092\u53d6\u5f97`);\n              }\n              resolve();\n            }\n          );\n        });\n        \n        \/\/ \u4fdd\u5b58\u6e08\u307f\u306e\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1ID\u3092\u53d6\u5f97\uff08\u91cd\u8907\u30c1\u30a7\u30c3\u30af\u7528\uff09\n        const existingReplyIds = new Set();\n        await new Promise((resolve, reject) =&gt; {\n          db.each(&#39;SELECT message_id FROM slack_thread_replies WHERE channel_id = ?&#39;, [channelId], \n            (err, row) =&gt; {\n              if (!err && row && row.message_id) {\n                existingReplyIds.add(row.message_id);\n              }\n            },\n            (err) =&gt; {\n              if (err) {\n                console.error(&#39;\u65e2\u5b58\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1ID\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, err);\n              } else {\n                console.log(`${existingReplyIds.size}\u4ef6\u306e\u65e2\u5b58\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1ID\u3092\u53d6\u5f97`);\n              }\n              resolve();\n            }\n          );\n        });\n        \n        \/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u4fdd\u5b58\u306e\u30d7\u30ea\u30da\u30a2\u30c9\u30b9\u30c6\u30fc\u30c8\u30e1\u30f3\u30c8\n        const msgStmt = db.prepare(`\n          INSERT OR IGNORE INTO slack_messages (message_id, user_id, channel_id, text, timestamp)\n          VALUES (?, ?, ?, ?, ?)\n        `);\n        \n        \/\/ \u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u4fdd\u5b58\u306e\u30d7\u30ea\u30da\u30a2\u30c9\u30b9\u30c6\u30fc\u30c8\u30e1\u30f3\u30c8\n        const replyStmt = db.prepare(`\n          INSERT OR IGNORE INTO slack_thread_replies (message_id, parent_message_id, user_id, channel_id, text, timestamp, reply_id, thread_ts, channel_name)\n          VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)\n        `);\n        \n        \/\/ \u904e\u53bb\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u53d6\u5f97\uff08\u521d\u56de\u306f\u904e\u53bb3\u30f6\u6708\u5206\u3001\u305d\u308c\u4ee5\u964d\u306f\u5dee\u5206\u306e\u307f\uff09\n        let cursor = null;\n        let totalMessages = 0;\n        let newMessages = 0;\n        let skippedMessages = 0;\n        let totalReplies = 0;\n        let newReplies = 0;\n        let skippedReplies = 0;\n        let hasMore = true;\n        let oldestMsgDate = null;\n        \n        while (hasMore) {\n          try {\n            \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u5c65\u6b74\u3092\u53d6\u5f97\n            const result = await slackClient.conversations.history({\n              channel: channelId,\n              cursor: cursor,\n              limit: 200, \/\/ 1\u56de\u306e\u30ea\u30af\u30a8\u30b9\u30c8\u3067\u6700\u5927200\u4ef6\n              oldest: Math.max(oldestTimestamp, latestMsg || 0), \/\/ 3\u30f6\u6708\u524d\u304b\u6700\u65b0\u306e\u4fdd\u5b58\u6e08\u307f\u30e1\u30c3\u30bb\u30fc\u30b8\u306e\u65b0\u3057\u3044\u65b9\u3092\u57fa\u6e96\u306b\u3059\u308b\n              include_all_metadata: true \/\/ \u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u60c5\u5831\u306a\u3069\u3059\u3079\u3066\u306e\u30e1\u30bf\u30c7\u30fc\u30bf\u3092\u542b\u3081\u308b\n            });\n            \n            \/\/ API\u30ec\u30b9\u30dd\u30f3\u30b9\u306e\u30b5\u30f3\u30d7\u30eb\u3092\u30ed\u30b0\u306b\u51fa\u529b\uff08\u30c7\u30d0\u30c3\u30b0\u7528\uff09\n            if (result.messages && result.messages.length &gt; 0) {\n              console.log(`API\u30ec\u30b9\u30dd\u30f3\u30b9\u30b5\u30f3\u30d7\u30eb: \u6700\u521d\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u306e\u30d5\u30a3\u30fc\u30eb\u30c9\u4e00\u89a7:`, Object.keys(result.messages[0]).join(&#39;, &#39;));\n              \n              \/\/ \u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u3092\u542b\u3080\u30e1\u30c3\u30bb\u30fc\u30b8\u304c\u3042\u308b\u304b\u78ba\u8a8d\n              const msgWithReactions = result.messages.find(msg =&gt; msg.reactions && msg.reactions.length &gt; 0);\n              if (msgWithReactions) {\n                console.log(`\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u3092\u542b\u3080\u30e1\u30c3\u30bb\u30fc\u30b8\u4f8b:`, JSON.stringify(msgWithReactions.reactions, null, 2));\n              } else {\n                console.log(`\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u3092\u542b\u3080\u30e1\u30c3\u30bb\u30fc\u30b8\u304c\u898b\u3064\u304b\u308a\u307e\u305b\u3093\u3067\u3057\u305f`);\n              }\n            }\n            \n            totalMessages += result.messages.length;\n            \n            if (result.messages.length === 0) {\n              console.log(&#39;\u65b0\u3057\u3044\u30e1\u30c3\u30bb\u30fc\u30b8\u306f\u3042\u308a\u307e\u305b\u3093&#39;);\n              hasMore = false;\n              break;\n            }\n            \n            \/\/ \u6700\u3082\u53e4\u3044\u30e1\u30c3\u30bb\u30fc\u30b8\u306e\u65e5\u6642\u3092\u8a18\u9332\uff08\u30c7\u30d0\u30c3\u30b0\u7528\uff09\n            if (result.messages.length &gt; 0) {\n              const oldestMsg = result.messages[result.messages.length - 1];\n              oldestMsgDate = new Date(parseFloat(oldestMsg.ts) * 1000).toLocaleString();\n            }\n            \n            \/\/ DB\u306b\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u4fdd\u5b58\n            for (const msg of result.messages) {\n              if (msg.user && msg.ts && msg.text) {\n                \/\/ \u65e2\u306b\u4fdd\u5b58\u6e08\u307f\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u306f\u30b9\u30ad\u30c3\u30d7\n                if (existingMessageIds.has(msg.ts)) {\n                  skippedMessages++;\n                } else {\n                  \/\/ \u65b0\u3057\u3044\u30e1\u30c3\u30bb\u30fc\u30b8\u3092DB\u306b\u4fdd\u5b58\n                  msgStmt.run(msg.ts, msg.user, channelId, msg.text, msg.ts);\n                  newMessages++;\n                  existingMessageIds.add(msg.ts); \/\/ \u91cd\u8907\u30c1\u30a7\u30c3\u30af\u7528\u30bb\u30c3\u30c8\u306b\u8ffd\u52a0\n                }\n                \n                \/\/ \u30b9\u30ec\u30c3\u30c9\u60c5\u5831\u306e\u30c7\u30d0\u30c3\u30b0\n                if (msg.thread_ts) {\n                  console.log(`\u30b9\u30ec\u30c3\u30c9\u691c\u51fa: message_id=${msg.ts}, thread_ts=${msg.thread_ts}, reply_count=${msg.reply_count || 0}`);\n                }\n                \n                \/\/ \u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u304c\u3042\u308b\u5834\u5408\u3001\u305d\u308c\u3089\u3092\u53d6\u5f97\u3057\u3066\u4fdd\u5b58\n                if (msg.thread_ts && msg.reply_count && msg.reply_count &gt; 0) {\n                  try {\n                    console.log(`\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u3092\u53d6\u5f97\u3057\u307e\u3059: thread_ts=${msg.thread_ts}, reply_count=${msg.reply_count}`);\n                    \n                    \/\/ \u3053\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u306e\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u3092\u53d6\u5f97\n                    const repliesResult = await slackClient.conversations.replies({\n                      channel: channelId,\n                      ts: msg.thread_ts,\n                      limit: 200 \/\/ \u4e00\u5ea6\u306b\u53d6\u5f97\u3059\u308b\u6700\u5927\u6570\n                    });\n                    \n                    console.log(`\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u53d6\u5f97\u7d50\u679c: ${repliesResult.messages.length}\u4ef6`);\n                    \n                    \/\/ \u6700\u521d\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\uff08\u89aa\u30e1\u30c3\u30bb\u30fc\u30b8\uff09\u3092\u30b9\u30ad\u30c3\u30d7\u3057\u3001\u8fd4\u4fe1\u306e\u307f\u3092\u51e6\u7406\n                    const replies = repliesResult.messages.slice(1);\n                    totalReplies += replies.length;\n                    \n                    \/\/ \u5404\u8fd4\u4fe1\u3092DB\u306b\u4fdd\u5b58\n                    for (const reply of replies) {\n                      if (reply.user && reply.ts && reply.text) {\n                        \/\/ \u65e2\u306b\u4fdd\u5b58\u6e08\u307f\u306e\u8fd4\u4fe1\u306f\u30b9\u30ad\u30c3\u30d7\n                        if (existingReplyIds.has(reply.ts)) {\n                          skippedReplies++;\n                        } else {\n                          \/\/ \u65b0\u3057\u3044\u8fd4\u4fe1\u3092DB\u306b\u4fdd\u5b58\n                          replyStmt.run(reply.ts, msg.thread_ts, reply.user, channelId, reply.text, reply.ts, reply.reply_id, reply.thread_ts, reply.channel_name);\n                          newReplies++;\n                          existingReplyIds.add(reply.ts); \/\/ \u91cd\u8907\u30c1\u30a7\u30c3\u30af\u7528\u30bb\u30c3\u30c8\u306b\u8ffd\u52a0\n                        }\n                      }\n                    }\n                  } catch (replyError) {\n                    console.error(`\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u306e\u53d6\u5f97\u30a8\u30e9\u30fc: thread_ts=${msg.thread_ts}`, replyError);\n                  }\n                }\n                \n                \/\/ reactions.get API\u3092\u4f7f\u7528\u3057\u3066\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u60c5\u5831\u3092\u53d6\u5f97\n                try {\n                  console.log(`\u30e1\u30c3\u30bb\u30fc\u30b8ID: ${msg.ts} \u306e\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u60c5\u5831\u3092\u53d6\u5f97\u4e2d...`);\n                  \n                  \/\/ \u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u60c5\u5831\u3092\u76f4\u63a5\u53d6\u5f97\n                  const reactionInfo = await slackClient.reactions.get({\n                    channel: channelId,\n                    timestamp: msg.ts,\n                    full: true\n                  });\n                  \n                  \/\/ API\u30ec\u30b9\u30dd\u30f3\u30b9\u306e\u78ba\u8a8d\uff08\u30c7\u30d0\u30c3\u30b0\u7528\uff09\n                  console.log(`reactions.get API\u30ec\u30b9\u30dd\u30f3\u30b9:`, Object.keys(reactionInfo).join(&#39;, &#39;));\n                  \n                  \/\/ \u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u60c5\u5831\u304c\u3042\u308c\u3070\u51e6\u7406\n                  if (reactionInfo.message && reactionInfo.message.reactions && reactionInfo.message.reactions.length &gt; 0) {\n                    console.log(`\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u691c\u51fa: message_id=${msg.ts}, reaction_count=${reactionInfo.message.reactions.length}`);\n                    console.log(`\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u8a73\u7d30:`, JSON.stringify(reactionInfo.message.reactions, null, 2));\n                    \n                    \/\/ \u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u60c5\u5831\u3092DB\u306b\u4fdd\u5b58\u3059\u308b\u30b9\u30c6\u30fc\u30c8\u30e1\u30f3\u30c8\n                    const reactionStmt = db.prepare(`\n                      INSERT OR IGNORE INTO slack_reactions (message_id, user_id, name, timestamp)\n                      VALUES (?, ?, ?, ?)\n                    `);\n                    \n                    let totalSavedReactions = 0;\n                    \n                    \/\/ \u5404\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u3092\u51e6\u7406\n                    for (const reaction of reactionInfo.message.reactions) {\n                      if (reaction.name && reaction.users && reaction.users.length &gt; 0) {\n                        console.log(`\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u300c${reaction.name}\u300d\u306e\u30e6\u30fc\u30b6\u30fc\u6570: ${reaction.users.length}`);\n                        \n                        for (const userId of reaction.users) {\n                          try {\n                            \/\/ \u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u3092DB\u306b\u4fdd\u5b58\uff08\u30a4\u30f3\u30e9\u30a4\u30f3\u3067SQL\u3092\u5b9f\u884c\uff09\n                            db.run(`\n                              INSERT OR IGNORE INTO slack_reactions (message_id, user_id, name, timestamp)\n                              VALUES (?, ?, ?, ?)\n                            `, [msg.ts, userId, reaction.name, msg.ts], function(err) {\n                              if (err) {\n                                console.error(`\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u4fdd\u5b58\u30a8\u30e9\u30fc: ${err.message}`);\n                              } else {\n                                console.log(`\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u4fdd\u5b58\u6210\u529f: message_id=${msg.ts}, user=${userId}, emoji=${reaction.name}`);\n                                totalSavedReactions++;\n                              }\n                            });\n                          } catch (insertError) {\n                            console.error(`\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u633f\u5165\u30a8\u30e9\u30fc: ${insertError.message}`);\n                          }\n                        }\n                      } else {\n                        console.log(`\u8b66\u544a: \u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u300c${reaction.name}\u300d\u306b\u30e6\u30fc\u30b6\u30fc\u60c5\u5831\u304c\u3042\u308a\u307e\u305b\u3093`);\n                      }\n                    }\n                    \n                    \/\/ \u4fdd\u5b58\u7d71\u8a08\u306e\u51fa\u529b\n                    console.log(`\u30e1\u30c3\u30bb\u30fc\u30b8ID: ${msg.ts} \u306b\u3064\u3044\u3066 ${totalSavedReactions}\u4ef6\u306e\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u3092\u4fdd\u5b58\u3057\u307e\u3057\u305f`);\n                    \n                    \/\/ \u30b9\u30c6\u30fc\u30c8\u30e1\u30f3\u30c8\u3092\u30af\u30ed\u30fc\u30ba\n                    \n                    \/\/ API\u30ec\u30fc\u30c8\u5236\u9650\u3092\u907f\u3051\u308b\u305f\u3081\u5c11\u3057\u5f85\u6a5f\n                    await new Promise(resolve =&gt; setTimeout(resolve, 200));\n                  } else {\n                    console.log(`\u30e1\u30c3\u30bb\u30fc\u30b8ID: ${msg.ts} \u306b\u306f\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u304c\u3042\u308a\u307e\u305b\u3093`);\n                    if (reactionInfo.message) {\n                      console.log(`\u30e1\u30c3\u30bb\u30fc\u30b8\u30d5\u30a3\u30fc\u30eb\u30c9:`, Object.keys(reactionInfo.message).join(&#39;, &#39;));\n                    } else {\n                      console.log(`\u30e1\u30c3\u30bb\u30fc\u30b8\u30d5\u30a3\u30fc\u30eb\u30c9\u304c\u5b58\u5728\u3057\u307e\u305b\u3093`);\n                    }\n                  }\n                } catch (reactionError) {\n                  console.error(`\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u53d6\u5f97\u30a8\u30e9\u30fc (message_id=${msg.ts}):`, reactionError);\n                }\n              }\n            }\n            \n            \/\/ \u6b21\u306e\u30da\u30fc\u30b8\u304c\u3042\u308b\u304b\u78ba\u8a8d\n            cursor = result.response_metadata?.next_cursor;\n            hasMore = Boolean(cursor);\n            \n            console.log(`\u6b21\u306e\u30ab\u30fc\u30bd\u30eb: ${cursor || &#39;\u306a\u3057&#39;}`);\n            \n          } catch (historyError) {\n            console.error(`\u30c1\u30e3\u30f3\u30cd\u30eb\u5c65\u6b74\u306e\u53d6\u5f97\u30a8\u30e9\u30fc: ${channelId}`, historyError);\n            hasMore = false;\n          }\n        }\n        \n        \/\/ \u30d7\u30ea\u30da\u30a2\u30c9\u30b9\u30c6\u30fc\u30c8\u30e1\u30f3\u30c8\u3092\u30af\u30ed\u30fc\u30ba\n        msgStmt.finalize();\n        replyStmt.finalize();\n        \n        console.log(`\u30c1\u30e3\u30f3\u30cd\u30eb ${channelName} (${channelId}) \u306e\u540c\u671f\u304c\u5b8c\u4e86\u3057\u307e\u3057\u305f\u3002\u30e1\u30c3\u30bb\u30fc\u30b8: ${newMessages}\u4ef6\uff08\u30b9\u30ad\u30c3\u30d7: ${skippedMessages}\u4ef6\uff09\u3001\u8fd4\u4fe1: ${newReplies}\u4ef6\uff08\u30b9\u30ad\u30c3\u30d7: ${skippedReplies}\u4ef6\uff09`);\n        \n        \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u3054\u3068\u306e\u7d50\u679c\u3092\u4fdd\u5b58\n        syncResults.push({\n          channelId,\n          channelName,\n          newMessages,\n          skippedMessages,\n          newReplies,\n          skippedReplies,\n          totalMessages,\n          totalReplies,\n          oldestMsgDate\n        });\n        \n      } catch (channelSyncError) {\n        console.error(`\u30c1\u30e3\u30f3\u30cd\u30eb ${channelId} \u306e\u540c\u671f\u30a8\u30e9\u30fc:`, channelSyncError);\n        syncResults.push({\n          channelId,\n          error: channelSyncError.message\n        });\n      }\n    }\n    \n    \/\/ \u30a4\u30c1\u63a8\u3057braver\u306e\u30ea\u30bb\u30c3\u30c8\u3068\u518d\u9078\u51fa\n    try {\n      await resetAndSelectDailyPickupUser();\n    } catch (pickupError) {\n      console.error(&#39;\u540c\u671f\u5f8c\u306e\u30a4\u30c1\u63a8\u3057braver\u9078\u51fa\u30a8\u30e9\u30fc:&#39;, pickupError);\n    }\n    \n    return res.json({\n      success: true,\n      syncedChannels: syncResults,\n      channelCount: syncResults.length,\n      period: `${getJSTDate().toLocaleDateString(&#39;ja-JP&#39;)} \u307e\u3067\u306e3\u30f6\u6708\u5206`,\n      message: &quot;\u30b9\u30bf\u30f3\u30d7\u60c5\u5831\u3082\u542b\u3081\u3066\u5168\u30c7\u30fc\u30bf\u3092\u540c\u671f\u3057\u307e\u3057\u305f&quot;\n    });\n  } catch (error) {\n    console.error(&#39;Slack\u30c7\u30fc\u30bf\u540c\u671f\u30a8\u30e9\u30fc:&#39;, error);\n    return res.status(500).json({ error: `\u540c\u671f\u51e6\u7406\u4e2d\u306b\u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u307e\u3057\u305f: ${error.message}` });\n  }\n});\n\n\/\/ Slack\u30e6\u30fc\u30b6\u30fc\u4e00\u89a7\u3092\u53d6\u5f97\u3059\u308bAPI\napp.get(&#39;\/api\/users&#39;, async (req, res) =&gt; {\n  try {\n    const query = `\n      SELECT \n        u.user_id, \n        u.name, \n        u.real_name, \n        u.display_name, \n        u.avatar,\n        u.is_bot,\n        (SELECT COUNT(*) FROM slack_messages WHERE user_id = u.user_id) AS message_count,\n        (SELECT COUNT(*) FROM slack_thread_replies WHERE user_id = u.user_id) AS reply_count,\n        (\n          SELECT COUNT(*) FROM slack_messages \n          WHERE user_id = u.user_id \n          AND (text LIKE &#39;\/yolo%&#39; OR text LIKE &#39;\/Yolo%&#39;)\n        ) AS attendance_count,\n        (\n          SELECT COUNT(*) FROM slack_messages \n          WHERE user_id = u.user_id \n          AND (text LIKE &#39;\/yolo_office%&#39; OR text LIKE &#39;\/Yolo_office%&#39;)\n        ) AS office_count,\n        (\n          SELECT COUNT(*) FROM slack_messages \n          WHERE user_id = u.user_id \n          AND length(text) &gt;= 50\n        ) AS weekly_report_count,\n        (\n          SELECT COUNT(*) FROM slack_reactions\n          WHERE user_id = u.user_id\n        ) AS sent_reaction_count,\n        (\n          SELECT COUNT(*) FROM slack_reactions r\n          JOIN slack_messages m ON r.message_id = m.message_id\n          WHERE m.user_id = u.user_id\n        ) AS received_reaction_count\n      FROM \n        slack_users u\n      WHERE \n        u.is_bot = 0\n      ORDER BY \n        message_count DESC\n    `;\n    \n    db.all(query, [], (err, users) =&gt; {\n      if (err) {\n        console.error(&#39;\u30e6\u30fc\u30b6\u30fc\u4e00\u89a7\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, err);\n        return res.status(500).json({ error: &#39;\u30e6\u30fc\u30b6\u30fc\u30c7\u30fc\u30bf\u306e\u53d6\u5f97\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n      }\n      \n      \/\/ \u30ea\u30a2\u30eb\u51fa\u793e\u7387\u3092\u8a08\u7b97\n      users.forEach(user =&gt; {\n        const officeCount = user.office_count || 0;\n        const attendanceCount = user.attendance_count || 0;\n        user.office_ratio = attendanceCount &gt; 0 ? Math.round((officeCount \/ attendanceCount) * 100) : 0;\n      });\n      \n      res.json(users);\n    });\n  } catch (error) {\n    console.error(&#39;\u30e6\u30fc\u30b6\u30fc\u4e00\u89a7\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, error);\n    res.status(500).json({ error: &#39;\u30e6\u30fc\u30b6\u30fc\u30c7\u30fc\u30bf\u306e\u53d6\u5f97\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n  }\n});\n\n\/\/ \u30e6\u30fc\u30b6\u30fc\u8a73\u7d30\u60c5\u5831\u3092\u53d6\u5f97\u3059\u308b\u4ee3\u66ffAPI\napp.get(&#39;\/api\/user-detail\/:userId&#39;, (req, res) =&gt; {\n  try {\n    const userId = req.params.userId;\n    console.log(`\u30e6\u30fc\u30b6\u30fc\u8a73\u7d30\u60c5\u5831\u30ea\u30af\u30a8\u30b9\u30c8: ${userId}`);\n    \n    \/\/ \u57fa\u672c\u7684\u306a\u30e6\u30fc\u30b6\u30fc\u60c5\u5831\u3092\u53d6\u5f97\u3059\u308b\u30b7\u30f3\u30d7\u30eb\u306a\u30af\u30a8\u30ea\n    const query = `\n      SELECT \n        user_id, \n        name, \n        real_name, \n        display_name, \n        avatar,\n        is_bot\n      FROM \n        slack_users\n      WHERE \n        user_id = ?\n    `;\n    \n    db.get(query, [userId], (err, user) =&gt; {\n      if (err) {\n        console.error(&#39;\u30e6\u30fc\u30b6\u30fc\u8a73\u7d30\u60c5\u5831\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, err);\n        return res.status(500).json({ error: &#39;\u30e6\u30fc\u30b6\u30fc\u8a73\u7d30\u60c5\u5831\u306e\u53d6\u5f97\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n      }\n      \n      if (!user) {\n        return res.status(404).json({ error: &#39;\u30e6\u30fc\u30b6\u30fc\u304c\u898b\u3064\u304b\u308a\u307e\u305b\u3093&#39; });\n      }\n      \n      \/\/ \u8ffd\u52a0\u60c5\u5831\u3092\u53d6\u5f97\n      const countQuery = `\n        SELECT \n          (SELECT COUNT(*) FROM slack_messages WHERE user_id = ?) AS message_count,\n          (SELECT COUNT(*) FROM slack_thread_replies WHERE user_id = ?) AS reply_count,\n          (SELECT COUNT(*) FROM slack_messages WHERE user_id = ? AND (text LIKE &#39;\/yolo%&#39; OR text LIKE &#39;\/Yolo%&#39;)) AS attendance_count,\n          (SELECT COUNT(*) FROM slack_messages WHERE user_id = ? AND (text LIKE &#39;\/yolo_office%&#39; OR text LIKE &#39;\/Yolo_office%&#39;)) AS office_count,\n          (SELECT COUNT(*) FROM slack_messages WHERE user_id = ? AND LENGTH(text) &gt;= 50) AS weekly_report_count,\n          (SELECT COUNT(*) FROM slack_reactions WHERE user_id = ?) AS sent_reaction_count,\n          (SELECT COUNT(*) FROM slack_reactions r JOIN slack_messages m ON r.message_id = m.message_id WHERE m.user_id = ?) AS received_reaction_count\n      `;\n      \n      db.get(countQuery, [userId, userId, userId, userId, userId, userId, userId], (countErr, counts) =&gt; {\n        if (countErr) {\n          console.error(&#39;\u30e6\u30fc\u30b6\u30fc\u30ab\u30a6\u30f3\u30c8\u60c5\u5831\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, countErr);\n          \/\/ \u30a8\u30e9\u30fc\u304c\u3042\u3063\u3066\u3082\u57fa\u672c\u60c5\u5831\u306f\u8fd4\u3059\n        } else if (counts) {\n          \/\/ \u30ab\u30a6\u30f3\u30c8\u60c5\u5831\u3092\u30e6\u30fc\u30b6\u30fc\u60c5\u5831\u306b\u8ffd\u52a0\n          Object.assign(user, counts);\n        }\n        \n        \/\/ \u9031\u5831\u30fb\u767a\u8a00\u30ea\u30b9\u30c8\u3092\u53d6\u5f97\uff0850\u6587\u5b57\u4ee5\u4e0a\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\uff09\n        const reportsQuery = `\n          SELECT \n            m.message_id,\n            m.text as text_preview,\n            m.timestamp as date,\n            c.name as channel_name,\n            m.channel_id\n          FROM \n            slack_messages m\n            LEFT JOIN slack_channels c ON m.channel_id = c.channel_id\n          WHERE \n            m.user_id = ? \n            AND LENGTH(m.text) &gt;= 50\n          ORDER BY \n            m.timestamp DESC\n          LIMIT 20\n        `;\n        \n        db.all(reportsQuery, [userId], (reportsErr, reports) =&gt; {\n          if (reportsErr) {\n            console.error(&#39;\u9031\u5831\u30fb\u767a\u8a00\u30ea\u30b9\u30c8\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, reportsErr);\n            user.reports = [];\n          } else {\n            \/\/ \u65e5\u4ed8\u3092\u30d5\u30a9\u30fc\u30de\u30c3\u30c8\n            reports.forEach(report =&gt; {\n              if (report.date) {\n                report.date = new Date(parseFloat(report.date) * 1000);\n              }\n              \/\/ \u30c6\u30ad\u30b9\u30c8\u3092\u5236\u9650\uff08\u9577\u3059\u304e\u308b\u5834\u5408\uff09\n              if (report.text_preview && report.text_preview.length &gt; 200) {\n                report.text_preview = report.text_preview.substring(0, 200) + &#39;...&#39;;\n              }\n            });\n            user.reports = reports || [];\n          }\n          \n          \/\/ \u8fd4\u4fe1\u30ea\u30b9\u30c8\u3092\u53d6\u5f97\n          const repliesQuery = `\n            SELECT \n              r.message_id,\n              r.text as text_preview,\n              r.timestamp as date,\n              r.parent_message_id,\n              c.name as channel_name,\n              r.channel_id,\n              (SELECT text FROM slack_messages WHERE message_id = r.parent_message_id) as parent_preview\n            FROM \n              slack_thread_replies r\n              LEFT JOIN slack_channels c ON r.channel_id = c.channel_id\n            WHERE \n              r.user_id = ?\n            ORDER BY \n              r.timestamp DESC\n            LIMIT 20\n          `;\n          \n          db.all(repliesQuery, [userId], (repliesErr, replies) =&gt; {\n            if (repliesErr) {\n              console.error(&#39;\u8fd4\u4fe1\u30ea\u30b9\u30c8\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, repliesErr);\n              user.replies = [];\n            } else {\n              \/\/ \u65e5\u4ed8\u3092\u30d5\u30a9\u30fc\u30de\u30c3\u30c8\n              replies.forEach(reply =&gt; {\n                if (reply.date) {\n                  reply.date = new Date(parseFloat(reply.date) * 1000);\n                }\n                \/\/ \u30c6\u30ad\u30b9\u30c8\u3092\u5236\u9650\uff08\u9577\u3059\u304e\u308b\u5834\u5408\uff09\n                if (reply.text_preview && reply.text_preview.length &gt; 200) {\n                  reply.text_preview = reply.text_preview.substring(0, 200) + &#39;...&#39;;\n                }\n                if (reply.parent_preview && reply.parent_preview.length &gt; 100) {\n                  reply.parent_preview = reply.parent_preview.substring(0, 100) + &#39;...&#39;;\n                }\n              });\n              user.replies = replies || [];\n            }\n            \n            \/\/ \u9001\u4fe1\u3057\u305f\u30b9\u30bf\u30f3\u30d7\u3092\u53d6\u5f97\n            const sentStampsQuery = `\n              SELECT \n                r.name as reaction,\n                r.timestamp as date,\n                m.text as message_preview,\n                r.message_id,\n                (SELECT u.display_name || COALESCE(&#39; (&#39; || u.real_name || &#39;)&#39;, &#39;&#39;) FROM slack_users u JOIN slack_messages msg ON u.user_id = msg.user_id WHERE msg.message_id = r.message_id) as target_user\n              FROM \n                slack_reactions r\n                LEFT JOIN slack_messages m ON r.message_id = m.message_id\n              WHERE \n                r.user_id = ?\n              ORDER BY \n                r.timestamp DESC\n              LIMIT 20\n            `;\n            \n            db.all(sentStampsQuery, [userId], (sentStampsErr, sentStamps) =&gt; {\n              if (sentStampsErr) {\n                console.error(&#39;\u9001\u4fe1\u30b9\u30bf\u30f3\u30d7\u30ea\u30b9\u30c8\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, sentStampsErr);\n                user.sent_stamps = [];\n              } else {\n                \/\/ \u65e5\u4ed8\u3092\u30d5\u30a9\u30fc\u30de\u30c3\u30c8\n                sentStamps.forEach(stamp =&gt; {\n                  if (stamp.date) {\n                    stamp.date = new Date(parseFloat(stamp.date) * 1000);\n                  }\n                  \/\/ \u30c6\u30ad\u30b9\u30c8\u3092\u5236\u9650\uff08\u9577\u3059\u304e\u308b\u5834\u5408\uff09\n                  if (stamp.message_preview && stamp.message_preview.length &gt; 150) {\n                    stamp.message_preview = stamp.message_preview.substring(0, 150) + &#39;...&#39;;\n                  }\n                });\n                user.sent_stamps = sentStamps || [];\n              }\n              \n              \/\/ \u53d7\u4fe1\u3057\u305f\u30b9\u30bf\u30f3\u30d7\u3092\u53d6\u5f97\n              const receivedStampsQuery = `\n                SELECT \n                  r.name as reaction,\n                  r.timestamp as date,\n                  m.text as message_preview,\n                  r.message_id,\n                  (SELECT u.display_name || COALESCE(&#39; (&#39; || u.real_name || &#39;)&#39;, &#39;&#39;) FROM slack_users u WHERE u.user_id = r.user_id) as from_user\n                FROM \n                  slack_reactions r\n                  JOIN slack_messages m ON r.message_id = m.message_id\n                WHERE \n                  m.user_id = ?\n                ORDER BY \n                  r.timestamp DESC\n                LIMIT 20\n              `;\n              \n              db.all(receivedStampsQuery, [userId], (receivedStampsErr, receivedStamps) =&gt; {\n                if (receivedStampsErr) {\n                  console.error(&#39;\u53d7\u4fe1\u30b9\u30bf\u30f3\u30d7\u30ea\u30b9\u30c8\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, receivedStampsErr);\n                  user.received_stamps = [];\n                } else {\n                  \/\/ \u65e5\u4ed8\u3092\u30d5\u30a9\u30fc\u30de\u30c3\u30c8\n                  receivedStamps.forEach(stamp =&gt; {\n                    if (stamp.date) {\n                      stamp.date = new Date(parseFloat(stamp.date) * 1000);\n                    }\n                    \/\/ \u30c6\u30ad\u30b9\u30c8\u3092\u5236\u9650\uff08\u9577\u3059\u304e\u308b\u5834\u5408\uff09\n                    if (stamp.message_preview && stamp.message_preview.length &gt; 150) {\n                      stamp.message_preview = stamp.message_preview.substring(0, 150) + &#39;...&#39;;\n                    }\n                  });\n                  user.received_stamps = receivedStamps || [];\n                }\n                \n                \/\/ AI\u5206\u6790\u7d50\u679c\u306f\u524a\u9664\n                user.analysis = &#39;&#39;;\n                \n                \/\/ \u6210\u529f\u30ec\u30b9\u30dd\u30f3\u30b9\n                res.json(user);\n              });\n            });\n          });\n        });\n      });\n    });\n  } catch (error) {\n    console.error(&#39;\u30e6\u30fc\u30b6\u30fc\u8a73\u7d30\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, error);\n    res.status(500).json({ error: &#39;\u30e6\u30fc\u30b6\u30fc\u8a73\u7d30\u60c5\u5831\u306e\u53d6\u5f97\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n  }\n});\n\n\/\/ \u7279\u5b9a\u30e6\u30fc\u30b6\u30fc\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u5206\u6790\u3059\u308bAPI\napp.get(&#39;\/api\/analyze-user\/:userId&#39;, async (req, res) =&gt; {\n  try {\n    const userId = req.params.userId;\n    const isPositive = req.query.positive === &#39;true&#39;;\n    console.log(`\u30e6\u30fc\u30b6\u30fc\u5206\u6790\u30ea\u30af\u30a8\u30b9\u30c8: ${userId}, \u30dd\u30b8\u30c6\u30a3\u30d6\u5206\u6790: ${isPositive}`);\n    \n    \/\/ \u30e6\u30fc\u30b6\u30fc\u540d\u3092\u53d6\u5f97\n    const userQuery = `SELECT name, real_name, display_name FROM slack_users WHERE user_id = ?`;\n    \n    db.get(userQuery, [userId], async (err, user) =&gt; {\n      if (err) {\n        console.error(&#39;\u30e6\u30fc\u30b6\u30fc\u60c5\u5831\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, err);\n        return res.status(500).json({ error: &#39;\u30e6\u30fc\u30b6\u30fc\u60c5\u5831\u306e\u53d6\u5f97\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n      }\n      \n      if (!user) {\n        return res.status(404).json({ error: &#39;\u30e6\u30fc\u30b6\u30fc\u304c\u898b\u3064\u304b\u308a\u307e\u305b\u3093&#39; });\n      }\n      \n      const userName = user.display_name || user.real_name || user.name || userId;\n      \n      \/\/ \u30e6\u30fc\u30b6\u30fc\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u53d6\u5f97\uff08\u6700\u5927100\u4ef6\uff09\n      const messageQuery = `\n        SELECT text, timestamp FROM slack_messages \n        WHERE user_id = ? \n        ORDER BY timestamp DESC \n        LIMIT 100\n      `;\n      \n      db.all(messageQuery, [userId], async (err, messages) =&gt; {\n        if (err) {\n          console.error(&#39;\u30e1\u30c3\u30bb\u30fc\u30b8\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, err);\n          return res.status(500).json({ error: &#39;\u30e1\u30c3\u30bb\u30fc\u30b8\u306e\u53d6\u5f97\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n        }\n        \n        \/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u304c\u5c11\u306a\u3059\u304e\u308b\u5834\u5408\n        if (messages.length &lt; 3) {\n          return res.json({\n            success: true,\n            source: &quot;simple&quot;,\n            analysis: {\n              userName: userName,\n              summary: `${userName}\u3055\u3093\u306e\u5206\u6790\u306b\u5fc5\u8981\u306a\u30e1\u30c3\u30bb\u30fc\u30b8\u304c\u4e0d\u8db3\u3057\u3066\u3044\u307e\u3059\u3002\u3082\u3046\u5c11\u3057Slack\u3067\u306e\u4f1a\u8a71\u30c7\u30fc\u30bf\u304c\u5fc5\u8981\u3067\u3059\u3002`,\n              communicationStyle: &quot;\u30c7\u30fc\u30bf\u4e0d\u8db3&quot;,\n              strengths: [&quot;\u30c7\u30fc\u30bf\u304c\u4e0d\u8db3\u3057\u3066\u3044\u307e\u3059&quot;],\n              interests: [&quot;\u30c7\u30fc\u30bf\u304c\u4e0d\u8db3\u3057\u3066\u3044\u307e\u3059&quot;],\n              workStyle: &quot;\u30c7\u30fc\u30bf\u304c\u4e0d\u8db3\u3057\u3066\u3044\u307e\u3059&quot;,\n              tags: [&quot;\u30c7\u30fc\u30bf\u4e0d\u8db3&quot;]\n            }\n          });\n        }\n        \n        try {\n          \/\/ OpenAI API\u3092\u4f7f\u3063\u3066\u5206\u6790\n          const messagesText = messages.map(m =&gt; m.text).join(&#39;\\n\\n&#39;);\n          \n          const prompt = currentAiAnalysisPrompt + `\n\n\u30e6\u30fc\u30b6\u30fc\u540d: ${userName}\n\u30e1\u30c3\u30bb\u30fc\u30b8\u6570: ${messages.length}\u4ef6\n\n\u3010\u30e1\u30c3\u30bb\u30fc\u30b8\u4e00\u89a7\u3011\n${messagesText}\n\n\u5206\u6790\u5bfe\u8c61\u306e\u30e6\u30fc\u30b6\u30fc\u540d\u306f\u300c${userName}\u300d\u3067\u3059\u3002\u3053\u306e\u30e6\u30fc\u30b6\u30fc\u306e\u7279\u5fb4\u3092600\u6587\u5b57\u7a0b\u5ea6\u3067\u5177\u4f53\u7684\u306b\u5206\u6790\u3057\u3066\u304f\u3060\u3055\u3044\u3002`;\n          \n          \/\/ OpenAI API\u3092\u547c\u3073\u51fa\u3057\n          const openaiResponse = await fetch(&#39;https:\/\/api.openai.com\/v1\/chat\/completions&#39;, {\n            method: &#39;POST&#39;,\n            headers: {\n              &#39;Content-Type&#39;: &#39;application\/json&#39;,\n              &#39;Authorization&#39;: `Bearer ${process.env.OPENAI_API_KEY}`\n            },\n            body: JSON.stringify({\n              model: &quot;gpt-3.5-turbo&quot;,\n              messages: [\n                {\n                  role: &quot;system&quot;,\n                  content: currentAiAnalysisPrompt\n                },\n                {\n                  role: &quot;user&quot;,\n                  content: prompt\n                }\n              ],\n              temperature: 0.6,\n              max_tokens: 500\n            })\n          });\n          \n          if (!openaiResponse.ok) {\n            console.error(&#39;OpenAI API\u30a8\u30e9\u30fc:&#39;, await openaiResponse.text());\n            throw new Error(&#39;OpenAI API\u304b\u3089\u306e\u5fdc\u7b54\u306b\u554f\u984c\u304c\u3042\u308a\u307e\u3057\u305f&#39;);\n          }\n          \n          const openaiData = await openaiResponse.json();\n          const analysisText = openaiData.choices[0].message.content.trim();\n          \n          \/\/ \u30af\u30ea\u30fc\u30f3\u306a\u5206\u6790\u7d50\u679c\u3092\u8fd4\u3059\n          const sanitizedAnalysis = sanitizeApiResponse(analysisText);\n          \n          \/\/ \u5206\u6790\u7d50\u679c\u3092JSON\u5f62\u5f0f\u3067\u8fd4\u3059\n          return res.json({\n            success: true,\n            source: &quot;openai&quot;,\n            analysis: {\n              userName: userName,\n              summary: sanitizedAnalysis.length &gt; 400 ? sanitizedAnalysis.substring(0, 400) + &quot;...&quot; : sanitizedAnalysis,\n              communicationStyle: &quot;\u5206\u6790\u5b8c\u4e86&quot;,\n              strengths: [&quot;OpenAI\u306b\u3088\u308b\u5206\u6790\u3092\u8868\u793a\u4e2d&quot;],\n              interests: [&quot;OpenAI\u306b\u3088\u308b\u5206\u6790\u3092\u8868\u793a\u4e2d&quot;],\n              workStyle: &quot;OpenAI\u306b\u3088\u308b\u5206\u6790\u3092\u8868\u793a\u4e2d&quot;,\n              tags: [&quot;AI\u5206\u6790&quot;, &quot;\u30dd\u30b8\u30c6\u30a3\u30d6\u8a55\u4fa1&quot;]\n            }\n          });\n          \n        } catch (error) {\n          console.error(&#39;OpenAI API\u547c\u3073\u51fa\u3057\u30a8\u30e9\u30fc:&#39;, error);\n          \n          \/\/ \u30a8\u30e9\u30fc\u6642\u306f\u7c21\u6613\u5206\u6790\u3092\u8fd4\u3059\n          return res.json({\n            success: true,\n            source: &quot;error_fallback&quot;,\n            error: error.message,\n            analysis: {\n              userName: userName,\n              summary: `${userName}\u3055\u3093\u306e\u5206\u6790\u4e2d\u306b\u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u307e\u3057\u305f\u3002\u6642\u9593\u3092\u304a\u3044\u3066\u518d\u5ea6\u304a\u8a66\u3057\u304f\u3060\u3055\u3044\u3002\u30a8\u30e9\u30fc: ${error.message}`,\n              communicationStyle: &quot;\u30a8\u30e9\u30fc\u767a\u751f&quot;,\n              strengths: [&quot;\u30c7\u30fc\u30bf\u5206\u6790\u30a8\u30e9\u30fc&quot;],\n              interests: [&quot;\u30c7\u30fc\u30bf\u5206\u6790\u30a8\u30e9\u30fc&quot;],\n              workStyle: &quot;\u30c7\u30fc\u30bf\u5206\u6790\u30a8\u30e9\u30fc&quot;,\n              tags: [&quot;\u30a8\u30e9\u30fc&quot;]\n            }\n          });\n        }\n      });\n    });\n    \n  } catch (error) {\n    console.error(&#39;\u30e6\u30fc\u30b6\u30fc\u5206\u6790\u4e2d\u306b\u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u307e\u3057\u305f:&#39;, error);\n    res.status(500).json({ error: error.message });\n  }\n});\n\n\/\/ \u30b0\u30e9\u30d5\u8868\u793a\u7528\u306eTOP30\u30e6\u30fc\u30b6\u30fc\u3092\u53d6\u5f97\u3059\u308bAPI\napp.get(&#39;\/api\/stats\/top-users&#39;, async (req, res) =&gt; {\n  try {\n    const type = req.query.type || req.query.chart || &#39;messages&#39;;\n    const period = req.query.period || &#39;3months&#39;; \/\/ \u65b0\u3057\u3044\u30d1\u30e9\u30e1\u30fc\u30bf: \u671f\u9593\uff083months, 2weeks, 3days\uff09\n    \n    let column = &#39;message_count&#39;;\n    let title = &#39;\u30e1\u30c3\u30bb\u30fc\u30b8\u6295\u7a3f\u6570&#39;;\n    \n    \/\/ \u671f\u9593\u306b\u5fdc\u3058\u305f\u65e5\u4ed8\u6761\u4ef6\u3092\u8a2d\u5b9a (\u65e5\u672c\u6642\u9593\u30d9\u30fc\u30b9)\n    let dateCondition = &#39;&#39;;\n    const now = getJSTDate(); \/\/ \u65e5\u672c\u6642\u9593\u306enow\u3092\u4f7f\u7528\n    \n    if (period === &#39;3months&#39;) {\n      \/\/ \u904e\u53bb3\u30f6\u6708\n      const threeMonthsAgo = new Date(now);\n      threeMonthsAgo.setMonth(now.getMonth() - 3);\n      dateCondition = `AND timestamp &gt; &#39;${threeMonthsAgo.getTime() \/ 1000}&#39;`;\n      title += &#39;\uff08\u76f4\u8fd13\u30f6\u6708\uff09&#39;;\n    } else if (period === &#39;2weeks&#39;) {\n      \/\/ \u904e\u53bb1\u30f6\u6708\uff08\u671f\u9593\u540d\u306f\u5909\u66f4\u3057\u3066\u3044\u306a\u3044\u304c\u5185\u5bb9\u3092\u5909\u66f4\uff09\n      const oneMonthAgo = new Date(now);\n      oneMonthAgo.setMonth(now.getMonth() - 1);\n      dateCondition = `AND timestamp &gt; &#39;${oneMonthAgo.getTime() \/ 1000}&#39;`;\n      title += &#39;\uff08\u76f4\u8fd11\u30f6\u6708\uff09&#39;;\n    } else if (period === &#39;3days&#39;) {\n      \/\/ \u904e\u53bb1\u9031\u9593\uff08\u671f\u9593\u540d\u306f\u5909\u66f4\u3057\u3066\u3044\u306a\u3044\u304c\u5185\u5bb9\u3092\u5909\u66f4\uff09\n      const oneWeekAgo = new Date(now);\n      oneWeekAgo.setDate(now.getDate() - 7);\n      dateCondition = `AND timestamp &gt; &#39;${oneWeekAgo.getTime() \/ 1000}&#39;`;\n      title += &#39;\uff08\u76f4\u8fd11\u9031\u9593\uff09&#39;;\n    }\n    \n    \/\/ \u30bf\u30a4\u30d7\u306b\u5fdc\u3058\u3066\u53d6\u5f97\u3059\u308b\u30ab\u30e9\u30e0\u3092\u5909\u66f4\n    switch (type) {\n      case &#39;messages&#39;:\n        column = `(\n          SELECT COUNT(*) FROM slack_messages \n          WHERE user_id = u.user_id\n          ${dateCondition}\n        )`;\n        title = &#39;\u30e1\u30c3\u30bb\u30fc\u30b8\u6295\u7a3f\u6570&#39; + (title.includes(&#39;\uff08&#39;) ? title.substring(title.indexOf(&#39;\uff08&#39;)) : &#39;&#39;);\n        break;\n      case &#39;replies&#39;:\n        column = `(\n          SELECT COUNT(*) FROM slack_thread_replies\n          WHERE user_id = u.user_id\n          ${dateCondition}\n        )`;\n        title = &#39;\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u6570&#39; + (title.includes(&#39;\uff08&#39;) ? title.substring(title.indexOf(&#39;\uff08&#39;)) : &#39;&#39;);\n        break;\n      case &#39;received_reactions&#39;:\n        column = `(\n          SELECT COUNT(*) FROM slack_reactions r\n          JOIN slack_messages m ON r.message_id = m.message_id\n          WHERE m.user_id = u.user_id\n          ${dateCondition.replace(&#39;timestamp&#39;, &#39;m.timestamp&#39;)}\n        )`;\n        title = &#39;\u30b9\u30bf\u30f3\u30d7\u3092\u8cb0\u3063\u305f\u6570&#39; + (title.includes(&#39;\uff08&#39;) ? title.substring(title.indexOf(&#39;\uff08&#39;)) : &#39;&#39;);\n        break;\n      case &#39;sent_reactions&#39;:\n        column = `(\n          SELECT COUNT(*) FROM slack_reactions r\n          WHERE r.user_id = u.user_id\n          ${dateCondition.replace(&#39;timestamp&#39;, &#39;r.timestamp&#39;)}\n        )`;\n        title = &#39;\u30b9\u30bf\u30f3\u30d7\u3092\u9001\u3063\u305f\u6570&#39; + (title.includes(&#39;\uff08&#39;) ? title.substring(title.indexOf(&#39;\uff08&#39;)) : &#39;&#39;);\n        break;\n      case &#39;attendance&#39;:\n        column = `(\n          SELECT COUNT(*) FROM slack_messages \n          WHERE user_id = u.user_id \n          AND (\n            text LIKE &#39;\/yolo%&#39; OR\n            text LIKE &#39;\/Yolo%&#39;\n          )\n          ${dateCondition}\n        )`;\n        title = &#39;\u51fa\u52e4\u6570&#39; + (title.includes(&#39;\uff08&#39;) ? title.substring(title.indexOf(&#39;\uff08&#39;)) : &#39;&#39;);\n        break;\n      case &#39;office&#39;:\n        column = `(\n          SELECT COUNT(*) FROM slack_messages \n          WHERE user_id = u.user_id \n          AND (\n            text LIKE &#39;\/yolo_office%&#39; OR \n            text LIKE &#39;\/Yolo_office%&#39;\n          )\n          ${dateCondition}\n        )`;\n        title = &#39;\u30ea\u30a2\u30eb\u51fa\u793e\u6570&#39; + (title.includes(&#39;\uff08&#39;) ? title.substring(title.indexOf(&#39;\uff08&#39;)) : &#39;&#39;);\n        break;\n      case &#39;reports&#39;:\n        column = `(\n          SELECT COUNT(*) FROM slack_messages \n          WHERE user_id = u.user_id \n          AND length(text) &gt;= 50\n          ${dateCondition}\n        )`;\n        title = &#39;\u767a\u4fe1\u6570&#39; + (title.includes(&#39;\uff08&#39;) ? title.substring(title.indexOf(&#39;\uff08&#39;)) : &#39;&#39;);\n        break;\n      default:\n        column = `(SELECT COUNT(*) FROM slack_messages WHERE user_id = u.user_id ${dateCondition})`;\n        title = &#39;\u30e1\u30c3\u30bb\u30fc\u30b8\u6295\u7a3f\u6570&#39; + (title.includes(&#39;\uff08&#39;) ? title.substring(title.indexOf(&#39;\uff08&#39;)) : &#39;&#39;);\n    }\n    \n    \/\/ SQL\u5b9f\u884c\n    const query = `\n      SELECT \n        u.user_id, \n        u.name, \n        u.real_name, \n        u.display_name,\n        u.avatar,\n        ${column}  count_value\n      FROM \n        slack_users u\n      WHERE \n        u.is_bot = 0\n        AND ${column} &gt; 0\n      ORDER BY \n        count_value DESC\n      LIMIT 30\n    `;\n    \n    db.all(query, [], (err, users) =&gt; {\n      if (err) {\n        console.error(&#39;\u30b0\u30e9\u30d5\u30c7\u30fc\u30bf\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, err);\n        return res.status(500).json({ error: &#39;\u30b0\u30e9\u30d5\u30c7\u30fc\u30bf\u306e\u53d6\u5f97\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39;, success: false });\n      }\n      \n      \/\/ \u8868\u793a\u540d\u3092\u8a2d\u5b9a\n      users.forEach(user =&gt; {\n        user.displayName = user.display_name || user.real_name || user.name || &#39;Unknown&#39;;\n      });\n      \n      res.json({\n        success: true,\n        title: title,\n        data: users,\n        users: users  \/\/ client\u304c users \u30d5\u30a3\u30fc\u30eb\u30c9\u3092\u671f\u5f85\u3057\u3066\u3044\u308b\u5834\u5408\u306b\u5bfe\u5fdc\n      });\n    });\n  } catch (error) {\n    console.error(&#39;\u30b0\u30e9\u30d5\u30c7\u30fc\u30bf\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, error);\n    res.status(500).json({ error: &#39;\u30b0\u30e9\u30d5\u30c7\u30fc\u30bf\u306e\u53d6\u5f97\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39;, success: false });\n  }\n});\n\n\/\/ \u30b5\u30fc\u30d0\u30fc\u8d77\u52d5\napp.listen(port, () =&gt; {\n  console.log(`\u30b5\u30fc\u30d0\u30fc\u304c\u8d77\u52d5\u3057\u307e\u3057\u305f: http:\/\/localhost:${port}`);\n  console.log(`\u30c6\u30b9\u30c8\u30da\u30fc\u30b8: http:\/\/localhost:${port}\/simple`);\n  \n  \/\/ \u30b5\u30fc\u30d0\u30fc\u8d77\u52d5\u6642\u306b\u3082\u81ea\u52d5\u540c\u671f\u30c1\u30a7\u30c3\u30af\u3092\u5b9f\u884c\n  setTimeout(async () =&gt; {\n    console.log(&#39;\u30b5\u30fc\u30d0\u30fc\u8d77\u52d5\u5f8c\u306e\u81ea\u52d5\u540c\u671f\u30c1\u30a7\u30c3\u30af\u3092\u5b9f\u884c\u3057\u307e\u3059...&#39;);\n    try {\n      await checkAndSyncIfNeeded();\n      console.log(&#39;\u30b5\u30fc\u30d0\u30fc\u8d77\u52d5\u5f8c\u306e\u81ea\u52d5\u540c\u671f\u30c1\u30a7\u30c3\u30af\u5b8c\u4e86&#39;);\n    } catch (error) {\n      console.error(&#39;\u30b5\u30fc\u30d0\u30fc\u8d77\u52d5\u5f8c\u306e\u81ea\u52d5\u540c\u671f\u30c1\u30a7\u30c3\u30af\u3067\u30a8\u30e9\u30fc\u304c\u767a\u751f:&#39;, error);\n    }\n  }, 3000); \/\/ \u30b5\u30fc\u30d0\u30fc\u8d77\u52d5\u304b\u30893\u79d2\u5f8c\u306b\u5b9f\u884c\n});\n\n\/\/ \u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u3092\u5168\u524a\u9664\u3059\u308bAPI\napp.post(&#39;\/api\/reset-database&#39;, (req, res) =&gt; {\n  try {\n    console.log(&#39;\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u306e\u5168\u524a\u9664\u3092\u958b\u59cb\u3057\u307e\u3059...&#39;);\n    \n    \/\/ \u30c8\u30e9\u30f3\u30b6\u30af\u30b7\u30e7\u30f3\u3092\u958b\u59cb\n    db.serialize(() =&gt; {\n      \/\/ \u5404\u30c6\u30fc\u30d6\u30eb\u3092\u7a7a\u306b\u3059\u308b\n      db.run(&#39;DELETE FROM slack_messages&#39;, function(err) {\n        if (err) {\n          console.error(&#39;slack_messages\u30c6\u30fc\u30d6\u30eb\u524a\u9664\u30a8\u30e9\u30fc:&#39;, err);\n        } else {\n          console.log(`slack_messages\u304b\u3089${this.changes}\u4ef6\u306e\u30c7\u30fc\u30bf\u3092\u524a\u9664\u3057\u307e\u3057\u305f`);\n        }\n      });\n      \n      db.run(&#39;DELETE FROM slack_thread_replies&#39;, function(err) {\n        if (err) {\n          console.error(&#39;slack_thread_replies\u30c6\u30fc\u30d6\u30eb\u524a\u9664\u30a8\u30e9\u30fc:&#39;, err);\n        } else {\n          console.log(`slack_thread_replies\u304b\u3089${this.changes}\u4ef6\u306e\u30c7\u30fc\u30bf\u3092\u524a\u9664\u3057\u307e\u3057\u305f`);\n        }\n      });\n      \n      db.run(&#39;DELETE FROM slack_reactions&#39;, function(err) {\n        if (err) {\n          console.error(&#39;slack_reactions\u30c6\u30fc\u30d6\u30eb\u524a\u9664\u30a8\u30e9\u30fc:&#39;, err);\n        } else {\n          console.log(`slack_reactions\u304b\u3089${this.changes}\u4ef6\u306e\u30c7\u30fc\u30bf\u3092\u524a\u9664\u3057\u307e\u3057\u305f`);\n        }\n      });\n      \n      db.run(&#39;DELETE FROM slack_users&#39;, function(err) {\n        if (err) {\n          console.error(&#39;slack_users\u30c6\u30fc\u30d6\u30eb\u524a\u9664\u30a8\u30e9\u30fc:&#39;, err);\n        } else {\n          console.log(`slack_users\u304b\u3089${this.changes}\u4ef6\u306e\u30c7\u30fc\u30bf\u3092\u524a\u9664\u3057\u307e\u3057\u305f`);\n        }\n      });\n      \n      db.run(&#39;DELETE FROM slack_channels&#39;, function(err) {\n        if (err) {\n          console.error(&#39;slack_channels\u30c6\u30fc\u30d6\u30eb\u524a\u9664\u30a8\u30e9\u30fc:&#39;, err);\n        } else {\n          console.log(`slack_channels\u304b\u3089${this.changes}\u4ef6\u306e\u30c7\u30fc\u30bf\u3092\u524a\u9664\u3057\u307e\u3057\u305f`);\n        }\n      });\n      \n      db.run(&#39;DELETE FROM user_analysis&#39;, function(err) {\n        if (err) {\n          console.error(&#39;user_analysis\u30c6\u30fc\u30d6\u30eb\u524a\u9664\u30a8\u30e9\u30fc:&#39;, err);\n        } else {\n          console.log(`user_analysis\u304b\u3089${this.changes}\u4ef6\u306e\u30c7\u30fc\u30bf\u3092\u524a\u9664\u3057\u307e\u3057\u305f`);\n        }\n      });\n    });\n    \n    console.log(&#39;\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u306e\u5168\u524a\u9664\u304c\u5b8c\u4e86\u3057\u307e\u3057\u305f&#39;);\n    res.json({ success: true, message: &#39;\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u3092\u5168\u524a\u9664\u3057\u307e\u3057\u305f&#39; });\n  } catch (error) {\n    console.error(&#39;\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u524a\u9664\u30a8\u30e9\u30fc:&#39;, error);\n    res.status(500).json({ error: &#39;\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u306e\u524a\u9664\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n  }\n});\n\n\/\/ \u6700\u7d42\u540c\u671f\u65e5\u6642\u3092\u7279\u5b9a\u306e\u65e5\u4ed8\u306b\u8a2d\u5b9a\u3059\u308bAPI\uff08\u30c6\u30b9\u30c8\u7528\uff09\napp.post(&#39;\/api\/set-last-sync-date&#39;, express.json(), (req, res) =&gt; {\n  try {\n    \/\/ \u30ea\u30af\u30a8\u30b9\u30c8\u304b\u3089\u65e5\u6642\u60c5\u5831\u3092\u53d6\u5f97\n    const { date } = req.body;\n    \n    if (!date) {\n      \/\/ \u30c7\u30d5\u30a9\u30eb\u30c8\u30672024\u5e745\u67089\u65e514:00\u306b\u8a2d\u5b9a\n      const defaultDate = new Date(&#39;2024-05-09T14:00:00+09:00&#39;);\n      const defaultTimestamp = Math.floor(defaultDate.getTime() \/ 1000);\n      \n      \/\/ \u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u66f4\u65b0\n      db.run(&#39;DELETE FROM last_sync_date&#39;, function(deleteErr) {\n        if (deleteErr) {\n          console.error(&#39;\u6700\u7d42\u540c\u671f\u65e5\u6642\u306e\u524a\u9664\u30a8\u30e9\u30fc:&#39;, deleteErr);\n          return res.status(500).json({ error: &#39;\u6700\u7d42\u540c\u671f\u65e5\u6642\u306e\u8a2d\u5b9a\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n        }\n        \n        const dateString = defaultDate.toISOString().split(&#39;T&#39;)[0];\n        db.run(\n          &#39;INSERT INTO last_sync_date (last_sync_date, last_sync_timestamp) VALUES (?, ?)&#39;,\n          [dateString, defaultTimestamp],\n          function(insertErr) {\n            if (insertErr) {\n              console.error(&#39;\u6700\u7d42\u540c\u671f\u65e5\u6642\u306e\u633f\u5165\u30a8\u30e9\u30fc:&#39;, insertErr);\n              return res.status(500).json({ error: &#39;\u6700\u7d42\u540c\u671f\u65e5\u6642\u306e\u8a2d\u5b9a\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n            }\n            \n            console.log(`\u6700\u7d42\u540c\u671f\u65e5\u6642\u3092\u8a2d\u5b9a\u3057\u307e\u3057\u305f: ${dateString} (${defaultTimestamp})`);\n            return res.json({\n              success: true,\n              message: `\u6700\u7d42\u540c\u671f\u65e5\u6642\u3092 ${dateString} \u306b\u8a2d\u5b9a\u3057\u307e\u3057\u305f`,\n              date: dateString,\n              timestamp: defaultTimestamp\n            });\n          }\n        );\n      });\n    } else {\n      \/\/ \u6307\u5b9a\u3055\u308c\u305f\u65e5\u6642\u3092\u4f7f\u7528\n      const specifiedDate = new Date(date);\n      const timestamp = Math.floor(specifiedDate.getTime() \/ 1000);\n      \n      \/\/ \u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u66f4\u65b0\n      db.run(&#39;DELETE FROM last_sync_date&#39;, function(deleteErr) {\n        if (deleteErr) {\n          console.error(&#39;\u6700\u7d42\u540c\u671f\u65e5\u6642\u306e\u524a\u9664\u30a8\u30e9\u30fc:&#39;, deleteErr);\n          return res.status(500).json({ error: &#39;\u6700\u7d42\u540c\u671f\u65e5\u6642\u306e\u8a2d\u5b9a\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n        }\n        \n        const dateString = specifiedDate.toISOString().split(&#39;T&#39;)[0];\n        db.run(\n          &#39;INSERT INTO last_sync_date (last_sync_date, last_sync_timestamp) VALUES (?, ?)&#39;,\n          [dateString, timestamp],\n          function(insertErr) {\n            if (insertErr) {\n              console.error(&#39;\u6700\u7d42\u540c\u671f\u65e5\u6642\u306e\u633f\u5165\u30a8\u30e9\u30fc:&#39;, insertErr);\n              return res.status(500).json({ error: &#39;\u6700\u7d42\u540c\u671f\u65e5\u6642\u306e\u8a2d\u5b9a\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n            }\n            \n            console.log(`\u6700\u7d42\u540c\u671f\u65e5\u6642\u3092\u8a2d\u5b9a\u3057\u307e\u3057\u305f: ${dateString} (${timestamp})`);\n            return res.json({\n              success: true,\n              message: `\u6700\u7d42\u540c\u671f\u65e5\u6642\u3092 ${dateString} \u306b\u8a2d\u5b9a\u3057\u307e\u3057\u305f`,\n              date: dateString,\n              timestamp: timestamp\n            });\n          }\n        );\n      });\n    }\n  } catch (error) {\n    console.error(&#39;\u6700\u7d42\u540c\u671f\u65e5\u6642\u8a2d\u5b9a\u30a8\u30e9\u30fc:&#39;, error);\n    res.status(500).json({ error: &#39;\u6700\u7d42\u540c\u671f\u65e5\u6642\u306e\u8a2d\u5b9a\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n  }\n});\n\n\/\/ \u30b0\u30ed\u30fc\u30d0\u30eb\u95a2\u6570\u3068\u3057\u3066\u8ffd\u52a0\uff1aJSON\u89e3\u6790\u30a8\u30e9\u30fc\u8a3a\u65ad\u3068\u4fee\u6b63\nfunction fixJsonParseError(jsonString, errorMessage) {\n  console.log(&#39;JSON\u89e3\u6790\u30a8\u30e9\u30fc\u4fee\u6b63\u3092\u8a66\u307f\u307e\u3059...&#39;);\n  \n  \/\/ jsonString\u304cundefined\u3084null\u3067\u306a\u3044\u3053\u3068\u3092\u78ba\u8a8d\n  if (!jsonString) {\n    console.error(&#39;\u4fee\u6b63\u5bfe\u8c61\u306ejsonString\u304c\u672a\u5b9a\u7fa9\u3067\u3059&#39;);\n    return &#39;&#39;;\n  }\n  \n  console.log(`\u5143\u306e\u30a8\u30e9\u30fc: ${errorMessage}`);\n  console.log(`JSON\u6587\u5b57\u5217\u9577: ${jsonString.length}\u6587\u5b57`);\n  \n  \/\/ \u30a8\u30e9\u30fc\u30e1\u30c3\u30bb\u30fc\u30b8\u306e\u8a73\u7d30\u3092\u3055\u3089\u306b\u8868\u793a\n  if (errorMessage.includes(&#39;Unexpected non-whitespace character after JSON at position 67&#39;)) {\n    console.log(&#39;===== \u4f4d\u7f6e67\u30a8\u30e9\u30fc\u306e\u8a73\u7d30\u8a3a\u65ad =====&#39;);\n    console.log(`\u30a8\u30e9\u30fc\u4f4d\u7f6e\u5468\u8fba\uff0855-80\uff09\u306e\u6587\u5b57\u5217: &quot;${jsonString.substring(55, 80)}&quot;`);\n    \n    \/\/ \u5404\u6587\u5b57\u306e\u30b3\u30fc\u30c9\u30dd\u30a4\u30f3\u30c8\u3092\u8868\u793a\n    console.log(&#39;\u6587\u5b57\u3054\u3068\u306e\u30b3\u30fc\u30c9\u30dd\u30a4\u30f3\u30c8:&#39;);\n    for (let i = 60; i &lt; 75; i++) {\n      if (i &lt; jsonString.length) {\n        const char = jsonString.charAt(i);\n        console.log(`\u4f4d\u7f6e${i}: &quot;${char}&quot; (\u30b3\u30fc\u30c9: ${char.charCodeAt(0)}, 16\u9032\u6570: 0x${char.charCodeAt(0).toString(16)})`);\n      }\n    }\n    \n    \/\/ 16\u9032\u6570\u3067\u30c0\u30f3\u30d7\n    console.log(&#39;16\u9032\u6570\u30c0\u30f3\u30d7 (\u4f4d\u7f6e60-75):&#39;);\n    let hexDump = &#39;&#39;;\n    for (let i = 60; i &lt; 75; i++) {\n      if (i &lt; jsonString.length) {\n        hexDump += jsonString.charCodeAt(i).toString(16).padStart(2, &#39;0&#39;) + &#39; &#39;;\n      }\n    }\n    console.log(hexDump);\n  }\n  \n  \/\/ \u7279\u5b9a\u306e\u30a8\u30e9\u30fc\u30e1\u30c3\u30bb\u30fc\u30b8\u306b\u5bfe\u3059\u308b\u7279\u6b8a\u51e6\u7406\n  if (errorMessage.includes(&#39;Unexpected non-whitespace character after JSON at position 67&#39;) ||\n      (errorMessage.includes(&#39;Unexpected token&#39;) && errorMessage.includes(&#39;position 67&#39;))) {\n    console.log(&#39;\u7279\u5b9a\u306e\u4f4d\u7f6e67\u30a8\u30e9\u30fc\u3092\u691c\u51fa\u3057\u307e\u3057\u305f - \u5c02\u7528\u306e\u4fee\u6b63\u95a2\u6570\u3092\u547c\u3073\u51fa\u3057\u307e\u3059&#39;);\n    const fixedString = fixPosition67Error(jsonString);\n    \n    \/\/ \u4fee\u6b63\u524d\u5f8c\u306e\u6bd4\u8f03\u30ed\u30b0\n    if (fixedString !== jsonString) {\n      console.log(&#39;\u4fee\u6b63\u304c\u9069\u7528\u3055\u308c\u307e\u3057\u305f:&#39;);\n      console.log(`\u4fee\u6b63\u524d(\u4f4d\u7f6e60-75): &quot;${jsonString.substring(60, 75)}&quot;`);\n      console.log(`\u4fee\u6b63\u5f8c(\u4f4d\u7f6e60-75): &quot;${fixedString.substring(60, 75)}&quot;`);\n      \n      \/\/ \u4fee\u6b63\u304c\u6b63\u5e38\u306b\u30d1\u30fc\u30b9\u3067\u304d\u308b\u304b\u78ba\u8a8d\n      try {\n        JSON.parse(fixedString);\n        console.log(&#39;\u4fee\u6b63\u5f8c\u306eJSON\u306f\u6b63\u5e38\u306b\u30d1\u30fc\u30b9\u3067\u304d\u307e\u3057\u305f\uff01&#39;);\n      } catch (e) {\n        console.log(`\u4fee\u6b63\u5f8c\u3082\u30d1\u30fc\u30b9\u3067\u304d\u307e\u305b\u3093: ${e.message}`);\n      }\n    } else {\n      console.log(&#39;\u5c02\u7528\u95a2\u6570\u306b\u3088\u308b\u4fee\u6b63\u306f\u9069\u7528\u3055\u308c\u307e\u305b\u3093\u3067\u3057\u305f&#39;);\n    }\n    \n    return fixedString;\n  }\n  \n  \/\/ \u5171\u901a\u306e\u554f\u984c\uff1a\u4f4d\u7f6e\u3092\u7279\u5b9a\n  const positionMatch = errorMessage.match(\/position\\s+(\\d+)\/i);\n  if (!positionMatch) {\n    console.log(&#39;\u30a8\u30e9\u30fc\u4f4d\u7f6e\u3092\u7279\u5b9a\u3067\u304d\u307e\u305b\u3093\u3067\u3057\u305f&#39;);\n    return jsonString; \/\/ \u4fee\u6b63\u3067\u304d\u306a\u3044\u5834\u5408\u306f\u5143\u306e\u6587\u5b57\u5217\u3092\u8fd4\u3059\n  }\n  \n  const errorPos = parseInt(positionMatch[1]);\n  console.log(`\u30a8\u30e9\u30fc\u4f4d\u7f6e: ${errorPos}`);\n  \n  \/\/ \u30a8\u30e9\u30fc\u5468\u8fba\u306e\u30b3\u30f3\u30c6\u30ad\u30b9\u30c8\u3092\u8868\u793a\n  const contextStart = Math.max(0, errorPos - 15);\n  const contextEnd = Math.min(jsonString.length, errorPos + 15);\n  console.log(`\u30a8\u30e9\u30fc\u5468\u8fba: &quot;${jsonString.substring(contextStart, contextEnd)}&quot;`);\n  \n  \/\/ \u30a8\u30e9\u30fc\u4f4d\u7f6e\u306e\u6587\u5b57\u3068\u305d\u306e\u524d\u5f8c\n  const errorChar = jsonString.charAt(errorPos);\n  const beforeChar = errorPos &gt; 0 ? jsonString.charAt(errorPos - 1) : &#39;&#39;;\n  const afterChar = errorPos &lt; jsonString.length - 1 ? jsonString.charAt(errorPos + 1) : &#39;&#39;;\n  \n  console.log(`\u554f\u984c\u306e\u6587\u5b57: &quot;${beforeChar}[${errorChar}]${afterChar}&quot; (\u30b3\u30fc\u30c9: ${errorChar.charCodeAt(0)})`);\n  \n  \/\/ \u7279\u5b9a\u306e\u30b1\u30fc\u30b9\uff1a\u4f4d\u7f6e67\u554f\u984c\uff08\u975e\u5e38\u306b\u4e00\u822c\u7684\u306a\u30a8\u30e9\u30fc\uff09\n  if (errorPos &gt;= 66 && errorPos &lt;= 68) {\n    console.log(&#39;\u4f4d\u7f6e67\u5468\u8fba\u306e\u4e00\u822c\u7684\u306a\u554f\u984c\u3092\u691c\u51fa&#39;);\n    \n    \/\/ \u4f4d\u7f6e67\u306e\u6587\u5b57\u3092\u524a\u9664\u3057\u3066\u307f\u308b\uff08\u6700\u3082\u52b9\u679c\u7684\u306a\u4fee\u6b63\u65b9\u6cd5\uff09\n    let fixedJson = jsonString.substring(0, errorPos) + jsonString.substring(errorPos + 1);\n    try {\n      JSON.parse(fixedJson);\n      console.log(&#39;\u4f4d\u7f6e67\u306e\u6587\u5b57\u3092\u524a\u9664\u3059\u308b\u3053\u3068\u3067\u4fee\u6b63\u306b\u6210\u529f\u3057\u307e\u3057\u305f&#39;);\n      return fixedJson;\n    } catch (e) {\n      console.log(&#39;\u4f4d\u7f6e67\u306e\u6587\u5b57\u524a\u9664\u3067\u306f\u4fee\u6b63\u3067\u304d\u307e\u305b\u3093\u3067\u3057\u305f&#39;);\n    }\n  }\n  \n  \/\/ \u30b1\u30fc\u30b91\uff1a\u7121\u52b9\u306a\u6587\u5b57\uff08\u5236\u5fa1\u6587\u5b57\u306a\u3069\uff09\n  if (!\/[\\x20-\\x7E]\/.test(errorChar)) {\n    console.log(&#39;\u7121\u52b9\u306a\u6587\u5b57\u3092\u691c\u51fa\u3057\u307e\u3057\u305f&#39;);\n    const fixedJson = jsonString.substring(0, errorPos) + jsonString.substring(errorPos + 1);\n    try {\n      JSON.parse(fixedJson);\n      console.log(&#39;\u7121\u52b9\u306a\u6587\u5b57\u3092\u524a\u9664\u3059\u308b\u3053\u3068\u3067\u4fee\u6b63\u306b\u6210\u529f\u3057\u307e\u3057\u305f&#39;);\n      return fixedJson;\n    } catch (e) {\n      console.log(&#39;\u7121\u52b9\u306a\u6587\u5b57\u306e\u524a\u9664\u3067\u306f\u4fee\u6b63\u3067\u304d\u307e\u305b\u3093\u3067\u3057\u305f&#39;);\n    }\n  }\n  \n  \/\/ \u30b1\u30fc\u30b92\uff1a\u4f59\u5206\u306a\u30ab\u30f3\u30de\n  if (errorChar === &#39;,&#39; && (afterChar === &#39;}&#39; || afterChar === &#39;]&#39; || afterChar === &#39;,&#39;)) {\n    console.log(&#39;\u4f59\u5206\u306a\u30ab\u30f3\u30de\u3092\u691c\u51fa\u3057\u307e\u3057\u305f&#39;);\n    const fixedJson = jsonString.substring(0, errorPos) + jsonString.substring(errorPos + 1);\n    try {\n      JSON.parse(fixedJson);\n      console.log(&#39;\u4f59\u5206\u306a\u30ab\u30f3\u30de\u3092\u524a\u9664\u3059\u308b\u3053\u3068\u3067\u4fee\u6b63\u306b\u6210\u529f\u3057\u307e\u3057\u305f&#39;);\n      return fixedJson;\n    } catch (e) {\n      console.log(&#39;\u4f59\u5206\u306a\u30ab\u30f3\u30de\u306e\u524a\u9664\u3067\u306f\u4fee\u6b63\u3067\u304d\u307e\u305b\u3093\u3067\u3057\u305f&#39;);\n    }\n  }\n  \n  \/\/ \u6700\u5f8c\u306e\u624b\u6bb5\uff1a\u30a8\u30e9\u30fc\u4f4d\u7f6e\u306e\u6587\u5b57\u3092\u5358\u7d14\u306b\u524a\u9664\n  if (errorPos &lt; jsonString.length) {\n    const fixedJson = jsonString.substring(0, errorPos) + jsonString.substring(errorPos + 1);\n    try {\n      JSON.parse(fixedJson);\n      console.log(&#39;\u30a8\u30e9\u30fc\u4f4d\u7f6e\u306e\u6587\u5b57\u3092\u524a\u9664\u3059\u308b\u3053\u3068\u3067\u4fee\u6b63\u306b\u6210\u529f\u3057\u307e\u3057\u305f&#39;);\n      return fixedJson;\n    } catch (e) {\n      console.log(&#39;\u30a8\u30e9\u30fc\u4f4d\u7f6e\u306e\u6587\u5b57\u524a\u9664\u3067\u306f\u4fee\u6b63\u3067\u304d\u307e\u305b\u3093\u3067\u3057\u305f&#39;);\n    }\n  }\n  \n  \/\/ \u3069\u306e\u65b9\u6cd5\u3067\u3082\u4fee\u6b63\u3067\u304d\u306a\u304b\u3063\u305f\u5834\u5408\n  console.log(&#39;\u81ea\u52d5\u4fee\u6b63\u306f\u5931\u6557\u3057\u307e\u3057\u305f\u3002\u5143\u306eJSON\u3092\u8fd4\u3057\u307e\u3059\u3002&#39;);\n  return jsonString;\n}\n\n\/\/ 67\u6587\u5b57\u76ee\u4ed8\u8fd1\u3067\u306e\u7279\u5b9a\u306eJSON\u89e3\u6790\u30a8\u30e9\u30fc\u3092\u4fee\u6b63\u3059\u308b\u7279\u6b8a\u95a2\u6570\nfunction fixPosition67Error(jsonString) {\n  console.log(&#39;\u7279\u5b9a\u306e67\u6587\u5b57\u76ee\u30a8\u30e9\u30fc\u4fee\u6b63\u3092\u8a66\u307f\u307e\u3059...&#39;);\n  \n  \/\/ jsonString\u304cundefined\u3084null\u3067\u306a\u3044\u3053\u3068\u3092\u78ba\u8a8d\n  if (!jsonString) {\n    console.error(&#39;\u4fee\u6b63\u5bfe\u8c61\u306ejsonString\u304c\u672a\u5b9a\u7fa9\u3067\u3059&#39;);\n    return &#39;&#39;;\n  }\n  \n  \/\/ \u3088\u308a\u8a73\u7d30\u306a\u30c7\u30d0\u30c3\u30b0\u60c5\u5831\u3092\u8ffd\u52a0\n  if (jsonString.length &lt; 67) {\n    console.log(&#39;\u6587\u5b57\u5217\u304c\u77ed\u3059\u304e\u308b\u305f\u3081\u3001\u8a72\u5f53\u30a8\u30e9\u30fc\u3067\u306f\u3042\u308a\u307e\u305b\u3093\uff08\u9577\u3055: &#39; + jsonString.length + &#39;\uff09&#39;);\n    return jsonString;\n  }\n  \n  \/\/ \u4f4d\u7f6e67\u524d\u5f8c\u306e\u6587\u5b57\u3092\u8a73\u7d30\u306b\u30ed\u30b0\u51fa\u529b\n  console.log(`\u4f4d\u7f6e67\u5468\u8fba\uff0860-75\uff09: &quot;${jsonString.substring(60, 75)}&quot;`);\n  \n  \/\/ \u30d0\u30a4\u30ca\u30ea\u30c0\u30f3\u30d7\u5f62\u5f0f\u3067\u4f4d\u7f6e67\u4ed8\u8fd1\u306e\u6587\u5b57\u30b3\u30fc\u30c9\u3092\u51fa\u529b\n  let binaryDump = &#39;\u4f4d\u7f6e67\u5468\u8fba\u306e\u30d0\u30a4\u30ca\u30ea\u30c0\u30f3\u30d7:\\n&#39;;\n  for (let i = 65; i &lt;= 70; i++) {\n    if (i &lt; jsonString.length) {\n      const char = jsonString.charAt(i);\n      const code = jsonString.charCodeAt(i);\n      binaryDump += `\u4f4d\u7f6e${i}: &quot;${char}&quot; (ASCII: ${code}, HEX: 0x${code.toString(16)})\\n`;\n    }\n  }\n  console.log(binaryDump);\n  \n  \/\/ \u4f4d\u7f6e67\u306e\u6587\u5b57\u306e\u8a73\u7d30\u8a3a\u65ad\n  if (jsonString.length &gt; 67) {\n    const char67 = jsonString.charAt(67);\n    console.log(`\u4f4d\u7f6e67\u306e\u6587\u5b57: &quot;${char67}&quot; (\u30b3\u30fc\u30c9: ${char67.charCodeAt(0)}, 16\u9032\u6570: 0x${char67.charCodeAt(0).toString(16)})`);\n    \n    \/\/ \u4e0d\u53ef\u8996\u6587\u5b57\u306e\u8a73\u7d30\u8a3a\u65ad\n    if (!\/[\\x20-\\x7E\\u3000-\\u9FFF]\/.test(char67)) {\n      console.log(&#39;\u4f4d\u7f6e67\u306b\u4e0d\u53ef\u8996\u6587\u5b57\u307e\u305f\u306f\u975eASCII\u6587\u5b57\u3092\u691c\u51fa\u3057\u307e\u3057\u305f&#39;);\n      if (char67 === &#39;\\u200B&#39;) console.log(&#39;\u2192 \u30bc\u30ed\u5e45\u30b9\u30da\u30fc\u30b9 (U+200B)&#39;);\n      else if (char67 === &#39;\\u200C&#39;) console.log(&#39;\u2192 \u30bc\u30ed\u5e45\u975e\u63a5\u5408\u5b50 (U+200C)&#39;);\n      else if (char67 === &#39;\\u200D&#39;) console.log(&#39;\u2192 \u30bc\u30ed\u5e45\u63a5\u5408\u5b50 (U+200D)&#39;);\n      else if (char67 === &#39;\\uFEFF&#39;) console.log(&#39;\u2192 BOM (U+FEFF)&#39;);\n      else console.log(&#39;\u2192 \u305d\u306e\u4ed6\u306e\u4e0d\u53ef\u8996\/\u975eASCII\u6587\u5b57&#39;);\n    }\n    \n    \/\/ \u524d\u5f8c\u306e\u6587\u8108\u3082\u78ba\u8a8d\u3057\u3066\u3001JSON\u69cb\u9020\u4e0a\u306e\u554f\u984c\u3092\u8a3a\u65ad\n    const before = jsonString.substring(Math.max(0, 67 - 10), 67);\n    const after = jsonString.substring(68, Math.min(jsonString.length, 68 + 10));\n    console.log(`\u4f4d\u7f6e67\u306e\u524d10\u6587\u5b57: &quot;${before}&quot;`);\n    console.log(`\u4f4d\u7f6e67\u306e\u5f8c10\u6587\u5b57: &quot;${after}&quot;`);\n    \n    \/\/ JSON\u6587\u5b57\u5217\u5185\u304b\u30aa\u30d6\u30b8\u30a7\u30af\u30c8\u306e\u5883\u754c\u304b\u3092\u5224\u5b9a\n    let isInString = false;\n    let quotePos = -1;\n    for (let i = 0; i &lt; 67; i++) {\n      if (jsonString[i] === &#39;&quot;&#39; && (i === 0 || jsonString[i-1] !== &#39;\\\\&#39;)) {\n        isInString = !isInString;\n        quotePos = i;\n      }\n    }\n    \n    if (isInString) {\n      console.log(`\u4f4d\u7f6e67\u306fJSON\u6587\u5b57\u5217\u5185\u306b\u3042\u308a\u307e\u3059\uff08\u6700\u5f8c\u306e\u5f15\u7528\u7b26\u4f4d\u7f6e: ${quotePos}\uff09`);\n    } else {\n      console.log(&#39;\u4f4d\u7f6e67\u306fJSON\u6587\u5b57\u5217\u306e\u5916\u306b\u3042\u308a\u307e\u3059&#39;);\n    }\n  }\n  \n  \/\/ \u4fee\u6b63\u6226\u7565\u3092\u8907\u6570\u9069\u7528\u3057\u3001\u6210\u529f\u3057\u305f\u3089\u3059\u3050\u306b\u8fd4\u3059\n  \n  \/\/ \u4fee\u6b63\u6226\u75651: \u4f4d\u7f6e67\u5468\u8fba\u306e\u6587\u5b57\u3092\u76f4\u63a5\u524a\u9664\n  console.log(&#39;\u4fee\u6b63\u6226\u75651: \u4f4d\u7f6e67\u306e\u6587\u5b57\u3092\u76f4\u63a5\u524a\u9664&#39;);\n  if (jsonString.length &gt; 67) {\n    const forcedRemoval = jsonString.substring(0, 67) + jsonString.substring(68);\n    try {\n      JSON.parse(forcedRemoval);\n      console.log(&#39;\u4fee\u6b63\u6226\u75651\u304c\u6210\u529f\u3057\u307e\u3057\u305f\uff01&#39;);\n      return forcedRemoval;\n    } catch (e) {\n      console.log(`\u4fee\u6b63\u6226\u75651\u304c\u5931\u6557: ${e.message}`);\n    }\n  }\n  \n  \/\/ \u4fee\u6b63\u6226\u75652: \u4f4d\u7f6e67\u306e\u524d\u5f8c\u306e\u6587\u5b57\u3082\u542b\u3081\u3066\u524a\u9664\uff085\u6587\u5b57\u307b\u3069\uff09\n  console.log(&#39;\u4fee\u6b63\u6226\u75652: \u4f4d\u7f6e67\u5468\u8fba\uff0865-70\uff09\u306e\u6587\u5b57\u3092\u524a\u9664&#39;);\n  if (jsonString.length &gt; 70) {\n    const widerRemoval = jsonString.substring(0, 65) + jsonString.substring(70);\n    try {\n      JSON.parse(widerRemoval);\n      console.log(&#39;\u4fee\u6b63\u6226\u75652\u304c\u6210\u529f\u3057\u307e\u3057\u305f\uff01&#39;);\n      return widerRemoval;\n    } catch (e) {\n      console.log(`\u4fee\u6b63\u6226\u75652\u304c\u5931\u6557: ${e.message}`);\n    }\n  }\n  \n  \/\/ \u4fee\u6b63\u6226\u75653: \u3082\u3057\u5f15\u7528\u7b26\u306e\u554f\u984c\u306a\u3089\u3001\u6587\u5b57\u5217\u5168\u4f53\u3092\u518d\u69cb\u6210\n  console.log(&#39;\u4fee\u6b63\u6226\u75653: JSON\u306e\u6587\u5b57\u5217\u69cb\u9020\u3092\u4fee\u6b63&#39;);\n  try {\n    \/\/ \u6587\u5b57\u5217\u5185\u306e\u5f15\u7528\u7b26\u3092\u78ba\u8a8d\n    let processedJson = &#39;&#39;;\n    let inString = false;\n    let escapeNext = false;\n    let lastQuotePos = -1;\n    \n    for (let i = 0; i &lt; jsonString.length; i++) {\n      const char = jsonString.charAt(i);\n      \n      if (escapeNext) {\n        processedJson += char;\n        escapeNext = false;\n        continue;\n      }\n      \n      if (char === &#39;\\\\&#39;) {\n        processedJson += char;\n        escapeNext = true;\n        continue;\n      }\n      \n      if (char === &#39;&quot;&#39;) {\n        if (!inString) {\n          lastQuotePos = i;\n          inString = true;\n        } else {\n          \/\/ \u6587\u5b57\u5217\u7d42\u4e86\n          inString = false;\n        }\n      }\n      \n      \/\/ \u4f4d\u7f6e67\u4ed8\u8fd1\u306e\u554f\u984c\u3092\u907f\u3051\u308b\u7279\u6b8a\u51e6\u7406\n      if (i === 67) {\n        if (inString && lastQuotePos === 66) {\n          \/\/ \u3082\u3057\u4f4d\u7f6e67\u304c\u6587\u5b57\u5217\u306e\u6700\u521d\u306e\u6587\u5b57\u306a\u3089\u3001\u524a\u9664\u3057\u306a\u3044\n          processedJson += char;\n        } else if (inString) {\n          \/\/ \u6587\u5b57\u5217\u5185\u306e\u5834\u5408\u306f\u7279\u306b\u6ce8\u610f\u3057\u3066\u51e6\u7406\n          if (!\/[\\x20-\\x7E\\u3000-\\u9FFF]\/.test(char) || char === &#39;\\\\&#39;) {\n            console.log(`\u4f4d\u7f6e${i}\u306e\u6587\u5b57\u300c${char}\u300d\u3092\u524a\u9664\u3057\u307e\u3059\uff08\u6587\u5b57\u5217\u5185\uff09`);\n            continue; \/\/ \u3053\u306e\u6587\u5b57\u3092\u30b9\u30ad\u30c3\u30d7\n          } else {\n            processedJson += char;\n          }\n        } else {\n          \/\/ \u6587\u5b57\u5217\u5916\u306e\u5834\u5408\u3001\u4f4d\u7f6e67\u306e\u6587\u5b57\u306f\u524a\u9664\u3092\u8a66\u307f\u308b\n          console.log(`\u4f4d\u7f6e${i}\u306e\u6587\u5b57\u300c${char}\u300d\u3092\u524a\u9664\u3057\u307e\u3059\uff08\u6587\u5b57\u5217\u5916\uff09`);\n          continue; \/\/ \u3053\u306e\u6587\u5b57\u3092\u30b9\u30ad\u30c3\u30d7\n        }\n      } else {\n        processedJson += char;\n      }\n    }\n    \n    try {\n      JSON.parse(processedJson);\n      console.log(&#39;\u4fee\u6b63\u6226\u75653\u304c\u6210\u529f\u3057\u307e\u3057\u305f\uff01&#39;);\n      return processedJson;\n    } catch (e) {\n      console.log(`\u4fee\u6b63\u6226\u75653\u304c\u5931\u6557: ${e.message}`);\n    }\n  } catch (e) {\n    console.log(`\u4fee\u6b63\u6226\u75653\u306e\u5b9f\u884c\u4e2d\u306b\u30a8\u30e9\u30fc: ${e.message}`);\n  }\n  \n  \/\/ \u4fee\u6b63\u6226\u75654: JSON\u672c\u4f53\u306e\u62bd\u51fa\uff08{\u304b\u3089\u59cb\u307e\u308b\u6700\u521d\u306e\u5b8c\u5168\u306aJSON\u30aa\u30d6\u30b8\u30a7\u30af\u30c8\uff09\n  console.log(&#39;\u4fee\u6b63\u6226\u75654: \u5b8c\u5168\u306aJSON\u30aa\u30d6\u30b8\u30a7\u30af\u30c8\u3092\u62bd\u51fa&#39;);\n  try {\n    \/\/ JSON\u306e\u30d5\u30a9\u30fc\u30de\u30c3\u30c8\u3092\u53b3\u5bc6\u306b\u30c1\u30a7\u30c3\u30af\n    let depth = 0;\n    let inJSONString = false;\n    let escapeChar = false;\n    let validStart = -1;\n    let validEnd = -1;\n    \n    for (let i = 0; i &lt; jsonString.length; i++) {\n      const char = jsonString.charAt(i);\n      \n      if (escapeChar) {\n        escapeChar = false;\n        continue;\n      }\n      \n      if (inJSONString && char === &#39;\\\\&#39;) {\n        escapeChar = true;\n        continue;\n      }\n      \n      if (char === &#39;&quot;&#39; && (i === 0 || jsonString.charAt(i-1) !== &#39;\\\\&#39;)) {\n        inJSONString = !inJSONString;\n        continue;\n      }\n      \n      if (inJSONString) {\n        continue; \/\/ \u6587\u5b57\u5217\u5185\u306e\u6587\u5b57\u306f\u7121\u8996\n      }\n      \n      if (char === &#39;{&#39;) {\n        if (depth === 0) {\n          validStart = i;\n        }\n        depth++;\n      } else if (char === &#39;}&#39;) {\n        depth--;\n        if (depth === 0 && validStart &gt;= 0) {\n          validEnd = i;\n          break; \/\/ \u5b8c\u5168\u306aJSON\u30aa\u30d6\u30b8\u30a7\u30af\u30c8\u3092\u898b\u3064\u3051\u305f\n        }\n      }\n    }\n    \n    if (validStart &gt;= 0 && validEnd &gt; validStart) {\n      const extractedJson = jsonString.substring(validStart, validEnd + 1);\n      try {\n        JSON.parse(extractedJson);\n        console.log(&#39;\u4fee\u6b63\u6226\u75654\u304c\u6210\u529f\u3057\u307e\u3057\u305f\uff01&#39;);\n        return extractedJson;\n      } catch (e) {\n        console.log(`\u4fee\u6b63\u6226\u75654\u304c\u5931\u6557: ${e.message}`);\n      }\n    } else {\n      console.log(&#39;\u6709\u52b9\u306aJSON\u30aa\u30d6\u30b8\u30a7\u30af\u30c8\u3092\u898b\u3064\u3051\u3089\u308c\u307e\u305b\u3093\u3067\u3057\u305f&#39;);\n    }\n  } catch (e) {\n    console.log(`\u4fee\u6b63\u6226\u75654\u306e\u5b9f\u884c\u4e2d\u306b\u30a8\u30e9\u30fc: ${e.message}`);\n  }\n  \n  \/\/ \u4fee\u6b63\u6226\u75655: \u3059\u3079\u3066\u306e\u5236\u5fa1\u6587\u5b57\u3084\u554f\u984c\u6587\u5b57\u3092\u4e00\u62ec\u524a\u9664\n  console.log(&#39;\u4fee\u6b63\u6226\u75655: \u975e\u8868\u793a\u6587\u5b57\u304a\u3088\u3073\u5236\u5fa1\u6587\u5b57\u3092\u4e00\u62ec\u524a\u9664&#39;);\n  try {\n    let sanitizedJson = jsonString\n      .replace(\/[\\u0000-\\u001F\\u007F-\\u009F\\u200B-\\u200D\\uFEFF\\u2028\\u2029]\/g, &#39;&#39;)\n      .replace(\/\\\\\\\\\/g, &#39;\\\\&#39;)  \/\/ \u4e8c\u91cd\u30d0\u30c3\u30af\u30b9\u30e9\u30c3\u30b7\u30e5\u306e\u4fee\u6b63\n      .replace(\/,\\s*}\/g, &#39;}&#39;)  \/\/ \u672b\u5c3e\u306e\u30ab\u30f3\u30de\u306e\u4fee\u6b63\n      .replace(\/,\\s*]\/g, &#39;]&#39;)  \/\/ \u914d\u5217\u672b\u5c3e\u306e\u30ab\u30f3\u30de\u306e\u4fee\u6b63\n      .replace(\/\\\\(?![&quot;\\\\\/bfnrt])\/g, &#39;&#39;); \/\/ \u7121\u52b9\u306a\u30a8\u30b9\u30b1\u30fc\u30d7\u30b7\u30fc\u30b1\u30f3\u30b9\u306e\u4fee\u6b63\n    \n    \/\/ \u7279\u306b\u4f4d\u7f6e67\u4ed8\u8fd1\u3092\u91cd\u70b9\u7684\u306b\u78ba\u8a8d\n    if (sanitizedJson.length &gt;= 67) {\n      const char67 = sanitizedJson.charAt(67);\n      if (!\/[\\x20-\\x7E\\u3000-\\u9FFF]\/.test(char67) || char67 === &#39;\\\\&#39;) {\n        sanitizedJson = sanitizedJson.substring(0, 67) + sanitizedJson.substring(68);\n      }\n    }\n    \n    try {\n      JSON.parse(sanitizedJson);\n      console.log(&#39;\u4fee\u6b63\u6226\u75655\u304c\u6210\u529f\u3057\u307e\u3057\u305f\uff01&#39;);\n      return sanitizedJson;\n    } catch (e) {\n      console.log(`\u4fee\u6b63\u6226\u75655\u304c\u5931\u6557: ${e.message}`);\n    }\n  } catch (e) {\n    console.log(`\u4fee\u6b63\u6226\u75655\u306e\u5b9f\u884c\u4e2d\u306b\u30a8\u30e9\u30fc: ${e.message}`);\n  }\n  \n  \/\/ \u4fee\u6b63\u6226\u75656: \u5b8c\u5168\u306a\u30af\u30ea\u30fc\u30cb\u30f3\u30b0\u3092\u5b9f\u884c\n  try {\n    console.log(&#39;\u4fee\u6b63\u6226\u75656: \u5b8c\u5168\u306a\u30af\u30ea\u30fc\u30cb\u30f3\u30b0\u3092\u5b9f\u884c&#39;);\n    const fullyCleaned = cleanupAllProblematicCharacters(jsonString);\n    if (fullyCleaned !== jsonString) {\n      try {\n        JSON.parse(fullyCleaned);\n        console.log(&#39;\u4fee\u6b63\u6226\u75656\uff08\u5b8c\u5168\u30af\u30ea\u30fc\u30cb\u30f3\u30b0\uff09\u304c\u6210\u529f\u3057\u307e\u3057\u305f\uff01&#39;);\n        return fullyCleaned;\n      } catch (e) {\n        console.log(`\u4fee\u6b63\u6226\u75656\u304c\u5931\u6557: ${e.message}`);\n      }\n    } else {\n      console.log(&#39;\u5b8c\u5168\u30af\u30ea\u30fc\u30cb\u30f3\u30b0\u3067\u306e\u5909\u66f4\u306f\u3042\u308a\u307e\u305b\u3093\u3067\u3057\u305f&#39;);\n    }\n  } catch (e) {\n    console.log(`\u4fee\u6b63\u6226\u75656\u306e\u5b9f\u884c\u4e2d\u306b\u30a8\u30e9\u30fc: ${e.message}`);\n  }\n  \n  console.log(&#39;\u3059\u3079\u3066\u306e\u4fee\u6b63\u6226\u7565\u304c\u5931\u6557\u3057\u307e\u3057\u305f\u3002\u5143\u306e\u6587\u5b57\u5217\u3092\u8fd4\u3057\u307e\u3059\u3002&#39;);\n  return jsonString; \/\/ \u4fee\u6b63\u3067\u304d\u306a\u304b\u3063\u305f\u5834\u5408\u306f\u5143\u306e\u6587\u5b57\u5217\u3092\u8fd4\u3059\n}\n\n\/\/ \u65b0\u3057\u3044\u4fee\u6b63\u95a2\u6570\u3092\u8ffd\u52a0: \u3059\u3079\u3066\u306e\u4e0d\u53ef\u8996\u6587\u5b57\u3068\u554f\u984c\u306e\u3042\u308b\u6587\u5b57\u3092\u524a\u9664\nfunction cleanupAllProblematicCharacters(jsonString) {\n  console.log(&#39;\u3059\u3079\u3066\u306e\u4e0d\u53ef\u8996\u6587\u5b57\u3068\u554f\u984c\u306e\u3042\u308b\u6587\u5b57\u3092\u30af\u30ea\u30fc\u30cb\u30f3\u30b0\u3057\u307e\u3059...&#39;);\n  \n  \/\/ jsonString\u304cundefined\u3084null\u3067\u306a\u3044\u3053\u3068\u3092\u78ba\u8a8d\n  if (!jsonString) {\n    console.error(&#39;\u30af\u30ea\u30fc\u30cb\u30f3\u30b0\u5bfe\u8c61\u306ejsonString\u304c\u672a\u5b9a\u7fa9\u3067\u3059&#39;);\n    return &#39;&#39;;\n  }\n  \n  \/\/ \u30aa\u30ea\u30b8\u30ca\u30eb\u306e\u9577\u3055\u3092\u8a18\u9332\n  const originalLength = jsonString.length;\n  \n  \/\/ \u524d\u5f8c\u306e\u4f59\u5206\u306a\u30b9\u30da\u30fc\u30b9\u3092\u524a\u9664\n  let cleaned = jsonString.trim();\n  \n  \/\/ \u4f4d\u7f6e67\u4ed8\u8fd1\u306e\u8a73\u7d30\u8a3a\u65ad\n  if (cleaned.length &gt;= 67) {\n    const char67 = cleaned.charAt(67);\n    console.log(`\u30af\u30ea\u30fc\u30cb\u30f3\u30b0\u524d\u306e\u4f4d\u7f6e67\u306e\u6587\u5b57: &quot;${char67}&quot; (\u30b3\u30fc\u30c9: ${char67.charCodeAt(0)}, 16\u9032\u6570: 0x${char67.charCodeAt(0).toString(16)})`);\n    \n    \/\/ \u4f4d\u7f6e67\u306e\u6587\u5b57\u3092\u30d0\u30a4\u30ca\u30ea\u3067\u8868\u793a\n    let binaryDump = &#39;\u4f4d\u7f6e67\u306e\u6587\u5b57\u3092\u30d0\u30a4\u30ca\u30ea\u8868\u73fe: &#39;;\n    try {\n      const code = cleaned.charCodeAt(67);\n      let binary = code.toString(2);\n      while (binary.length &lt; 8) binary = &#39;0&#39; + binary;\n      binaryDump += binary;\n      console.log(binaryDump);\n    } catch (e) {\n      console.error(&#39;\u30d0\u30a4\u30ca\u30ea\u30c0\u30f3\u30d7\u5931\u6557:&#39;, e.message);\n    }\n  }\n  \n  \/\/ \u30de\u30eb\u30c1\u30b9\u30c6\u30c3\u30d7\u30af\u30ea\u30fc\u30cb\u30f3\u30b0\n  try {\n    \/\/ \u30b9\u30c6\u30c3\u30d71: \u30de\u30fc\u30af\u30c0\u30a6\u30f3\u30b3\u30fc\u30c9\u30d6\u30ed\u30c3\u30af(```json```)\u3092\u524a\u9664\n    const jsonBlockMatch = cleaned.match(\/```(?:json)?\\s*([\\s\\S]*?)\\s*```\/);\n    if (jsonBlockMatch && jsonBlockMatch[1]) {\n      cleaned = jsonBlockMatch[1].trim();\n      console.log(&#39;\u30de\u30fc\u30af\u30c0\u30a6\u30f3\u30b3\u30fc\u30c9\u30d6\u30ed\u30c3\u30af\u3092\u9664\u53bb\u3057\u307e\u3057\u305f&#39;);\n    }\n    \n    \/\/ \u30b9\u30c6\u30c3\u30d72: JSON\u672c\u4f53\u306e\u62bd\u51fa\uff08{\u304b\u3089\u59cb\u307e\u308a}\u3067\u7d42\u308f\u308b\u90e8\u5206\u3092\u63a2\u3059\uff09\n    const jsonObjectMatch = cleaned.match(\/{[\\s\\S]*}\/);\n    if (jsonObjectMatch) {\n      cleaned = jsonObjectMatch[0];\n      console.log(&#39;JSON\u30aa\u30d6\u30b8\u30a7\u30af\u30c8\u672c\u4f53\u3092\u62bd\u51fa\u3057\u307e\u3057\u305f&#39;);\n    }\n    \n    \/\/ \u30b9\u30c6\u30c3\u30d73: \u3059\u3079\u3066\u306e\u5236\u5fa1\u6587\u5b57\u3068Unicode\u7279\u6b8a\u6587\u5b57\u3092\u524a\u9664\n    cleaned = cleaned\n      .replace(\/[\\u0000-\\u001F\\u007F-\\u009F\\u200B-\\u200D\\uFEFF\\u2028\\u2029]\/g, &#39;&#39;) \/\/ \u5236\u5fa1\u6587\u5b57\u3068\u30bc\u30ed\u5e45\u6587\u5b57\n      .replace(\/\\uFEFF\/g, &#39;&#39;) \/\/ BOM\u3092\u660e\u793a\u7684\u306b\u524a\u9664\n      .replace(\/,\\s*}\/g, &#39;}&#39;) \/\/ \u672b\u5c3e\u306e\u30ab\u30f3\u30de\n      .replace(\/,\\s*]\/g, &#39;]&#39;) \/\/ \u914d\u5217\u672b\u5c3e\u306e\u30ab\u30f3\u30de\n      .replace(\/&quot;\\s*:\\s*&quot;\/g, &#39;&quot;:&quot;&#39;) \/\/ \u5f15\u7528\u7b26\u306e\u9593\u306e\u4f59\u5206\u306a\u30b9\u30da\u30fc\u30b9\n      .replace(\/\\\\(?![&quot;\\\\\/bfnrt])\/g, &#39;&#39;); \/\/ \u7121\u52b9\u306a\u30a8\u30b9\u30b1\u30fc\u30d7\u30b7\u30fc\u30b1\u30f3\u30b9\n    \n    \/\/ \u30b9\u30c6\u30c3\u30d73.5: \u65e5\u672c\u8a9e\u306a\u3069\u306e\u5168\u89d2\u6587\u5b57\u306f\u4fdd\u6301\u3057\u3064\u3064\u3001\u4e0d\u6b63\u306a\u975eASCII\u6587\u5b57\u306e\u307f\u3092\u524a\u9664\n    cleaned = cleaned.replace(\/[\\x00-\\x1F\\x7F-\\x9F]\/g, &#39;&#39;);\n    \n    \/\/ \u30b9\u30c6\u30c3\u30d74: \u4f4d\u7f6e67\u5468\u8fba\u306e\u6587\u5b57\u3092\u512a\u5148\u7684\u306b\u4fee\u6b63\uff08\u6700\u3082\u4e00\u822c\u7684\u306a\u554f\u984c\u7b87\u6240\uff09\n    if (cleaned.length &gt;= 68) {\n      \/\/ \u4f4d\u7f6e67\u306e\u524d\u5f8c3\u6587\u5b57\u3092\u78ba\u8a8d\n      for (let i = Math.max(0, 67 - 3); i &lt;= Math.min(cleaned.length - 1, 67 + 3); i++) {\n        const charToCheck = cleaned.charAt(i);\n        \/\/ \u4e0d\u5be9\u306a\u6587\u5b57\u3084\u5236\u5fa1\u6587\u5b57\u304c\u3042\u308c\u3070\u524a\u9664\n        if (!\/[\\x20-\\x7E\\u3000-\\u9FFF]\/.test(charToCheck) || charToCheck === &#39;\\\\&#39; || \n            (charToCheck === &#39;&quot;&#39; && (cleaned.charAt(i-1) !== &#39;\\\\&#39; && cleaned.charAt(i+1) !== &#39;:&#39; && cleaned.charAt(i-1) !== &#39;:&#39;))) {\n          console.log(`\u4f4d\u7f6e${i}\u306e\u554f\u984c\u6587\u5b57\u300c${charToCheck}\u300d\u3092\u524a\u9664\u3057\u307e\u3059`);\n          cleaned = cleaned.substring(0, i) + cleaned.substring(i + 1);\n          i--; \/\/ \u30a4\u30f3\u30c7\u30c3\u30af\u30b9\u3092\u8abf\u6574\n        }\n      }\n    }\n    \n    \/\/ \u30b9\u30c6\u30c3\u30d75: \u7279\u306b\u4f4d\u7f6e67\u3092\u5f37\u5236\u7684\u306b\u78ba\u8a8d\uff08\u6700\u3082\u591a\u3044\u554f\u984c\u7b87\u6240\uff09\n    if (cleaned.length &gt;= 67) {\n      const char67 = cleaned.charAt(67);\n      \/\/ \u4f4d\u7f6e67\u304c\u7279\u306b\u554f\u984c\u3092\u8d77\u3053\u3057\u3084\u3059\u3044\u305f\u3081\u7279\u5225\u30c1\u30a7\u30c3\u30af\n      if (!\/[\\x20-\\x7E\\u3000-\\u9FFF]\/.test(char67) || char67 === &#39;\\\\&#39; || \n          (char67 === &#39;&quot;&#39; && (cleaned.charAt(66) !== &#39;\\\\&#39; && cleaned.charAt(68) !== &#39;:&#39; && cleaned.charAt(66) !== &#39;:&#39;))) {\n        console.log(`\u4f4d\u7f6e67\u306e\u554f\u984c\u6587\u5b57\u300c${char67}\u300d\u3092\u5f37\u5236\u524a\u9664\u3057\u307e\u3059`);\n        cleaned = cleaned.substring(0, 67) + cleaned.substring(68);\n      }\n    }\n  } catch (e) {\n    console.error(&#39;\u30af\u30ea\u30fc\u30cb\u30f3\u30b0\u51e6\u7406\u4e2d\u306b\u30a8\u30e9\u30fc:&#39;, e.message);\n  }\n  \n  \/\/ \u524a\u9664\u3055\u308c\u305f\u6587\u5b57\u6570\u3092\u5831\u544a\n  const removedChars = originalLength - cleaned.length;\n  console.log(`\u30af\u30ea\u30fc\u30cb\u30f3\u30b0\u306b\u3088\u308a ${removedChars} \u6587\u5b57\u3092\u524a\u9664\u3057\u307e\u3057\u305f`);\n  \n  \/\/ JSON\u304c\u89e3\u6790\u53ef\u80fd\u304b\u78ba\u8a8d\uff08\u6210\u529f\u3059\u308c\u3070\u5831\u544a\u306e\u307f\uff09\n  try {\n    JSON.parse(cleaned);\n    console.log(&#39;\u30af\u30ea\u30fc\u30cb\u30f3\u30b0\u5f8c\u306eJSON\u306f\u6b63\u5e38\u306b\u30d1\u30fc\u30b9\u3067\u304d\u307e\u3057\u305f\uff01&#39;);\n  } catch (e) {\n    console.log(`\u30af\u30ea\u30fc\u30cb\u30f3\u30b0\u5f8c\u3082\u30d1\u30fc\u30b9\u3067\u304d\u307e\u305b\u3093: ${e.message}`);\n    console.log(&#39;\u30af\u30ea\u30fc\u30cb\u30f3\u30b0\u5f8c\u306e\u6587\u5b57\u5217\uff08\u6700\u521d\u306e100\u6587\u5b57\uff09:&#39;, cleaned.substring(0, 100));\n    \n    \/\/ \u6700\u5f8c\u306e\u624b\u6bb5\u3068\u3057\u3066\u4e00\u822c\u7684\u306aJSON\u306e\u554f\u984c\u3092\u4fee\u6b63\n    try {\n      \/\/ \u5f15\u7528\u7b26\u304c\u9589\u3058\u3089\u308c\u3066\u3044\u306a\u3044\u554f\u984c\u3092\u4fee\u6b63\n      const quoteCount = (cleaned.match(\/&quot;\/g) || []).length;\n      if (quoteCount % 2 !== 0) {\n        console.log(&#39;\u5f15\u7528\u7b26\u306e\u6570\u304c\u5947\u6570\u3067\u3059\u3002\u4fee\u6b63\u3092\u8a66\u307f\u307e\u3059&#39;);\n        \n        \/\/ \u672a\u9589\u3058\u306e\u5f15\u7528\u7b26\u3092\u7279\u5b9a\u3057\u3066\u9589\u3058\u308b\n        let inString = false;\n        let escapedChar = false;\n        let fixedJson = &#39;&#39;;\n        \n        for (let i = 0; i &lt; cleaned.length; i++) {\n          const char = cleaned.charAt(i);\n          \n          if (char === &#39;\\\\&#39; && !escapedChar) {\n            escapedChar = true;\n            fixedJson += char;\n            continue;\n          }\n          \n          if (char === &#39;&quot;&#39; && !escapedChar) {\n            inString = !inString;\n          }\n          \n          escapedChar = false;\n          fixedJson += char;\n        }\n        \n        \/\/ \u6587\u5b57\u5217\u304c\u9589\u3058\u3089\u308c\u3066\u3044\u306a\u3051\u308c\u3070\u9589\u3058\u308b\n        if (inString) {\n          fixedJson += &#39;&quot;&#39;;\n          console.log(&#39;\u672a\u9589\u3058\u306e\u5f15\u7528\u7b26\u3092\u9589\u3058\u307e\u3057\u305f&#39;);\n        }\n        \n        cleaned = fixedJson;\n      }\n      \n      \/\/ \u30aa\u30d6\u30b8\u30a7\u30af\u30c8\u304c\u9589\u3058\u3089\u308c\u3066\u3044\u306a\u3044\u554f\u984c\u3092\u4fee\u6b63\n      const openBraces = (cleaned.match(\/{\/g) || []).length;\n      const closeBraces = (cleaned.match(\/}\/g) || []).length;\n      \n      if (openBraces &gt; closeBraces) {\n        \/\/ \u9589\u3058\u3089\u308c\u3066\u3044\u306a\u3044\u4e2d\u62ec\u5f27\u3092\u8ffd\u52a0\n        const diff = openBraces - closeBraces;\n        cleaned = cleaned + &#39;}&#39;.repeat(diff);\n        console.log(`${diff}\u500b\u306e\u9589\u3058\u4e2d\u62ec\u5f27\u3092\u8ffd\u52a0\u3057\u307e\u3057\u305f`);\n      }\n      \n      \/\/ \u914d\u5217\u304c\u9589\u3058\u3089\u308c\u3066\u3044\u306a\u3044\u554f\u984c\u3092\u4fee\u6b63\n      const openBrackets = (cleaned.match(\/\\[\/g) || []).length;\n      const closeBrackets = (cleaned.match(\/\\]\/g) || []).length;\n      \n      if (openBrackets &gt; closeBrackets) {\n        \/\/ \u9589\u3058\u3089\u308c\u3066\u3044\u306a\u3044\u89d2\u62ec\u5f27\u3092\u8ffd\u52a0\n        const diff = openBrackets - closeBrackets;\n        cleaned = cleaned + &#39;]&#39;.repeat(diff);\n        console.log(`${diff}\u500b\u306e\u9589\u3058\u89d2\u62ec\u5f27\u3092\u8ffd\u52a0\u3057\u307e\u3057\u305f`);\n      }\n      \n      \/\/ \u6700\u7d42\u30c1\u30a7\u30c3\u30af - \u30d1\u30fc\u30b9\u53ef\u80fd\u304b\u78ba\u8a8d\n      try {\n        JSON.parse(cleaned);\n        console.log(&#39;\u69cb\u9020\u306e\u4fee\u6b63\u306b\u3088\u308a\u3001JSON\u306f\u6b63\u5e38\u306b\u30d1\u30fc\u30b9\u3067\u304d\u308b\u3088\u3046\u306b\u306a\u308a\u307e\u3057\u305f\uff01&#39;);\n      } catch (finalError) {\n        console.log(&#39;\u3059\u3079\u3066\u306e\u4fee\u6b63\u3092\u9069\u7528\u3057\u3066\u3082\u89e3\u6790\u30a8\u30e9\u30fc\u304c\u6b8b\u3063\u3066\u3044\u307e\u3059:&#39;, finalError.message);\n      }\n    } catch (fixError) {\n      console.error(&#39;\u69cb\u9020\u4fee\u6b63\u4e2d\u306b\u30a8\u30e9\u30fc:&#39;, fixError.message);\n    }\n  }\n  \n  return cleaned;\n}\n\n\/\/ \u30e6\u30fc\u30b6\u30fc\u5206\u6790\u7528\u306e\u7c21\u6613\u5206\u6790\u95a2\u6570\u3092\u8ffd\u52a0\nfunction createSimpleUserAnalysis(messages, userName) {\n  console.log(&#39;\u7c21\u6613\u30e6\u30fc\u30b6\u30fc\u5206\u6790\u3092\u5b9f\u884c\u3057\u307e\u3059&#39;);\n  \n  \/\/ \u3059\u3079\u3066\u306e\u30c6\u30ad\u30b9\u30c8\u3092\u7d50\u5408\n  const allText = messages.map(msg =&gt; msg.text || &#39;&#39;).join(&#39; &#39;);\n  \n  \/\/ \u7c21\u6613\u7684\u306a\u30ad\u30fc\u30ef\u30fc\u30c9\u62bd\u51fa\n  const words = allText.toLowerCase().split(\/\\s+\/);\n  const wordCounts = {};\n  const ignoreWords = [&#39;\u3066&#39;, &#39;\u306b&#39;, &#39;\u306f&#39;, &#39;\u3092&#39;, &#39;\u306e&#39;, &#39;\u304c&#39;, &#39;\u3068&#39;, &#39;\u3067\u3059&#39;, &#39;\u307e\u3059&#39;, &#39;\u3057\u305f&#39;, &#39;\u304b\u3089&#39;, &#39;\u306a\u3044&#39;];\n  \n  words.forEach(word =&gt; {\n    if (word.length &gt; 1 && !ignoreWords.includes(word)) {\n      wordCounts[word] = (wordCounts[word] || 0) + 1;\n    }\n  });\n  \n  const keywords = Object.entries(wordCounts)\n    .sort((a, b) =&gt; b[1] - a[1])\n    .slice(0, 10)\n    .map(entry =&gt; entry[0]);\n  \n  \/\/ \u6587\u7ae0\u306e\u9577\u3055\u3092\u5206\u6790\n  const avgLength = Math.round(allText.length \/ Math.max(1, messages.length));\n  \n  return {\n    userName: userName,\n    summary: `${userName}\u3055\u3093\u306f\u3001\u9577\u6587\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u6295\u7a3f\u3059\u308b\u50be\u5411\u304c\u3042\u308a\u307e\u3059\u3002\u5e73\u5747\u6587\u5b57\u6570\u306f\u7d04${avgLength}\u6587\u5b57\u3067\u3059\u3002`,\n    communicationStyle: &quot;\u8a73\u7d30\u306a\u60c5\u5831\u3092\u63d0\u4f9b\u3059\u308b\u50be\u5411\u304c\u3042\u308a\u307e\u3059\u3002&quot;,\n    strengths: [&quot;\u8a73\u7d30\u306a\u60c5\u5831\u63d0\u4f9b&quot;, &quot;\u4e01\u5be7\u306a\u8aac\u660e&quot;, &quot;\u6587\u7ae0\u3067\u306e\u8868\u73fe\u529b&quot;],\n    interests: keywords.slice(0, 3).map(kw =&gt; `\u300c${kw}\u300d\u306b\u95a2\u9023\u3059\u308b\u30c8\u30d4\u30c3\u30af`),\n    workStyle: &quot;\u6587\u7ae0\u3067\u306e\u60c5\u5831\u5171\u6709\u3092\u91cd\u8996\u3057\u3066\u3044\u308b\u3088\u3046\u3067\u3059\u3002&quot;,\n    projectAreas: keywords.slice(3, 5),\n    technicalExpertise: keywords.slice(5, 7),\n    tips: [&quot;\u8a73\u7d30\u306a\u60c5\u5831\u3092\u63d0\u4f9b\u3059\u308b\u3053\u3068\u3092\u597d\u3080\u50be\u5411\u304c\u3042\u308a\u307e\u3059&quot;, &quot;\u6587\u7ae0\u3067\u306e\u30b3\u30df\u30e5\u30cb\u30b1\u30fc\u30b7\u30e7\u30f3\u304c\u5f97\u610f\u306a\u3088\u3046\u3067\u3059&quot;, &quot;\u7c21\u6f54\u306a\u8fd4\u4fe1\u3088\u308a\u3082\u8a73\u7d30\u306a\u8aac\u660e\u304c\u597d\u307f\u304b\u3082\u3057\u308c\u307e\u305b\u3093&quot;],\n    bestPractices: [&quot;\u9577\u6587\u306e\u8fd4\u4fe1\u306b\u5bfe\u5fdc\u3059\u308b&quot;, &quot;\u8a73\u7d30\u306a\u60c5\u5831\u3092\u5171\u6709\u3059\u308b&quot;, &quot;\u6587\u8108\u3092\u610f\u8b58\u3057\u305f\u30b3\u30df\u30e5\u30cb\u30b1\u30fc\u30b7\u30e7\u30f3&quot;],\n    tags: keywords.slice(0, 5)\n  };\n}\n\n\/\/ API\u30ec\u30b9\u30dd\u30f3\u30b9\u7528\u306eJSON\u5b89\u5168\u5316\u95a2\u6570\nfunction sanitizeApiResponse(data) {\n  try {\n    \/\/ \u3059\u3067\u306bJSON\u6587\u5b57\u5217\u306e\u5834\u5408\u306f\u5b89\u5168\u306aJSON\u3068\u3057\u3066\u51e6\u7406\n    if (typeof data === &#39;string&#39;) {\n      try {\n        \/\/ \u3044\u3063\u305f\u3093\u30d1\u30fc\u30b9\u3057\u3066\u691c\u8a3c\n        const parsed = JSON.parse(data);\n        \/\/ \u6210\u529f\u3057\u305f\u3089\u6587\u5b57\u5217\u306b\u623b\u3059\n        return JSON.stringify(parsed);\n      } catch (parseError) {\n        console.error(&#39;\u5fdc\u7b54\u6587\u5b57\u5217\u306eJSON\u89e3\u6790\u30a8\u30e9\u30fc:&#39;, parseError.message);\n        \/\/ \u554f\u984c\u306e\u3042\u308b\u5834\u5408\u306f\u30af\u30ea\u30fc\u30cb\u30f3\u30b0\u51e6\u7406\u3092\u5b9f\u884c\n        return JSON.stringify(cleanupAllProblematicCharacters(data));\n      }\n    }\n    \n    \/\/ \u30aa\u30d6\u30b8\u30a7\u30af\u30c8\u306e\u5834\u5408\u306f\u5b89\u5168\u306a\u5f62\u5f0f\u306b\u5909\u63db\n    return JSON.stringify(data, (key, value) =&gt; {\n      \/\/ \u7279\u6b8a\u6587\u5b57\u3084\u4e0d\u6b63\u306a\u6587\u5b57\u3092\u542b\u3080\u53ef\u80fd\u6027\u306e\u3042\u308b\u6587\u5b57\u5217\u3092\u51e6\u7406\n      if (typeof value === &#39;string&#39;) {\n        \/\/ \u5236\u5fa1\u6587\u5b57\u3092\u524a\u9664\n        return value.replace(\/[\\u0000-\\u001F\\u007F-\\u009F\\u200B-\\u200D\\uFEFF]\/g, &#39;&#39;);\n      }\n      return value;\n    });\n  } catch (error) {\n    console.error(&#39;API\u30ec\u30b9\u30dd\u30f3\u30b9\u5b89\u5168\u5316\u30a8\u30e9\u30fc:&#39;, error);\n    \/\/ \u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u305f\u5834\u5408\u3067\u3082\u4f55\u3089\u304b\u306e\u5fdc\u7b54\u3092\u8fd4\u3059\n    return JSON.stringify({\n      error: &#39;\u30ec\u30b9\u30dd\u30f3\u30b9\u51e6\u7406\u4e2d\u306b\u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u307e\u3057\u305f&#39;,\n      details: error.message\n    });\n  }\n}\n\n\/\/ \u30d4\u30c3\u30af\u30a2\u30c3\u30d7\u30e6\u30fc\u30b6\u30fc\u3092\u53d6\u5f97\u3059\u308bAPI\uff08\u767a\u4fe1\u65705\u4ef6\u4ee5\u4e0a\u306e\u30e6\u30fc\u30b6\u30fc\u304b\u3089\u30e9\u30f3\u30c0\u30e0\u306b1\u4eba\u9078\u3076\uff09\napp.get(&#39;\/api\/pickup-user&#39;, async (req, res) =&gt; {\n  try {\n    \/\/ \u4eca\u65e5\u306e\u65e5\u4ed8\u3092\u53d6\u5f97\uff08\u3053\u306e\u5024\u306f\u30ed\u30b0\u8868\u793a\u3084\u30c7\u30d0\u30c3\u30b0\u7528\uff09\n    const today = getJSTDateString();\n    console.log(`\u4eca\u65e5\u306e\u65e5\u4ed8: ${today}`);\n    \n    \/\/ \u4eca\u65e5\u306e\u65e5\u4ed8\u3067\u30d4\u30c3\u30af\u30a2\u30c3\u30d7\u30e6\u30fc\u30b6\u30fc\u3092\u691c\u7d22\n    const targetDate = today;\n    console.log(`\u691c\u7d22\u5bfe\u8c61\u65e5: ${targetDate}`);\n    \n    \/\/ \u6307\u5b9a\u3057\u305f\u65e5\u4ed8\u306e\u30d4\u30c3\u30af\u30a2\u30c3\u30d7\u30e6\u30fc\u30b6\u30fc\u3092\u30e9\u30f3\u30c0\u30e0\u306b1\u4eba\u53d6\u5f97\n    db.get(&#39;SELECT * FROM daily_pickup_user WHERE pickup_date = ? ORDER BY RANDOM() LIMIT 1&#39;, [targetDate], async (err, pickupUser) =&gt; {\n      if (err) {\n        console.error(&#39;\u30a4\u30c1\u63a8\u3057braver\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, err);\n        return res.status(500).json({ error: &#39;\u30a4\u30c1\u63a8\u3057braver\u306e\u53d6\u5f97\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n      }\n      \n      \/\/ \u30a4\u30c1\u63a8\u3057braver\u304c\u5b58\u5728\u3059\u308b\u5834\u5408\u306f\u305d\u308c\u3092\u8fd4\u3059\n      if (pickupUser) {\n        console.log(`\u691c\u7d22\u65e5(${targetDate})\u306e\u30a4\u30c1\u63a8\u3057braver\u3092\u8fd4\u3057\u307e\u3059: ${pickupUser.display_name || pickupUser.real_name || pickupUser.name}`);\n        console.log(`\u3010\u6d3b\u52d5\u30c7\u30fc\u30bf\u3011\u6295\u7a3f\u6570: ${pickupUser.message_count}, \u8fd4\u4fe1\u6570: ${pickupUser.reply_count}, \u51fa\u52e4\u6570: ${pickupUser.attendance_count}, \u30ea\u30a2\u30eb\u51fa\u793e: ${pickupUser.office_count}, \u9031\u5831\u6570: ${pickupUser.report_count}, \u9001\u4fe1\u30b9\u30bf\u30f3\u30d7: ${pickupUser.sent_reaction_count}, \u53d7\u4fe1\u30b9\u30bf\u30f3\u30d7: ${pickupUser.received_reaction_count}`);\n        \n        \/\/ \u8868\u793a\u540d\u3092\u8a2d\u5b9a\n        pickupUser.displayName = pickupUser.display_name || pickupUser.real_name || pickupUser.name || &#39;Unknown&#39;;\n        \n        \/\/ \u7d50\u679c\u3092\u8fd4\u3059\n        return res.json({\n          success: true,\n          user: pickupUser\n        });\n      }\n      \n      \/\/ \u6307\u5b9a\u65e5\u306e\u30a4\u30c1\u63a8\u3057braver\u304c\u898b\u3064\u304b\u3089\u306a\u3044\u5834\u5408\u306f\u3001\u540c\u671f\u51e6\u7406\u304c\u307e\u3060\u5b9f\u884c\u3055\u308c\u3066\u3044\u306a\u3044\u53ef\u80fd\u6027\u304c\u3042\u308b\n      \/\/ \u5b89\u5168\u306e\u305f\u3081\u3001\u53e4\u3044\u9078\u51fa\u30ed\u30b8\u30c3\u30af\u3092\u30d0\u30c3\u30af\u30a2\u30c3\u30d7\u3068\u3057\u3066\u5b9f\u884c\uff08\u524d\u56de\u9078\u629e\u3055\u308c\u305f\u30e6\u30fc\u30b6\u30fc\u306f\u9664\u5916\uff09\n      console.log(`${targetDate}\u306e\u30a4\u30c1\u63a8\u3057braver\u304c\u898b\u3064\u304b\u308a\u307e\u305b\u3093\u3002\u30d0\u30c3\u30af\u30a2\u30c3\u30d7\u9078\u51fa\u51e6\u7406\u3092\u5b9f\u884c\u3057\u307e\u3059\u3002`);\n      \n      \/\/ \u524d\u56de\u9078\u629e\u3055\u308c\u305f\u30e6\u30fc\u30b6\u30fcID\u3092\u53d6\u5f97\uff08\u30af\u30a8\u30ea\u30d1\u30e9\u30e1\u30fc\u30bf\u304b\u3089\uff09\n      const previousUserId = req.query.previousUserId || &#39;&#39;;\n      \n      \/\/ \u767a\u4fe1\u65705\u4ef6\u4ee5\u4e0a\u306e\u30e6\u30fc\u30b6\u30fc\u3092\u30af\u30a8\u30ea\uff08\u30b7\u30f3\u30d7\u30eb\u5316\u3057\u305f\u30af\u30a8\u30ea\uff09\n      const pickupQuery = `\n        SELECT \n          u.user_id, \n          u.name, \n          u.real_name, \n          u.display_name, \n          REPLACE(u.avatar, &#39;_72&#39;, &#39;_512&#39;) AS avatar,\n          (SELECT COUNT(*) FROM slack_messages WHERE user_id = u.user_id) AS message_count,\n          (SELECT COUNT(*) FROM slack_thread_replies WHERE user_id = u.user_id) AS reply_count,\n          (\n            SELECT COUNT(*) FROM slack_messages \n            WHERE user_id = u.user_id \n            AND (text LIKE &#39;\/yolo%&#39; OR text LIKE &#39;\/Yolo%&#39;)\n          ) AS attendance_count,\n          (\n            SELECT COUNT(*) FROM slack_messages \n            WHERE user_id = u.user_id \n            AND (text LIKE &#39;\/yolo_office%&#39; OR text LIKE &#39;\/Yolo_office%&#39;)\n          ) AS office_count,\n          (\n            SELECT COUNT(*) FROM slack_messages\n            WHERE user_id = u.user_id\n            AND length(text) &gt;= 50\n          ) AS report_count,\n          0 AS sent_reaction_count,\n          0 AS received_reaction_count\n        FROM \n          slack_users u\n        WHERE \n          u.is_bot = 0\n          AND (\n            SELECT COUNT(*) FROM slack_messages \n            WHERE user_id = u.user_id\n          ) &gt;= 5\n      `;\n      \n      \/\/ \u30af\u30a8\u30ea\u3092\u5b9f\u884c\n      db.all(pickupQuery, [], async (err, users) =&gt; {\n        if (err) {\n          console.error(&#39;\u30d4\u30c3\u30af\u30a2\u30c3\u30d7\u30e6\u30fc\u30b6\u30fc\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, err);\n          return res.status(500).json({ error: &#39;\u30d4\u30c3\u30af\u30a2\u30c3\u30d7\u30e6\u30fc\u30b6\u30fc\u306e\u53d6\u5f97\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n        }\n        \n        \/\/ \u6761\u4ef6\u306b\u5408\u3046\u30e6\u30fc\u30b6\u30fc\u304c\u3044\u306a\u3044\u5834\u5408\n        if (users.length === 0) {\n          return res.json({ \n            success: false, \n            error: &#39;\u767a\u4fe1\u65705\u4ef6\u4ee5\u4e0a\u306e\u30e6\u30fc\u30b6\u30fc\u304c\u898b\u3064\u304b\u308a\u307e\u305b\u3093\u3067\u3057\u305f&#39; \n          });\n        }\n        \n        \/\/ \u30e9\u30f3\u30c0\u30e0\u306b1\u4eba\u9078\u3076\n        const randomIndex = Math.floor(Math.random() * users.length);\n        const selectedUser = users[randomIndex];\n        \n        \/\/ \u8868\u793a\u540d\u3092\u8a2d\u5b9a\n        selectedUser.displayName = selectedUser.display_name || selectedUser.real_name || selectedUser.name || &#39;Unknown&#39;;\n        \n        \/\/ \u9ad8\u753b\u8cea\u30a2\u30d0\u30bf\u30fcURL\u3092\u78ba\u4fdd\uff08\u30d0\u30c3\u30af\u30a2\u30c3\u30d7\uff09\n        if (selectedUser.avatar && selectedUser.avatar.includes(&#39;_72&#39;)) {\n          selectedUser.avatar_hd = selectedUser.avatar.replace(&#39;_72&#39;, &#39;_512&#39;);\n        } else {\n          selectedUser.avatar_hd = selectedUser.avatar;\n        }\n        \n        \/\/ \u6d3b\u52d5\u30c7\u30fc\u30bf\u304b\u3089\u57fa\u672c\u60c5\u5831\u3092\u4f5c\u6210\n        const messageCount = selectedUser.message_count || 0;\n        const replyCount = selectedUser.reply_count || 0;\n        const reportCount = selectedUser.report_count || 0;\n        const attendanceCount = selectedUser.attendance_count || 0;\n        const officeCount = selectedUser.office_count || 0;\n        \n        const activityInfo = `${selectedUser.displayName}\u3055\u3093\u306f\u3001\u6295\u7a3f\u6570${messageCount}\u4ef6\u3001\u767a\u4fe1\u6570${reportCount}\u4ef6\u3068\u6d3b\u767a\u306b\u6d3b\u52d5\u3055\u308c\u3066\u3044\u307e\u3059\u3002\u51fa\u52e4\u6570\u306f${attendanceCount}\u65e5\u3001\u30ea\u30a2\u30eb\u51fa\u793e\u306f${officeCount}\u65e5\u3067\u3059\u3002\u30c1\u30fc\u30e0\u5185\u3067\u306e\u7a4d\u6975\u7684\u306a\u30b3\u30df\u30e5\u30cb\u30b1\u30fc\u30b7\u30e7\u30f3\u304c\u7279\u5fb4\u7684\u3067\u3059\uff01`;\n        \n        \/\/ \u7c21\u6613\u7684\u306a\u5206\u6790\u7d50\u679c\u3092\u4f5c\u6210\n        selectedUser.analysis = activityInfo;\n        \n        \/\/ \u7d50\u679c\u3092\u8fd4\u3059\n        return res.json({\n          success: true,\n          user: {\n            ...selectedUser,\n            \/\/ \u6570\u5024\u578b\u3092\u78ba\u4fdd\n            message_count: parseInt(selectedUser.message_count || 0),\n            reply_count: parseInt(selectedUser.reply_count || 0),\n            attendance_count: parseInt(selectedUser.attendance_count || 0),\n            office_count: parseInt(selectedUser.office_count || 0),\n            report_count: parseInt(selectedUser.report_count || 0),\n            sent_reaction_count: parseInt(selectedUser.sent_reaction_count || 0),\n            received_reaction_count: parseInt(selectedUser.received_reaction_count || 0)\n          }\n        });\n      });\n    });\n  } catch (error) {\n    console.error(&#39;\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u30c6\u30b9\u30c8\u306b\u5931\u6557\u3057\u307e\u3057\u305f:&#39;, error);\n    res.status(500).json({\n      success: false,\n      error: `\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u30c6\u30b9\u30c8\u306b\u5931\u6557\u3057\u307e\u3057\u305f: ${error.message}`\n    });\n  }\n});\n\n\/\/ reactions.get API\u3092\u30c6\u30b9\u30c8\u3059\u308b\u30a8\u30f3\u30c9\u30dd\u30a4\u30f3\u30c8\napp.get(&#39;\/api\/test-reactions-direct&#39;, async (req, res) =&gt; {\n  try {\n    console.log(&#39;\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u76f4\u63a5\u53d6\u5f97\u30c6\u30b9\u30c8\u3092\u5b9f\u884c\u3057\u307e\u3059...&#39;);\n    \n    \/\/ \u30c6\u30b9\u30c8\u7528\u306e\u30c1\u30e3\u30f3\u30cd\u30ebID\uff08\u6700\u521d\u306e\u30c1\u30e3\u30f3\u30cd\u30eb\u3092\u4f7f\u7528\uff09\n    const testChannelId = slackChannelIds[0];\n    console.log(`\u30c6\u30b9\u30c8\u7528\u30c1\u30e3\u30f3\u30cd\u30ebID: ${testChannelId}`);\n    \n    \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u60c5\u5831\u3092\u53d6\u5f97\u3057\u3066\u30ed\u30b0\u306b\u51fa\u529b\n    const channelInfo = await slackClient.conversations.info({\n      channel: testChannelId\n    });\n    console.log(`\u30c6\u30b9\u30c8\u7528\u30c1\u30e3\u30f3\u30cd\u30eb\u540d: ${channelInfo.channel.name}`);\n    \n    \/\/ \u904e\u53bb3\u30f6\u6708\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u53d6\u5f97\n    const threeMonthsAgo = new Date();\n    threeMonthsAgo.setMonth(threeMonthsAgo.getMonth() - 3);\n    const oldestTimestamp = threeMonthsAgo.getTime() \/ 1000;\n    \n    \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u5c65\u6b74\u3092\u53d6\u5f97\uff08\u6700\u65b0100\u4ef6\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\uff09\n    const result = await slackClient.conversations.history({\n      channel: testChannelId,\n      limit: 100,\n      oldest: oldestTimestamp\n    });\n    \n    console.log(`\u30e1\u30c3\u30bb\u30fc\u30b8\u4ef6\u6570: ${result.messages.length}\u4ef6`);\n    \n    \/\/ \u5404\u30e1\u30c3\u30bb\u30fc\u30b8\u306e\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u60c5\u5831\u3092\u53d6\u5f97\n    let totalReactions = 0;\n    let messagesWithReactions = 0;\n    const reactionResults = [];\n    \n    \/\/ \u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u63a5\u7d9a\n    const db = new sqlite3.Database(&#39;slack_data.db&#39;);\n    \n    try {\n      \/\/ \u30c8\u30e9\u30f3\u30b6\u30af\u30b7\u30e7\u30f3\u958b\u59cb\n      await new Promise((resolve, reject) =&gt; {\n        db.run(&#39;BEGIN TRANSACTION&#39;, err =&gt; {\n          if (err) reject(err);\n          else resolve();\n        });\n      });\n      \n      \/\/ \u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u30c6\u30fc\u30d6\u30eb\u3092\u30af\u30ea\u30a2\n      await new Promise((resolve, reject) =&gt; {\n        db.run(&#39;DELETE FROM slack_reactions&#39;, err =&gt; {\n          if (err) reject(err);\n          else resolve();\n        });\n      });\n      \n      \/\/ \u30d7\u30ea\u30da\u30a2\u30c9\u30b9\u30c6\u30fc\u30c8\u30e1\u30f3\u30c8\u4f5c\u6210\n      const stmt = db.prepare(`\n        INSERT INTO slack_reactions (message_id, user_id, name, timestamp)\n        VALUES (?, ?, ?, ?)\n      `);\n      \n      \/\/ \u6700\u521d\u306e10\u4ef6\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u306b\u3064\u3044\u3066\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u60c5\u5831\u3092\u53d6\u5f97\n      for (const msg of result.messages.slice(0, 10)) {\n        try {\n          console.log(`\u30e1\u30c3\u30bb\u30fc\u30b8\u78ba\u8a8d (ts=${msg.ts}): \u30c6\u30ad\u30b9\u30c8=&quot;${msg.text.substring(0, 30)}...&quot;`);\n          \n          \/\/ reactions.get API\u3092\u4f7f\u7528\u3057\u3066\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u60c5\u5831\u3092\u53d6\u5f97\n          const reactionInfo = await slackClient.reactions.get({\n            channel: testChannelId,\n            timestamp: msg.ts,\n            full: true\n          });\n          \n          console.log(`\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u53d6\u5f97\u7d50\u679c:`, JSON.stringify(reactionInfo, null, 2));\n          \n          if (reactionInfo.message && reactionInfo.message.reactions && reactionInfo.message.reactions.length &gt; 0) {\n            messagesWithReactions++;\n            \n            for (const reaction of reactionInfo.message.reactions) {\n              if (reaction.users && reaction.users.length &gt; 0) {\n                for (const userId of reaction.users) {\n                  \/\/ \u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u3092DB\u306b\u4fdd\u5b58\n                  stmt.run(msg.ts, userId, reaction.name, msg.ts);\n                  totalReactions++;\n                }\n              }\n            }\n            \n            reactionResults.push({\n              ts: msg.ts,\n              text: msg.text.substring(0, 50) + (msg.text.length &gt; 50 ? &#39;...&#39; : &#39;&#39;),\n              reactions: reactionInfo.message.reactions\n            });\n          }\n        } catch (reactionError) {\n          console.error(`\u30e1\u30c3\u30bb\u30fc\u30b8 ${msg.ts} \u306e\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u53d6\u5f97\u30a8\u30e9\u30fc:`, reactionError);\n        }\n        \n        \/\/ API\u5236\u9650\u3092\u8003\u616e\u3057\u3066\u5c11\u3057\u5f85\u6a5f\n        await new Promise(resolve =&gt; setTimeout(resolve, 200));\n      }\n      \n      \/\/ \u30b9\u30c6\u30fc\u30c8\u30e1\u30f3\u30c8\u3092finalize\n      stmt.finalize();\n      \n      \/\/ \u30c8\u30e9\u30f3\u30b6\u30af\u30b7\u30e7\u30f3\u3092\u30b3\u30df\u30c3\u30c8\n      await new Promise((resolve, reject) =&gt; {\n        db.run(&#39;COMMIT&#39;, err =&gt; {\n          if (err) reject(err);\n          else resolve();\n        });\n      });\n      \n      console.log(`${totalReactions}\u4ef6\u306e\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u3092\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u306b\u4fdd\u5b58\u3057\u307e\u3057\u305f`);\n    } catch (dbError) {\n      console.error(&#39;\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u64cd\u4f5c\u30a8\u30e9\u30fc:&#39;, dbError);\n      \n      \/\/ \u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u305f\u5834\u5408\u306f\u30ed\u30fc\u30eb\u30d0\u30c3\u30af\n      await new Promise((resolve) =&gt; {\n        db.run(&#39;ROLLBACK&#39;, () =&gt; resolve());\n      });\n    } finally {\n      \/\/ \u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u3092\u9589\u3058\u308b\n      db.close();\n    }\n    \n    \/\/ \u30ec\u30b9\u30dd\u30f3\u30b9\u3092\u8fd4\u3059\n    res.json({\n      success: true,\n      messageCount: result.messages.length,\n      messagesWithReactions: messagesWithReactions,\n      totalReactions: totalReactions,\n      samples: reactionResults\n    });\n  } catch (error) {\n    console.error(&#39;\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u76f4\u63a5\u53d6\u5f97\u30c6\u30b9\u30c8\u30a8\u30e9\u30fc:&#39;, error);\n    res.status(500).json({\n      success: false,\n      error: `\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u76f4\u63a5\u53d6\u5f97\u30c6\u30b9\u30c8\u306b\u5931\u6557\u3057\u307e\u3057\u305f: ${error.message}`\n    });\n  }\n});\n\n\/\/ \u30a2\u30d7\u30ea\u8a2d\u5b9a\u3092\u53d6\u5f97\u3059\u308b\u95a2\u6570\nfunction getSettings() {\n  try {\n    \/\/ \u30c7\u30d5\u30a9\u30eb\u30c8\u8a2d\u5b9a\n    const defaultSettings = {\n      aiPrompt: &#39;\u4ee5\u4e0b\u306eSlack\u30e1\u30c3\u30bb\u30fc\u30b8\u304b\u3089\u30e6\u30fc\u30b6\u30fc\u306e\u7279\u5fb4\u3092\u5206\u6790\u3057\u3066\u304f\u3060\u3055\u3044\u3002\u7c21\u6f54\u306b200\u6587\u5b57\u7a0b\u5ea6\u3067\u307e\u3068\u3081\u3066\u304f\u3060\u3055\u3044\u3002&#39;,\n      syncPeriod: 90,  \/\/ 90\u65e5\uff083\u30f6\u6708\uff09\n      characterImage: &#39;\/images\/brave-character.png&#39;,  \/\/ \u30c7\u30d5\u30a9\u30eb\u30c8\u306e\u30ad\u30e3\u30e9\u30af\u30bf\u30fc\u753b\u50cf\n      slackBotToken: process.env.SLACK_BOT_TOKEN || null,\n      slackChannelIds: process.env.SLACK_CHANNEL_IDS ? process.env.SLACK_CHANNEL_IDS.split(&#39;,&#39;) : [],\n      anthropicApiKey: process.env.ANTHROPIC_API_KEY || null,\n      openaiApiKey: process.env.OPENAI_API_KEY || null\n    };\n    \n    \/\/ \u8a2d\u5b9a\u30d5\u30a1\u30a4\u30eb\u304b\u3089\u306e\u8aad\u307f\u8fbc\u307f\u3092\u8a66\u307f\u308b\n    try {\n      const settingsFile = fs.readFileSync(&#39;settings.json&#39;, &#39;utf8&#39;);\n      const settings = JSON.parse(settingsFile);\n      console.log(&#39;\u8a2d\u5b9a\u30d5\u30a1\u30a4\u30eb\u304b\u3089\u8a2d\u5b9a\u3092\u8aad\u307f\u8fbc\u307f\u307e\u3057\u305f&#39;);\n      return { ...defaultSettings, ...settings };\n    } catch (readError) {\n      console.log(&#39;\u8a2d\u5b9a\u30d5\u30a1\u30a4\u30eb\u306e\u8aad\u307f\u8fbc\u307f\u306b\u5931\u6557\u3057\u307e\u3057\u305f: &#39; + readError.message);\n      return defaultSettings;\n    }\n  } catch (error) {\n    console.error(&#39;\u8a2d\u5b9a\u30d5\u30a1\u30a4\u30eb\u306e\u8aad\u307f\u8fbc\u307f\u306b\u5931\u6557\u3057\u307e\u3057\u305f:&#39;, error);\n    return {\n      aiPrompt: &#39;\u4ee5\u4e0b\u306eSlack\u30e1\u30c3\u30bb\u30fc\u30b8\u304b\u3089\u30e6\u30fc\u30b6\u30fc\u306e\u7279\u5fb4\u3092\u5206\u6790\u3057\u3066\u304f\u3060\u3055\u3044\u3002\u7c21\u6f54\u306b200\u6587\u5b57\u7a0b\u5ea6\u3067\u307e\u3068\u3081\u3066\u304f\u3060\u3055\u3044\u3002&#39;,\n      syncPeriod: 90,  \/\/ 90\u65e5\uff083\u30f6\u6708\uff09\n      characterImage: &#39;\/images\/brave-character.png&#39;,  \/\/ \u30c7\u30d5\u30a9\u30eb\u30c8\u306e\u30ad\u30e3\u30e9\u30af\u30bf\u30fc\u753b\u50cf\n      slackBotToken: process.env.SLACK_BOT_TOKEN || null,\n      slackChannelIds: process.env.SLACK_CHANNEL_IDS ? process.env.SLACK_CHANNEL_IDS.split(&#39;,&#39;) : [],\n      anthropicApiKey: process.env.ANTHROPIC_API_KEY || null,\n      openaiApiKey: process.env.OPENAI_API_KEY || null\n    };\n  }\n}\n\n\/\/ \u300c\u4eca\u65e5\u306e\u518d\u540c\u671f\u300d\u7528API\napp.get(&#39;\/api\/clean-sync-slack-data&#39;, async (req, res) =&gt; {\n  try {\n    console.log(&#39;\u30af\u30ea\u30fc\u30f3\u30a2\u30c3\u30d7\u304a\u3088\u3073\u518d\u540c\u671f\u3092\u958b\u59cb\u3057\u307e\u3059...&#39;);\n    \n    \/\/ \u6700\u7d42\u540c\u671f\u65e5\u6642\u3092\u5148\u306b\u66f4\u65b0\uff08\u540c\u6642\u5b9f\u884c\u9632\u6b62\u306e\u305f\u3081\uff09\n    try {\n      \/\/ \u65e5\u672c\u6642\u9593\u306e\u73fe\u5728\u65e5\u6642\u3092\u53d6\u5f97\n      const jstNow = getJSTDate();\n      const timestamp = jstNow.getTime();\n      const datetimeStr = jstNow.toISOString().replace(&#39;T&#39;, &#39; &#39;).replace(&#39;Z&#39;, &#39;&#39;);\n      \n      console.log(`\u540c\u671f\u958b\u59cb\u524d\u306b\u6700\u7d42\u540c\u671f\u65e5\u6642\u3092\u66f4\u65b0\u3057\u307e\u3059 (JST): ${datetimeStr}, ${timestamp}`);\n      \n      \/\/ \u65e2\u5b58\u306e\u6700\u7d42\u540c\u671f\u30ec\u30b3\u30fc\u30c9\u3092\u53d6\u5f97\n      const lastSyncData = await new Promise((resolve, reject) =&gt; {\n        db.get(&#39;SELECT * FROM last_sync_date ORDER BY id DESC LIMIT 1&#39;, [], (err, row) =&gt; {\n          if (err) {\n            console.error(&#39;\u6700\u7d42\u540c\u671f\u30c7\u30fc\u30bf\u306e\u78ba\u8a8d\u306b\u5931\u6557:&#39;, err);\n            reject(err);\n            return;\n          }\n          resolve(row);\n        });\n      });\n      \n      if (lastSyncData) {\n        \/\/ \u65e2\u5b58\u306e\u30ec\u30b3\u30fc\u30c9\u304c\u3042\u308b\u5834\u5408\u306f\u66f4\u65b0\n        await new Promise((resolve, reject) =&gt; {\n          db.run(\n            &#39;UPDATE last_sync_date SET last_sync = ?, last_sync_timestamp = ? WHERE id = ?&#39;, \n            [datetimeStr, timestamp, lastSyncData.id], \n            function(err) {\n              if (err) {\n                console.error(&#39;\u6700\u7d42\u540c\u671f\u65e5\u6642\u306e\u66f4\u65b0\u306b\u5931\u6557:&#39;, err);\n                reject(err);\n              } else {\n                console.log(`\u6700\u7d42\u540c\u671f\u65e5\u6642\u3092\u66f4\u65b0\u3057\u307e\u3057\u305f (JST): ${datetimeStr}, ${timestamp}`);\n                resolve();\n              }\n            }\n          );\n        });\n      } else {\n        \/\/ \u65e2\u5b58\u306e\u30ec\u30b3\u30fc\u30c9\u304c\u306a\u3044\u5834\u5408\u306f\u65b0\u898f\u4f5c\u6210\n        await new Promise((resolve, reject) =&gt; {\n          db.run(\n            &#39;INSERT INTO last_sync_date (last_sync, last_sync_timestamp) VALUES (?, ?)&#39;, \n            [datetimeStr, timestamp], \n            function(err) {\n              if (err) {\n                console.error(&#39;\u6700\u7d42\u540c\u671f\u65e5\u6642\u306e\u65b0\u898f\u4f5c\u6210\u306b\u5931\u6557:&#39;, err);\n                reject(err);\n              } else {\n                console.log(`\u6700\u7d42\u540c\u671f\u65e5\u6642\u3092\u65b0\u898f\u4f5c\u6210\u3057\u307e\u3057\u305f (JST): ${datetimeStr}, ${timestamp}`);\n                resolve();\n              }\n            }\n          );\n        });\n      }\n    } catch (syncDateError) {\n      console.error(&#39;\u6700\u7d42\u540c\u671f\u65e5\u6642\u306e\u66f4\u65b0\u306b\u5931\u6557:&#39;, syncDateError);\n      \/\/ \u540c\u671f\u65e5\u6642\u306e\u66f4\u65b0\u306b\u5931\u6557\u3057\u3066\u3082\u51e6\u7406\u306f\u7d9a\u884c\n    }\n    \n    \/\/ \u30af\u30ea\u30fc\u30f3\u30a2\u30c3\u30d7\u51e6\u7406\n    try {\n      \/\/ \u4eca\u65e5\u306e\u65e5\u4ed8\u3092\u53d6\u5f97\uff08\u65e5\u672c\u6642\u9593\uff09\n      const today = getJSTDateString();\n      \n      \/\/ \u4eca\u65e5\u306e\u30a4\u30c1\u63a8\u3057braver\u30c7\u30fc\u30bf\u3092\u524a\u9664\n      await new Promise((resolve, reject) =&gt; {\n        db.run(&#39;DELETE FROM daily_pickup_user WHERE pickup_date = ?&#39;, [today], function(err) {\n          if (err) {\n            console.error(&#39;\u30a4\u30c1\u63a8\u3057braver\u30ea\u30bb\u30c3\u30c8\u30a8\u30e9\u30fc:&#39;, err);\n            reject(err);\n          } else {\n            console.log(`\u4eca\u65e5\u306e\u30a4\u30c1\u63a8\u3057braver\u3092\u30ea\u30bb\u30c3\u30c8\u3057\u307e\u3057\u305f: ${this.changes}\u4ef6\u306e\u30c7\u30fc\u30bf\u3092\u524a\u9664`);\n            resolve();\n          }\n        });\n      });\n      \n      \/\/ 1\u9031\u9593\u524d\u306e\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7\u3092\u8a08\u7b97\uff08\u65e5\u672c\u6642\u9593\u30d9\u30fc\u30b9\uff09\n      const oneWeekAgo = getJSTDate();\n      oneWeekAgo.setDate(oneWeekAgo.getDate() - 7);\n      const oneWeekAgoTimestamp = oneWeekAgo.getTime() \/ 1000;\n      \n      console.log(`\u76f4\u8fd11\u9031\u9593\u5206(${oneWeekAgo.toLocaleString(&#39;ja-JP&#39;)}\u4ee5\u964d)\u306eSlack\u30c7\u30fc\u30bf\u3092\u524a\u9664\u3057\u307e\u3059...`);\n      \n      \/\/ \u524a\u9664\u51e6\u7406\u3092Promise\u3068\u3057\u3066\u5b9f\u884c\n      const deleteMessages = new Promise((resolve, reject) =&gt; {\n        db.run(&#39;DELETE FROM slack_messages WHERE timestamp &gt;= ?&#39;, [oneWeekAgoTimestamp], function(err) {\n          if (err) {\n            console.error(&#39;\u76f4\u8fd11\u9031\u9593\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u524a\u9664\u30a8\u30e9\u30fc:&#39;, err);\n            reject(err);\n          } else {\n            console.log(`\u76f4\u8fd11\u9031\u9593\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u524a\u9664\u3057\u307e\u3057\u305f: ${this.changes}\u4ef6\u306e\u30c7\u30fc\u30bf\u3092\u524a\u9664`);\n            resolve();\n          }\n        });\n      });\n      \n      const deleteThreadReplies = new Promise((resolve, reject) =&gt; {\n        db.run(&#39;DELETE FROM slack_thread_replies WHERE timestamp &gt;= ?&#39;, [oneWeekAgoTimestamp], function(err) {\n          if (err) {\n            console.error(&#39;\u76f4\u8fd11\u9031\u9593\u306e\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u524a\u9664\u30a8\u30e9\u30fc:&#39;, err);\n            reject(err);\n          } else {\n            console.log(`\u76f4\u8fd11\u9031\u9593\u306e\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u3092\u524a\u9664\u3057\u307e\u3057\u305f: ${this.changes}\u4ef6\u306e\u30c7\u30fc\u30bf\u3092\u524a\u9664`);\n            resolve();\n          }\n        });\n      });\n      \n      const deleteReactions = new Promise((resolve, reject) =&gt; {\n        db.run(&#39;DELETE FROM slack_reactions WHERE timestamp &gt;= ?&#39;, [oneWeekAgoTimestamp], function(err) {\n          if (err) {\n            console.error(&#39;\u76f4\u8fd11\u9031\u9593\u306e\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u524a\u9664\u30a8\u30e9\u30fc:&#39;, err);\n            reject(err);\n          } else {\n            console.log(`\u76f4\u8fd11\u9031\u9593\u306e\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u3092\u524a\u9664\u3057\u307e\u3057\u305f: ${this.changes}\u4ef6\u306e\u30c7\u30fc\u30bf\u3092\u524a\u9664`);\n            resolve();\n          }\n        });\n      });\n      \n      \/\/ \u3059\u3079\u3066\u306e\u524a\u9664\u51e6\u7406\u3092\u5f85\u3064\n      await Promise.all([deleteMessages, deleteThreadReplies, deleteReactions]);\n      console.log(&#39;\u3059\u3079\u3066\u306e\u30c7\u30fc\u30bf\u524a\u9664\u51e6\u7406\u304c\u5b8c\u4e86\u3057\u307e\u3057\u305f&#39;);\n      \n      \/\/ \u5168\u30c1\u30e3\u30f3\u30cd\u30eb\u306e\u540c\u671f\u3092\u5b9f\u884c\n      if (!slackToken || !slackChannelIds.length) {\n        return res.status(500).json({ \n          error: &#39;Slack API\u306e\u8a2d\u5b9a\u304c\u4e0d\u8db3\u3057\u3066\u3044\u307e\u3059\u3002\u74b0\u5883\u5909\u6570\u3092\u78ba\u8a8d\u3057\u3066\u304f\u3060\u3055\u3044\u3002&#39; \n        });\n      }\n      \n      const channelsToSync = global.slackChannelIds || slackChannelIds;\n      console.log(`\u8a2d\u5b9a\u3055\u308c\u3066\u3044\u308b\u5168\u30c1\u30e3\u30f3\u30cd\u30eb (${channelsToSync.length}\u4ef6) \u3092\u540c\u671f\u3057\u307e\u3059: ${channelsToSync.join(&#39;, &#39;)}`);\n      \n      \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u3054\u3068\u306e\u540c\u671f\u7d50\u679c\u3092\u4fdd\u5b58\u3059\u308b\u914d\u5217\n      const syncResults = [];\n      const errorChannels = [];\n      \n      \/\/ API\u30ec\u30fc\u30c8\u5236\u9650\u3092\u56de\u907f\u3059\u308b\u305f\u3081\u306e\u30c7\u30a3\u30ec\u30a4\u95a2\u6570\n      const delay = ms =&gt; new Promise(resolve =&gt; setTimeout(resolve, ms));\n      \n      \/\/ \u5404\u30c1\u30e3\u30f3\u30cd\u30eb\u3092\u9806\u756a\u306b\u51e6\u7406\uff08\u30ec\u30fc\u30c8\u5236\u9650\u3092\u56de\u907f\u3059\u308b\u305f\u3081\u9806\u6b21\u51e6\u7406\uff09\n      for (const channelId of channelsToSync) {\n        try {\n          console.log(`\u30c1\u30e3\u30f3\u30cd\u30eb ${channelId} \u306e\u540c\u671f\u3092\u958b\u59cb\u3057\u307e\u3059...`);\n          \n          \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u60c5\u5831\u3092\u53d6\u5f97 - \u30ec\u30fc\u30c8\u5236\u9650\u5bfe\u7b56\n          let channelInfo;\n          try {\n            channelInfo = await slackClient.conversations.info({\n              channel: channelId\n            });\n          } catch (error) {\n            if (error.code === &#39;rate_limited&#39;) {\n              console.log(`\u30ec\u30fc\u30c8\u5236\u9650\u306b\u3088\u308a\u5f85\u6a5f\u3057\u307e\u3059: ${error.retryAfter || 5}\u79d2`);\n              await delay((error.retryAfter || 5) * 1000);\n              channelInfo = await slackClient.conversations.info({\n                channel: channelId\n              });\n            } else {\n              throw error;\n            }\n          }\n          \n          const channelName = channelInfo.channel.name || &#39;Unknown Channel&#39;;\n          console.log(`\u30c1\u30e3\u30f3\u30cd\u30eb\u540d: ${channelName}`);\n          \n          \/\/ API\u30ea\u30af\u30a8\u30b9\u30c8\u9593\u306b\u30c7\u30a3\u30ec\u30a4\u3092\u633f\u5165\u3057\u3066\u30ec\u30fc\u30c8\u5236\u9650\u3092\u56de\u907f\n          await delay(1000);\n          \n          \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u5c65\u6b74\u3092\u53d6\u5f97 - \u30ec\u30fc\u30c8\u5236\u9650\u5bfe\u7b56\n          let result;\n          try {\n            result = await slackClient.conversations.history({\n              channel: channelId,\n              limit: 100\n            });\n          } catch (error) {\n            if (error.code === &#39;rate_limited&#39;) {\n              console.log(`\u30ec\u30fc\u30c8\u5236\u9650\u306b\u3088\u308a\u5f85\u6a5f\u3057\u307e\u3059: ${error.retryAfter || 5}\u79d2`);\n              await delay((error.retryAfter || 5) * 1000);\n              result = await slackClient.conversations.history({\n                channel: channelId,\n                limit: 100\n              });\n            } else {\n              throw error;\n            }\n          }\n          \n          console.log(`\u30c1\u30e3\u30f3\u30cd\u30eb ${channelName} \u304b\u3089 ${result.messages.length} \u4ef6\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u53d6\u5f97`);\n          \n          \/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u4fdd\u5b58\n          for (const message of result.messages) {\n            \/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u306e\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7\u3092\u30c1\u30a7\u30c3\u30af\n            if (message.ts && parseFloat(message.ts) &gt;= oneWeekAgoTimestamp) {\n              \/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u4fdd\u5b58\n              const messageType = message.subtype || &#39;message&#39;;\n              const userId = message.user || &#39;unknown_user&#39;;\n              const text = message.text || &#39;&#39;;\n              const timestamp = message.ts;\n              \n              \/\/ DB\u306b\u4fdd\u5b58\n              db.run(\n                &#39;INSERT OR REPLACE INTO slack_messages (message_id, channel_id, channel_name, user_id, text, message_type, timestamp, created_at) VALUES (?, ?, ?, ?, ?, ?, ?, ?)&#39;,\n                [\n                  timestamp,\n                  channelId,\n                  channelName,\n                  userId,\n                  text,\n                  messageType,\n                  timestamp,\n                  Math.floor(Date.now() \/ 1000)\n                ],\n                function(err) {\n                  if (err) {\n                    console.error(`\u30e1\u30c3\u30bb\u30fc\u30b8\u4fdd\u5b58\u30a8\u30e9\u30fc: ${err.message}`);\n                  }\n                }\n              );\n              \n              \/\/ \u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u304c\u3042\u308c\u3070\u53d6\u5f97\n              if (message.thread_ts && message.reply_count &gt; 0) {\n                console.log(`\u30b9\u30ec\u30c3\u30c9\u691c\u51fa: message_id=${message.ts}, thread_ts=${message.thread_ts}, reply_count=${message.reply_count}`);\n                console.log(`\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u3092\u53d6\u5f97\u3057\u307e\u3059: thread_ts=${message.thread_ts}, reply_count=${message.reply_count}`);\n                \n                \/\/ API\u30ea\u30af\u30a8\u30b9\u30c8\u9593\u306b\u30c7\u30a3\u30ec\u30a4\u3092\u633f\u5165\u3057\u3066\u30ec\u30fc\u30c8\u5236\u9650\u3092\u56de\u907f\n                await delay(1000);\n                \n                \/\/ \u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u3092\u53d6\u5f97 - \u30ec\u30fc\u30c8\u5236\u9650\u5bfe\u7b56\n                let repliesResult;\n                try {\n                  repliesResult = await slackClient.conversations.replies({\n                    channel: channelId,\n                    ts: message.thread_ts,\n                    limit: 100\n                  });\n                } catch (error) {\n                  if (error.code === &#39;rate_limited&#39;) {\n                    console.log(`\u30ec\u30fc\u30c8\u5236\u9650\u306b\u3088\u308a\u5f85\u6a5f\u3057\u307e\u3059: ${error.retryAfter || 5}\u79d2`);\n                    await delay((error.retryAfter || 5) * 1000);\n                    repliesResult = await slackClient.conversations.replies({\n                      channel: channelId,\n                      ts: message.thread_ts,\n                      limit: 100\n                    });\n                  } else {\n                    console.error(`\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u53d6\u5f97\u30a8\u30e9\u30fc: ${error.message}`);\n                    continue; \/\/ \u3053\u306e\u30b9\u30ec\u30c3\u30c9\u3092\u30b9\u30ad\u30c3\u30d7\u3057\u3066\u6b21\u3078\n                  }\n                }\n                \n                console.log(`\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u53d6\u5f97\u7d50\u679c: ${repliesResult.messages.length}\u4ef6`);\n                \n                \/\/ \u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u3092\u4fdd\u5b58\n                for (const reply of repliesResult.messages) {\n                  \/\/ \u89aa\u30e1\u30c3\u30bb\u30fc\u30b8\u306f\u9664\u5916\n                  if (reply.ts === message.thread_ts) continue;\n                  \n                  const replyUserId = reply.user || &#39;unknown_user&#39;;\n                  const replyText = reply.text || &#39;&#39;;\n                  const replyTimestamp = reply.ts;\n                  \n                  \/\/ DB\u306b\u4fdd\u5b58\n                  db.run(\n                    &#39;INSERT OR REPLACE INTO slack_thread_replies (reply_id, thread_ts, message_id, channel_id, channel_name, user_id, text, timestamp, created_at) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)&#39;,\n                    [\n                      replyTimestamp,\n                      message.thread_ts,\n                      replyTimestamp,\n                      channelId,\n                      channelName,\n                      replyUserId,\n                      replyText,\n                      replyTimestamp,\n                      Math.floor(Date.now() \/ 1000)\n                    ],\n                    function(err) {\n                      if (err) {\n                        console.error(`\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u4fdd\u5b58\u30a8\u30e9\u30fc: ${err.message}`);\n                      }\n                    }\n                  );\n                }\n                \n                \/\/ API\u547c\u3073\u51fa\u3057\u9593\u306b\u77ed\u3044\u30c7\u30a3\u30ec\u30a4\u3092\u5165\u308c\u308b\n                await delay(500);\n              }\n              \n              \/\/ \u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u3092\u51e6\u7406\n              if (message.reactions && message.reactions.length &gt; 0) {\n                console.log(`\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u691c\u51fa: message_id=${message.ts}, reaction_count=${message.reactions.length}`);\n                console.log(`\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u8a73\u7d30: ${JSON.stringify(message.reactions, null, 2)}`);\n                \n                for (const reaction of message.reactions) {\n                  const emojiName = reaction.name;\n                  const userIds = reaction.users || [];\n                  \n                  console.log(`\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u300c${emojiName}\u300d\u306e\u30e6\u30fc\u30b6\u30fc\u6570: ${userIds.length}`);\n                  \n                  \/\/ \u30e6\u30fc\u30b6\u30fc\u3054\u3068\u306b\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u3092\u4fdd\u5b58\n                  for (const reactUserId of userIds) {\n                    db.run(\n                      &#39;INSERT OR REPLACE INTO slack_reactions (message_id, channel_id, user_id, name, timestamp, created_at) VALUES (?, ?, ?, ?, ?, ?)&#39;,\n                      [\n                        message.ts,\n                        channelId,\n                        reactUserId,\n                        emojiName,\n                        message.ts,\n                        Math.floor(Date.now() \/ 1000)\n                      ],\n                      function(err) {\n                        if (err) {\n                          console.error(`\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u4fdd\u5b58\u30a8\u30e9\u30fc: ${err.message}`);\n                        } else {\n                          console.log(`\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u4fdd\u5b58\u6210\u529f: message_id=${message.ts}, user=${reactUserId}, emoji=${emojiName}`);\n                        }\n                      }\n                    );\n                  }\n                }\n              } else if (message.reactions === undefined) {\n                console.log(`\u30e1\u30c3\u30bb\u30fc\u30b8ID: ${message.ts} \u306b\u306f\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u304c\u3042\u308a\u307e\u305b\u3093`);\n              }\n            }\n            \n            \/\/ API\u547c\u3073\u51fa\u3057\u9593\u306b\u77ed\u3044\u30c7\u30a3\u30ec\u30a4\u3092\u5165\u308c\u308b\n            await delay(500);\n          }\n          \n          syncResults.push({\n            channelId: channelId,\n            channelName: channelName,\n            messageCount: result.messages.length,\n            success: true\n          });\n          \n          \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u3054\u3068\u306b\u5c11\u3057\u9577\u3081\u306e\u30c7\u30a3\u30ec\u30a4\u3092\u5165\u308c\u308b\n          await delay(2000);\n          \n        } catch (channelError) {\n          console.error(`\u30c1\u30e3\u30f3\u30cd\u30eb ${channelId} \u306e\u540c\u671f\u30a8\u30e9\u30fc:`, channelError);\n          errorChannels.push(channelId);\n          syncResults.push({\n            channelId: channelId,\n            error: channelError.message,\n            success: false\n          });\n          \n          \/\/ \u30a8\u30e9\u30fc\u5f8c\u3082\u5c11\u3057\u5f85\u6a5f\n          await delay(3000);\n        }\n      }\n      \n      \/\/ \u51e6\u7406\u7d50\u679c\u306e\u6982\u8981\u3092\u51fa\u529b\n      const allChannelsProcessed = syncResults.length;\n      const successChannels = syncResults.filter(r =&gt; r.success).length;\n      \n      console.log(`\u5168\u30c1\u30e3\u30f3\u30cd\u30eb\u306e\u540c\u671f\u5b8c\u4e86: \u6210\u529f=${successChannels}\u4ef6, \u5931\u6557=${errorChannels.length}\u4ef6`);\n      \n      \/\/ \u540c\u671f\u5b8c\u4e86\u5f8c\u306b\u30a4\u30c1\u63a8\u3057braver\u3092\u518d\u9078\u51fa\n      try {\n        console.log(&#39;\u540c\u671f\u5b8c\u4e86\u5f8c\u306b\u30a4\u30c1\u63a8\u3057braver\u3092\u518d\u9078\u51fa\u3057\u307e\u3059&#39;);\n        await resetAndSelectDailyPickupUser();\n        console.log(&#39;\u30a4\u30c1\u63a8\u3057braver\u306e\u518d\u9078\u51fa\u304c\u5b8c\u4e86\u3057\u307e\u3057\u305f&#39;);\n      } catch (pickupError) {\n        console.error(&#39;\u30a4\u30c1\u63a8\u3057braver\u518d\u9078\u51fa\u30a8\u30e9\u30fc:&#39;, pickupError);\n      }\n      \n      return res.json({\n        success: true,\n        channelCount: allChannelsProcessed,\n        errorChannels: errorChannels\n      });\n    } catch (syncError) {\n      console.error(`\u540c\u671f\u51e6\u7406\u30a8\u30e9\u30fc: ${syncError.message}`);\n      return res.status(500).json({ error: `\u540c\u671f\u51e6\u7406\u30a8\u30e9\u30fc: ${syncError.message}` });\n    }\n  } catch (error) {\n    console.error(`\u30af\u30ea\u30fc\u30f3\u30a2\u30c3\u30d7\u540c\u671f\u30a8\u30e9\u30fc: ${error.message}`);\n    return res.status(500).json({ error: `\u30af\u30ea\u30fc\u30f3\u30a2\u30c3\u30d7\u540c\u671f\u30a8\u30e9\u30fc: ${error.message}` });\n  }\n});\n\n\/\/ Slack\u30c7\u30fc\u30bf\u3092\u5dee\u5206\u540c\u671f\u3059\u308bAPI\napp.get(&#39;\/api\/sync-diff-slack-data&#39;, async (req, res) =&gt; {\n  try {\n    console.log(&#39;\u30c7\u30fc\u30bf\u540c\u671f\u30ea\u30af\u30a8\u30b9\u30c8\u3092\u53d7\u4fe1\u3057\u307e\u3057\u305f&#39;);\n    \n    if (!slackToken) {\n      console.error(&#39;Slack\u30c8\u30fc\u30af\u30f3\u304c\u8a2d\u5b9a\u3055\u308c\u3066\u3044\u307e\u305b\u3093&#39;);\n      return res.status(500).json({ \n        error: &#39;Slack API\u306e\u8a2d\u5b9a\u304c\u4e0d\u8db3\u3057\u3066\u3044\u307e\u3059\u3002\u74b0\u5883\u5909\u6570\u3092\u78ba\u8a8d\u3057\u3066\u304f\u3060\u3055\u3044\u3002&#39; \n      });\n    }\n\n    if (!slackChannelIds || !slackChannelIds.length) {\n      console.error(&#39;Slack\u30c1\u30e3\u30f3\u30cd\u30ebID\u304c\u8a2d\u5b9a\u3055\u308c\u3066\u3044\u307e\u305b\u3093&#39;);\n      return res.status(500).json({ \n        error: &#39;Slack\u30c1\u30e3\u30f3\u30cd\u30ebID\u304c\u8a2d\u5b9a\u3055\u308c\u3066\u3044\u307e\u305b\u3093\u3002\u74b0\u5883\u5909\u6570\u3092\u78ba\u8a8d\u3057\u3066\u304f\u3060\u3055\u3044\u3002&#39; \n      });\n    }\n\n    console.log(&#39;Slack\u30c7\u30fc\u30bf\u306e\u5dee\u5206\u540c\u671f\u3092\u958b\u59cb...&#39;);\n    console.log(`\u8a2d\u5b9a\u3055\u308c\u3066\u3044\u308b\u30c1\u30e3\u30f3\u30cd\u30eb\u6570: ${slackChannelIds.length}`);\n    console.log(`\u30c1\u30e3\u30f3\u30cd\u30ebID: ${slackChannelIds.join(&#39;, &#39;)}`);\n    \n    \/\/ \u540c\u671f\u3059\u308b\u30c1\u30e3\u30f3\u30cd\u30eb\u306e\u6307\u5b9a\n    const requestedChannelId = req.query.channelId;\n    let channelsToSync = [];\n    \n    if (requestedChannelId) {\n      \/\/ \u7279\u5b9a\u306e\u30c1\u30e3\u30f3\u30cd\u30eb\u304c\u6307\u5b9a\u3055\u308c\u305f\u5834\u5408\n      channelsToSync = [requestedChannelId];\n      console.log(`\u6307\u5b9a\u3055\u308c\u305f\u30c1\u30e3\u30f3\u30cd\u30eb ${requestedChannelId} \u306e\u307f\u3092\u540c\u671f\u3057\u307e\u3059`);\n    } else {\n      \/\/ \u3059\u3079\u3066\u306e\u30c1\u30e3\u30f3\u30cd\u30eb\u3092\u540c\u671f\n      channelsToSync = global.slackChannelIds || slackChannelIds;\n      console.log(`\u8a2d\u5b9a\u3055\u308c\u3066\u3044\u308b\u5168\u30c1\u30e3\u30f3\u30cd\u30eb (${channelsToSync.length}\u4ef6) \u3092\u540c\u671f\u3057\u307e\u3059: ${channelsToSync.join(&#39;, &#39;)}`);\n    }\n    \n    \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u3054\u3068\u306e\u540c\u671f\u7d50\u679c\u3092\u4fdd\u5b58\u3059\u308b\u914d\u5217\n    const syncResults = [];\n    \n    \/\/ \u5404\u30c1\u30e3\u30f3\u30cd\u30eb\u3092\u9806\u756a\u306b\u51e6\u7406\n    for (const channelId of channelsToSync) {\n      try {\n        console.log(`\u30c1\u30e3\u30f3\u30cd\u30eb ${channelId} \u306e\u540c\u671f\u3092\u958b\u59cb\u3057\u307e\u3059...`);\n        \n        \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u60c5\u5831\u306e\u53d6\u5f97\u3092\u8a66\u307f\u308b\n        const channelInfo = await slackClient.conversations.info({\n          channel: channelId\n        }).catch(error =&gt; {\n          console.error(`\u30c1\u30e3\u30f3\u30cd\u30eb ${channelId} \u306e\u60c5\u5831\u53d6\u5f97\u306b\u5931\u6557:`, error.message);\n          throw error;\n        });\n        \n        console.log(`\u30c1\u30e3\u30f3\u30cd\u30eb ${channelId} (${channelInfo.channel.name}) \u306e\u540c\u671f\u3092\u7d9a\u884c\u3057\u307e\u3059`);\n        \n        \/\/ \u4ee5\u4e0b\u3001\u65e2\u5b58\u306e\u540c\u671f\u51e6\u7406\u3092\u7d9a\u884c\n        \/\/ ... existing code ...\n      } catch (channelSyncError) {\n        console.error(`\u30c1\u30e3\u30f3\u30cd\u30eb ${channelId} \u306e\u540c\u671f\u30a8\u30e9\u30fc:`, channelSyncError);\n        syncResults.push({\n          channelId: channelId,\n          error: channelSyncError.message,\n          success: false\n        });\n      }\n    }\n    \n    \/\/ \u540c\u671f\u5b8c\u4e86\u5f8c\u306b\u30a4\u30c1\u63a8\u3057braver\u3092\u518d\u9078\u51fa\n    try {\n      console.log(&#39;\u540c\u671f\u5b8c\u4e86\u5f8c\u306b\u30a4\u30c1\u63a8\u3057braver\u3092\u518d\u9078\u51fa\u3057\u307e\u3059&#39;);\n      await resetAndSelectDailyPickupUser();\n      console.log(&#39;\u30a4\u30c1\u63a8\u3057braver\u306e\u518d\u9078\u51fa\u304c\u5b8c\u4e86\u3057\u307e\u3057\u305f&#39;);\n    } catch (pickupError) {\n      console.error(&#39;\u30a4\u30c1\u63a8\u3057braver\u518d\u9078\u51fa\u30a8\u30e9\u30fc:&#39;, pickupError);\n    }\n    \n    return res.json({\n      success: true,\n      syncedChannels: syncResults,\n      channelCount: syncResults.length,\n      period: `${getJSTDate().toLocaleDateString(&#39;ja-JP&#39;)} \u307e\u3067\u306e3\u30f6\u6708\u5206`,\n      message: &quot;\u30b9\u30bf\u30f3\u30d7\u60c5\u5831\u3082\u542b\u3081\u3066\u5168\u30c7\u30fc\u30bf\u3092\u540c\u671f\u3057\u307e\u3057\u305f&quot;\n    });\n  } catch (error) {\n    console.error(&#39;Slack\u30c7\u30fc\u30bf\u5dee\u5206\u540c\u671f\u30a8\u30e9\u30fc:&#39;, error);\n    return res.status(500).json({ error: `\u540c\u671f\u51e6\u7406\u4e2d\u306b\u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u307e\u3057\u305f: ${error.message}` });\n  }\n});\n\n\/\/ \u671d\u306e\u81ea\u52d5\u540c\u671f\u7528API\napp.get(&#39;\/api\/morning-sync&#39;, async (req, res) =&gt; {\n  try {\n    console.log(&#39;\u671d\u306e\u81ea\u52d5\u540c\u671f\u3092\u958b\u59cb\u3057\u307e\u3059...&#39;);\n    \n    \/\/ \u74b0\u5883\u5909\u6570\u306e\u78ba\u8a8d\n    if (!slackToken) {\n      console.error(&#39;Slack\u30c8\u30fc\u30af\u30f3\u304c\u8a2d\u5b9a\u3055\u308c\u3066\u3044\u307e\u305b\u3093&#39;);\n      return res.status(500).json({ \n        error: &#39;Slack API\u306e\u8a2d\u5b9a\u304c\u4e0d\u8db3\u3057\u3066\u3044\u307e\u3059\u3002\u74b0\u5883\u5909\u6570\u3092\u78ba\u8a8d\u3057\u3066\u304f\u3060\u3055\u3044\u3002&#39; \n      });\n    }\n\n    if (!slackChannelIds || !slackChannelIds.length) {\n      console.error(&#39;Slack\u30c1\u30e3\u30f3\u30cd\u30ebID\u304c\u8a2d\u5b9a\u3055\u308c\u3066\u3044\u307e\u305b\u3093&#39;);\n      return res.status(500).json({ \n        error: &#39;Slack\u30c1\u30e3\u30f3\u30cd\u30ebID\u304c\u8a2d\u5b9a\u3055\u308c\u3066\u3044\u307e\u305b\u3093\u3002\u74b0\u5883\u5909\u6570\u3092\u78ba\u8a8d\u3057\u3066\u304f\u3060\u3055\u3044\u3002&#39; \n      });\n    }\n\n    \/\/ \u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u306e\u63a5\u7d9a\u78ba\u8a8d\n    db.get(&quot;SELECT 1&quot;, (err) =&gt; {\n      if (err) {\n        console.error(&#39;\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u63a5\u7d9a\u30a8\u30e9\u30fc:&#39;, err);\n        return res.status(500).json({ \n          error: &#39;\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u63a5\u7d9a\u306b\u5931\u6557\u3057\u307e\u3057\u305f\u3002&#39; \n        });\n      }\n    });\n\n    \/\/ \u6700\u7d42\u540c\u671f\u65e5\u6642\u3092\u78ba\u8a8d\u3057\u3066\u300124\u6642\u9593\u4ee5\u4e0a\u7d4c\u904e\u3057\u3066\u3044\u308c\u3070\u540c\u671f\u3092\u5b9f\u884c\n    try {\n      \/\/ \u73fe\u5728\u306e\u6642\u523b\uff08\u65e5\u672c\u6642\u9593\uff09\n      const currentTime = getJSTDate().getTime();\n      \n      \/\/ \u6700\u7d42\u540c\u671f\u65e5\u6642\u3092\u53d6\u5f97\n      const lastSyncData = await new Promise((resolve, reject) =&gt; {\n        db.get(&#39;SELECT * FROM last_sync_date ORDER BY id DESC LIMIT 1&#39;, [], (err, row) =&gt; {\n          if (err) {\n            console.error(&#39;\u6700\u7d42\u540c\u671f\u30c7\u30fc\u30bf\u306e\u78ba\u8a8d\u306b\u5931\u6557:&#39;, err);\n            reject(err);\n            return;\n          }\n          resolve(row);\n        });\n      });\n      \n      \/\/ \u540c\u671f\u304c\u5fc5\u8981\u304b\u3069\u3046\u304b\u3092\u5224\u65ad\n      let needSync = false;\n      let reason = &quot;&quot;;\n      \n      if (!lastSyncData) {\n        console.log(&#39;\u521d\u56de\u30a2\u30af\u30bb\u30b9: \u540c\u671f\u30c7\u30fc\u30bf\u304c\u3042\u308a\u307e\u305b\u3093&#39;);\n        needSync = true;\n        reason = &quot;\u521d\u56de\u30a2\u30af\u30bb\u30b9\u306e\u305f\u3081&quot;;\n      } else {\n        \/\/ \u6700\u7d42\u540c\u671f\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7\u304b\u308924\u6642\u9593\u4ee5\u4e0a\u7d4c\u904e\u3057\u3066\u3044\u308b\u304b\u78ba\u8a8d\n        const lastSyncTimestamp = Number(lastSyncData.last_sync_timestamp);\n        if (isNaN(lastSyncTimestamp)) {\n          console.log(&#39;\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7\u304cNaN\u3067\u3059:&#39;, lastSyncData.last_sync_timestamp);\n          needSync = true;\n          reason = &quot;\u6709\u52b9\u306a\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7\u304c\u306a\u3044\u305f\u3081&quot;;\n        } else {\n          const timeDiff = currentTime - lastSyncTimestamp;\n          const hoursDiff = timeDiff \/ (1000 * 60 * 60); \/\/ \u30df\u30ea\u79d2 \u2192 \u6642\u9593\n          \n          console.log(`\u524d\u56de\u306e\u540c\u671f\u304b\u3089${hoursDiff.toFixed(2)}\u6642\u9593\u7d4c\u904e\u3057\u3066\u3044\u307e\u3059`);\n          console.log(`\u6700\u7d42\u540c\u671f\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7: ${lastSyncTimestamp} (${new Date(lastSyncTimestamp).toISOString()})`);\n          console.log(`\u73fe\u5728\u306e\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7 (JST): ${currentTime} (${new Date(currentTime).toISOString()})`);\n          console.log(`\u6642\u9593\u5dee: ${timeDiff}ms = ${hoursDiff}\u6642\u9593`);\n          \n          \/\/ 24\u6642\u9593\u4ee5\u4e0a\u7d4c\u904e\u3057\u3066\u3044\u308b\u5834\u5408\u306f\u540c\u671f\u304c\u5fc5\u8981\n          if (hoursDiff &gt;= 24) {\n            needSync = true;\n            reason = `\u524d\u56de\u306e\u540c\u671f\u304b\u3089${Math.floor(hoursDiff)}\u6642\u9593\u7d4c\u904e\u3057\u305f\u305f\u3081`;\n            console.log(`\u540c\u671f\u6761\u4ef6\u6210\u7acb: ${reason}`);\n          } else {\n            console.log(&#39;\u524d\u56de\u306e\u540c\u671f\u304b\u308924\u6642\u9593\u7d4c\u904e\u3057\u3066\u3044\u306a\u3044\u305f\u3081\u540c\u671f\u3092\u30b9\u30ad\u30c3\u30d7\u3057\u307e\u3059&#39;);\n            return res.json({\n              success: true,\n              message: &quot;\u524d\u56de\u306e\u540c\u671f\u304b\u308924\u6642\u9593\u7d4c\u904e\u3057\u3066\u3044\u306a\u3044\u305f\u3081\u540c\u671f\u3092\u30b9\u30ad\u30c3\u30d7\u3057\u307e\u3057\u305f&quot;,\n              lastSync: new Date(lastSyncTimestamp).toISOString(),\n              hoursSinceLastSync: hoursDiff.toFixed(2)\n            });\n          }\n        }\n      }\n      \n      \/\/ \u540c\u671f\u304c\u5fc5\u8981\u306a\u5834\u5408\u306e\u307f\u5b9f\u884c\n      if (needSync) {\n        console.log(`\u540c\u671f\u3092\u5b9f\u884c\u3057\u307e\u3059\uff08\u7406\u7531: ${reason}\uff09...`);\n        \n        \/\/ clean-sync-slack-data\u30a8\u30f3\u30c9\u30dd\u30a4\u30f3\u30c8\u3092\u5185\u90e8\u3067\u547c\u3073\u51fa\u3059\n        console.log(&#39;clean-sync-slack-data\u30a8\u30f3\u30c9\u30dd\u30a4\u30f3\u30c8\u3092\u5185\u90e8\u3067\u547c\u3073\u51fa\u3057\u307e\u3059...&#39;);\n        \n        const response = await fetch(`http:\/\/localhost:${port}\/api\/clean-sync-slack-data`, {\n          method: &#39;GET&#39;\n        });\n        \n        if (!response.ok) {\n          const errorText = await response.text();\n          throw new Error(`clean-sync-slack-data\u306e\u547c\u3073\u51fa\u3057\u306b\u5931\u6557\u3057\u307e\u3057\u305f: ${response.status} ${errorText}`);\n        }\n        \n        const result = await response.json();\n        \n        \/\/ clean-sync-slack-data\u306e\u7d50\u679c\u3092\u305d\u306e\u307e\u307e\u8fd4\u3059\n        console.log(&#39;\u671d\u306e\u81ea\u52d5\u540c\u671f\u304c\u5b8c\u4e86\u3057\u307e\u3057\u305f&#39;);\n        return res.json({\n          ...result,\n          message: &quot;\u671d\u306e\u81ea\u52d5\u540c\u671f\u304c\u5b8c\u4e86\u3057\u307e\u3057\u305f&quot;,\n          reason: reason\n        });\n      }\n    } catch (syncError) {\n      console.error(&#39;clean-sync-slack-data\u547c\u3073\u51fa\u3057\u30a8\u30e9\u30fc:&#39;, syncError);\n      return res.status(500).json({ \n        error: `\u540c\u671f\u51e6\u7406\u4e2d\u306b\u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u307e\u3057\u305f: ${syncError.message}` \n      });\n    }\n  } catch (error) {\n    console.error(&#39;\u671d\u306e\u81ea\u52d5\u540c\u671f\u30a8\u30e9\u30fc:&#39;, error);\n    return res.status(500).json({ \n      error: `\u540c\u671f\u51e6\u7406\u4e2d\u306b\u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u307e\u3057\u305f: ${error.message}` \n    });\n  }\n});\n\n\/\/ \u6700\u7d42\u540c\u671f\u65e5\u6642\u3092\u30c1\u30a7\u30c3\u30af\u3057\u3001\u5fc5\u8981\u306b\u5fdc\u3058\u3066\u540c\u671f\u3092\u5b9f\u884c\u3059\u308b\u95a2\u6570\nasync function checkAndSyncIfNeeded() {\n  console.log(&#39;\\n==========================&#39;);\n  console.log(&#39;checkAndSyncIfNeeded\u95a2\u6570\u304c\u547c\u3073\u51fa\u3055\u308c\u307e\u3057\u305f&#39;);\n  console.log(&#39;==========================\\n&#39;);\n  \n  try {\n    console.log(&#39;\\n======== \u81ea\u52d5\u540c\u671f\u30c1\u30a7\u30c3\u30af\u958b\u59cb ========&#39;);\n    const today = getJSTDateString();\n    const currentTime = getJSTDate().getTime();\n    \n    console.log(`\u4eca\u65e5\u306e\u65e5\u4ed8 (JST): ${today}`);\n    console.log(`\u73fe\u5728\u306e\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7 (JST): ${currentTime}`);\n    \n    \/\/ DB\u304b\u3089\u6700\u7d42\u540c\u671f\u65e5\u6642\u3092\u78ba\u8a8d\n    try {\n      const lastSyncData = await new Promise((resolve, reject) =&gt; {\n        db.get(&#39;SELECT * FROM last_sync_date ORDER BY id DESC LIMIT 1&#39;, [], (err, row) =&gt; {\n          if (err) {\n            console.error(&#39;\u6700\u7d42\u540c\u671f\u30c7\u30fc\u30bf\u306e\u78ba\u8a8d\u306b\u5931\u6557:&#39;, err);\n            reject(err);\n            return;\n          }\n          \n          if (row) {\n            console.log(&#39;\u6700\u7d42\u540c\u671f\u30c7\u30fc\u30bf:&#39;, JSON.stringify(row, null, 2));\n          } else {\n            console.log(&#39;\u6700\u7d42\u540c\u671f\u30c7\u30fc\u30bf\u304c\u5b58\u5728\u3057\u307e\u305b\u3093&#39;);\n          }\n          resolve(row);\n        });\n      });\n      \n      \/\/ \u540c\u671f\u304c\u5fc5\u8981\u304b\u3069\u3046\u304b\u3092\u5224\u65ad\n      let needSync = false;\n      let reason = &quot;&quot;;\n      \n      if (!lastSyncData) {\n        console.log(&#39;\u521d\u56de\u30a2\u30af\u30bb\u30b9: \u540c\u671f\u30c7\u30fc\u30bf\u304c\u3042\u308a\u307e\u305b\u3093&#39;);\n        \/\/ \u521d\u56de\u5b9f\u884c\n        needSync = true;\n        reason = &quot;\u521d\u56de\u30a2\u30af\u30bb\u30b9\u306e\u305f\u3081&quot;;\n      } else {\n        \/\/ \u6700\u7d42\u540c\u671f\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7\u304b\u308924\u6642\u9593\u4ee5\u4e0a\u7d4c\u904e\u3057\u3066\u3044\u308b\u304b\u78ba\u8a8d\n        const lastSyncTimestamp = Number(lastSyncData.last_sync_timestamp);\n        if (isNaN(lastSyncTimestamp)) {\n          console.log(&#39;\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7\u304cNaN\u3067\u3059:&#39;, lastSyncData.last_sync_timestamp);\n          needSync = true;\n          reason = &quot;\u6709\u52b9\u306a\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7\u304c\u306a\u3044\u305f\u3081&quot;;\n        } else {\n          const timeDiff = currentTime - lastSyncTimestamp;\n          const hoursDiff = timeDiff \/ (1000 * 60 * 60); \/\/ \u30df\u30ea\u79d2 \u2192 \u6642\u9593\n          \n          console.log(`\u524d\u56de\u306e\u540c\u671f\u304b\u3089${hoursDiff.toFixed(2)}\u6642\u9593\u7d4c\u904e\u3057\u3066\u3044\u307e\u3059`);\n          console.log(`\u6700\u7d42\u540c\u671f\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7: ${lastSyncTimestamp} (${new Date(lastSyncTimestamp).toISOString()})`);\n          console.log(`\u73fe\u5728\u306e\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7 (JST): ${currentTime} (${new Date(currentTime).toISOString()})`);\n          console.log(`\u6642\u9593\u5dee: ${timeDiff}ms = ${hoursDiff}\u6642\u9593`);\n          \n          \/\/ 24\u6642\u9593\u4ee5\u4e0a\u7d4c\u904e\u3057\u3066\u3044\u308b\u5834\u5408\u306f\u540c\u671f\u304c\u5fc5\u8981\n          if (hoursDiff &gt;= 24) {\n            needSync = true;\n            reason = `\u524d\u56de\u306e\u540c\u671f\u304b\u3089${Math.floor(hoursDiff)}\u6642\u9593\u7d4c\u904e\u3057\u305f\u305f\u3081`;\n            console.log(`\u81ea\u52d5\u540c\u671f\u6761\u4ef6\u6210\u7acb: ${reason}`);\n          } else {\n            console.log(&#39;\u524d\u56de\u306e\u540c\u671f\u304b\u308924\u6642\u9593\u7d4c\u904e\u3057\u3066\u3044\u306a\u3044\u305f\u3081\u30b9\u30ad\u30c3\u30d7\u3057\u307e\u3059&#39;);\n            \n            \/\/ \u6700\u7d42\u540c\u671f\u65e5\u6642\u306e\u53d6\u5f97\uff08YYYY-MM-DD\u5f62\u5f0f\u306b\u5909\u63db\uff09\n            let lastSyncDate;\n            try {\n              lastSyncDate = new Date(lastSyncData.last_sync);\n              const lastSyncDateStr = lastSyncDate.toISOString().split(&#39;T&#39;)[0];\n              console.log(`\u6700\u7d42\u540c\u671f\u65e5: ${lastSyncDateStr}, \u4eca\u65e5 (JST): ${today}`);\n              \n              \/\/ \u65e5\u4ed8\u304c\u5909\u308f\u3063\u305f\u5834\u5408\u306f\u30a4\u30c1\u63a8\u3057Braver\u306e\u9078\u51fa\u306e\u307f\u884c\u3046\n              if (lastSyncDateStr !== today) {\n                console.log(&#39;\u6700\u7d42\u540c\u671f\u65e5\u304c\u4eca\u65e5\u3067\u306f\u3042\u308a\u307e\u305b\u3093\u3002\u30a4\u30c1\u63a8\u3057Braver\u3092\u9078\u51fa\u3057\u307e\u3059...&#39;);\n                try {\n                  await resetAndSelectDailyPickupUser();\n                  \n                  \/\/ \u6700\u7d42\u540c\u671f\u65e5\u3082\u66f4\u65b0\uff08\u65e5\u672c\u6642\u9593\uff09\n                  const jstNow = getJSTDate();\n                  const timestamp = jstNow.getTime();\n                  const datetimeStr = jstNow.toISOString().replace(&#39;T&#39;, &#39; &#39;).replace(&#39;Z&#39;, &#39;&#39;);\n                  \n                  console.log(`\u6700\u7d42\u540c\u671f\u65e5\u3092\u66f4\u65b0\u3057\u307e\u3059 (JST): ${datetimeStr}, ${timestamp}`);\n                  await new Promise((resolve, reject) =&gt; {\n                    db.run(\n                      &#39;UPDATE last_sync_date SET last_sync = ?, last_sync_timestamp = ? WHERE id = ?&#39;, \n                      [datetimeStr, timestamp, lastSyncData.id], \n                      function(err) {\n                        if (err) {\n                          console.error(&#39;\u6700\u7d42\u540c\u671f\u65e5\u306e\u66f4\u65b0\u306b\u5931\u6557:&#39;, err);\n                          reject(err);\n                        } else {\n                          console.log(`\u6700\u7d42\u540c\u671f\u65e5\u3092\u66f4\u65b0\u3057\u307e\u3057\u305f (JST): ${datetimeStr}, ${timestamp}`);\n                          resolve();\n                        }\n                      }\n                    );\n                  });\n                } catch (syncError) {\n                  console.error(&#39;\u30a4\u30c1\u63a8\u3057braver\u9078\u51fa\u306e\u5b9f\u884c\u306b\u5931\u6557:&#39;, syncError);\n                }\n              }\n            } catch (dateError) {\n              console.error(&#39;\u65e5\u4ed8\u5909\u63db\u30a8\u30e9\u30fc:&#39;, dateError, lastSyncData.last_sync);\n            }\n            \n            return; \/\/ \u540c\u671f\u4e0d\u8981\u306e\u5834\u5408\u306f\u51e6\u7406\u3092\u7d42\u4e86\n          }\n        }\n      }\n      \n      \/\/ \u540c\u671f\u304c\u5fc5\u8981\u306a\u5834\u5408\u306f\u5b9f\u884c\n      if (needSync) {\n        console.log(`\u81ea\u52d5\u540c\u671f\u3092\u958b\u59cb\u3057\u307e\u3059\uff08\u7406\u7531: ${reason}\uff09...`);\n        \n        try {\n          \/\/ clean-sync-slack-data\u306e\u51e6\u7406\u3092\u547c\u3073\u51fa\u3059\n          console.log(`\u540c\u671fAPI\u3092\u547c\u3073\u51fa\u3057\u307e\u3059: http:\/\/localhost:${port}\/api\/clean-sync-slack-data`);\n          const response = await fetch(`http:\/\/localhost:${port}\/api\/clean-sync-slack-data`, {\n            method: &#39;GET&#39;\n          });\n          \n          if (!response.ok) {\n            const errorText = await response.text();\n            throw new Error(`\u540c\u671f\u30a8\u30e9\u30fc: ${response.status} ${response.statusText} - ${errorText}`);\n          }\n          \n          const data = await response.json();\n          console.log(&#39;\u81ea\u52d5\u540c\u671f\u5b8c\u4e86:&#39;, data);\n          \n          \/\/ \u6700\u7d42\u540c\u671f\u65e5\u6642\u3092\u66f4\u65b0\uff08\u65e5\u672c\u6642\u9593\uff09\n          const jstNow = getJSTDate();\n          const timestamp = jstNow.getTime();\n          const datetimeStr = jstNow.toISOString().replace(&#39;T&#39;, &#39; &#39;).replace(&#39;Z&#39;, &#39;&#39;);\n          \n          console.log(`\u540c\u671f\u5f8c\u306b\u6700\u7d42\u540c\u671f\u65e5\u6642\u3092\u66f4\u65b0 (JST): ${datetimeStr}, ${timestamp}`);\n          await new Promise((resolve, reject) =&gt; {\n            db.run(\n              &#39;UPDATE last_sync_date SET last_sync = ?, last_sync_timestamp = ? WHERE id = ?&#39;, \n              [datetimeStr, timestamp, lastSyncData ? lastSyncData.id : 1], \n              function(err) {\n                if (err) {\n                  console.error(&#39;\u6700\u7d42\u540c\u671f\u65e5\u6642\u306e\u66f4\u65b0\u306b\u5931\u6557:&#39;, err);\n                  reject(err);\n                } else {\n                  console.log(`\u6700\u7d42\u540c\u671f\u65e5\u6642\u3092\u66f4\u65b0\u3057\u307e\u3057\u305f (JST): ${datetimeStr}, ${timestamp}`);\n                  resolve();\n                }\n              }\n            );\n          });\n        } catch (syncError) {\n          console.error(&#39;\u81ea\u52d5\u540c\u671f\u306e\u5b9f\u884c\u306b\u5931\u6557:&#39;, syncError);\n        }\n      }\n    } catch (dbError) {\n      console.error(&#39;\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u64cd\u4f5c\u30a8\u30e9\u30fc:&#39;, dbError);\n    }\n    \n    console.log(&#39;======== \u81ea\u52d5\u540c\u671f\u30c1\u30a7\u30c3\u30af\u7d42\u4e86 ========\\n&#39;);\n  } catch (error) {\n    console.error(&#39;\u540c\u671f\u30c1\u30a7\u30c3\u30af\u51e6\u7406\u3067\u30a8\u30e9\u30fc\u304c\u767a\u751f:&#39;, error);\n  }\n}\n\n\/\/ \u30e1\u30a4\u30f3\u30da\u30fc\u30b8\u3078\u306e\u30a2\u30af\u30bb\u30b9\u6642\u306b\u540c\u671f\u30c1\u30a7\u30c3\u30af\u3092\u5b9f\u884c\napp.get(&#39;\/&#39;, async (req, res) =&gt; {\n  console.log(&#39;\u30eb\u30fc\u30c8\u30d1\u30b9\u306b\u30a2\u30af\u30bb\u30b9\u304c\u3042\u308a\u307e\u3057\u305f - checkAndSyncIfNeeded\u95a2\u6570\u3092\u547c\u3073\u51fa\u3057\u307e\u3059&#39;);\n  try {\n    await checkAndSyncIfNeeded();\n    console.log(&#39;checkAndSyncIfNeeded\u95a2\u6570\u306e\u5b9f\u884c\u304c\u5b8c\u4e86\u3057\u307e\u3057\u305f&#39;);\n  } catch (error) {\n    console.error(&#39;checkAndSyncIfNeeded\u95a2\u6570\u306e\u5b9f\u884c\u4e2d\u306b\u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u307e\u3057\u305f:&#39;, error);\n  }\n  res.sendFile(path.join(__dirname, &#39;public&#39;, &#39;index.html&#39;));\n});\n\n\/\/ \u305d\u306e\u4ed6\u306e\u30da\u30fc\u30b8\u30a2\u30af\u30bb\u30b9\u6642\u306b\u3082\u540c\u671f\u30c1\u30a7\u30c3\u30af\u3092\u5b9f\u884c\napp.get(&#39;\/simple&#39;, async (req, res) =&gt; {\n  await checkAndSyncIfNeeded();\n  res.sendFile(path.join(__dirname, &#39;public&#39;, &#39;simple.html&#39;));\n});\n\n\/\/ ... existing code ...\n\n\/\/ \u8a2d\u5b9a\u3092\u4fdd\u5b58\u3059\u308b\u95a2\u6570\nfunction saveSettings(settings) {\n  try {\n    fs.writeFileSync(SETTINGS_FILE_PATH, JSON.stringify(settings, null, 2), &#39;utf8&#39;);\n    console.log(&#39;\u8a2d\u5b9a\u30d5\u30a1\u30a4\u30eb\u3092\u4fdd\u5b58\u3057\u307e\u3057\u305f&#39;);\n    return true;\n  } catch (error) {\n    console.error(&#39;\u8a2d\u5b9a\u30d5\u30a1\u30a4\u30eb\u4fdd\u5b58\u30a8\u30e9\u30fc:&#39;, error);\n    return false;\n  }\n}\n\n\/\/ ... existing code ...\n\n\/\/ Google Sheets API\u306e\u8a2d\u5b9a\nconst SCOPES = [&#39;https:\/\/www.googleapis.com\/auth\/spreadsheets.readonly&#39;];\nconst SPREADSHEET_ID = process.env.GOOGLE_SHEET_ID;\nconst SHEET_NAME = &#39;\u55b6\u696d\u72b6\u6cc1&#39;;\n\n\/\/ Google Sheets API\u30af\u30e9\u30a4\u30a2\u30f3\u30c8\u306e\u521d\u671f\u5316\nasync function initGoogleSheetsClient() {\n  try {\n    const auth = new JWT({\n      email: process.env.GOOGLE_CLIENT_EMAIL,\n      key: process.env.GOOGLE_PRIVATE_KEY.replace(\/\\\\n\/g, &#39;\\n&#39;),\n      scopes: SCOPES,\n    });\n\n    const sheets = google.sheets({ version: &#39;v4&#39;, auth });\n    return sheets;\n  } catch (error) {\n    console.error(&#39;Google Sheets API\u30af\u30e9\u30a4\u30a2\u30f3\u30c8\u306e\u521d\u671f\u5316\u306b\u5931\u6557\u3057\u307e\u3057\u305f:&#39;, error);\n    throw error;\n  }\n}\n\n\/\/ \u55b6\u696d\u30c7\u30fc\u30bf\u306e\u53d6\u5f97\nasync function getSalesData() {\n  try {\n    const sheets = await initGoogleSheetsClient();\n    const response = await sheets.spreadsheets.values.get({\n      spreadsheetId: SPREADSHEET_ID,\n      range: `${SHEET_NAME}!A:Z`,\n    });\n\n    const rows = response.data.values;\n    if (!rows || rows.length === 0) {\n      console.log(&#39;\u30c7\u30fc\u30bf\u304c\u898b\u3064\u304b\u308a\u307e\u305b\u3093\u3067\u3057\u305f\u3002&#39;);\n      return [];\n    }\n\n    \/\/ \u30d8\u30c3\u30c0\u30fc\u884c\u3092\u53d6\u5f97\n    const headers = rows[0];\n    const data = rows.slice(1).map(row =&gt; {\n      const item = {};\n      headers.forEach((header, index) =&gt; {\n        item[header] = row[index] || &#39;&#39;;\n      });\n      return item;\n    });\n\n    return data;\n  } catch (error) {\n    console.error(&#39;\u55b6\u696d\u30c7\u30fc\u30bf\u306e\u53d6\u5f97\u306b\u5931\u6557\u3057\u307e\u3057\u305f:&#39;, error);\n    throw error;\n  }\n}\n\n\/\/ \u55b6\u696d\u30c7\u30fc\u30bf\u3092\u53d6\u5f97\u3059\u308bAPI\u30a8\u30f3\u30c9\u30dd\u30a4\u30f3\u30c8\napp.get(&#39;\/api\/sales-data&#39;, async (req, res) =&gt; {\n  try {\n    const salesData = await getSalesData();\n    res.json(salesData);\n  } catch (error) {\n    console.error(&#39;\u55b6\u696d\u30c7\u30fc\u30bf\u306e\u53d6\u5f97\u306b\u5931\u6557\u3057\u307e\u3057\u305f:&#39;, error);\n    res.status(500).json({ error: &#39;\u55b6\u696d\u30c7\u30fc\u30bf\u306e\u53d6\u5f97\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n  }\n});\n\/\/ ... existing code ...\n\n\/\/ \u300c\u76f4\u8fd13\u30f6\u6708\u306e\u30c7\u30fc\u30bf\u540c\u671f\u300d\u7528API\napp.get(&#39;\/api\/sync-three-months-data&#39;, async (req, res) =&gt; {\n  try {\n    console.log(&#39;\u76f4\u8fd13\u30f6\u6708\u306e\u30c7\u30fc\u30bf\u540c\u671f\u3092\u958b\u59cb\u3057\u307e\u3059...&#39;);\n    \n    \/\/ \u30af\u30ea\u30fc\u30f3\u30a2\u30c3\u30d7\u51e6\u7406\n    try {\n      \/\/ 3\u30f6\u6708\u524d\u306e\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7\u3092\u8a08\u7b97\uff08\u65e5\u672c\u6642\u9593\u30d9\u30fc\u30b9\uff09\n      const threeMonthsAgo = getJSTDate();\n      threeMonthsAgo.setDate(threeMonthsAgo.getDate() - 90);\n      const threeMonthsAgoTimestamp = threeMonthsAgo.getTime() \/ 1000;\n      \n      console.log(`\u76f4\u8fd13\u30f6\u6708\u5206(${threeMonthsAgo.toLocaleString(&#39;ja-JP&#39;)}\u4ee5\u964d)\u306eSlack\u30c7\u30fc\u30bf\u3092\u524a\u9664\u3057\u307e\u3059...`);\n      \n      \/\/ \u524a\u9664\u51e6\u7406\u3092Promise\u3068\u3057\u3066\u5b9f\u884c\n      const deleteMessages = new Promise((resolve, reject) =&gt; {\n        db.run(&#39;DELETE FROM slack_messages WHERE timestamp &gt;= ?&#39;, [threeMonthsAgoTimestamp], function(err) {\n          if (err) {\n            console.error(&#39;\u76f4\u8fd13\u30f6\u6708\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u524a\u9664\u30a8\u30e9\u30fc:&#39;, err);\n            reject(err);\n          } else {\n            console.log(`\u76f4\u8fd13\u30f6\u6708\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u524a\u9664\u3057\u307e\u3057\u305f: ${this.changes}\u4ef6\u306e\u30c7\u30fc\u30bf\u3092\u524a\u9664`);\n            resolve();\n          }\n        });\n      });\n      \n      const deleteThreadReplies = new Promise((resolve, reject) =&gt; {\n        db.run(&#39;DELETE FROM slack_thread_replies WHERE timestamp &gt;= ?&#39;, [threeMonthsAgoTimestamp], function(err) {\n          if (err) {\n            console.error(&#39;\u76f4\u8fd13\u30f6\u6708\u306e\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u524a\u9664\u30a8\u30e9\u30fc:&#39;, err);\n            reject(err);\n          } else {\n            console.log(`\u76f4\u8fd13\u30f6\u6708\u306e\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u3092\u524a\u9664\u3057\u307e\u3057\u305f: ${this.changes}\u4ef6\u306e\u30c7\u30fc\u30bf\u3092\u524a\u9664`);\n            resolve();\n          }\n        });\n      });\n      \n      const deleteReactions = new Promise((resolve, reject) =&gt; {\n        db.run(&#39;DELETE FROM slack_reactions WHERE timestamp &gt;= ?&#39;, [threeMonthsAgoTimestamp], function(err) {\n          if (err) {\n            console.error(&#39;\u76f4\u8fd13\u30f6\u6708\u306e\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u524a\u9664\u30a8\u30e9\u30fc:&#39;, err);\n            reject(err);\n          } else {\n            console.log(`\u76f4\u8fd13\u30f6\u6708\u306e\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u3092\u524a\u9664\u3057\u307e\u3057\u305f: ${this.changes}\u4ef6\u306e\u30c7\u30fc\u30bf\u3092\u524a\u9664`);\n            resolve();\n          }\n        });\n      });\n      \n      \/\/ \u3059\u3079\u3066\u306e\u524a\u9664\u51e6\u7406\u3092\u5f85\u3064\n      await Promise.all([deleteMessages, deleteThreadReplies, deleteReactions]);\n      console.log(&#39;\u3059\u3079\u3066\u306e\u30c7\u30fc\u30bf\u524a\u9664\u51e6\u7406\u304c\u5b8c\u4e86\u3057\u307e\u3057\u305f&#39;);\n      \n      \/\/ \u5168\u30c1\u30e3\u30f3\u30cd\u30eb\u306e\u540c\u671f\u3092\u5b9f\u884c\n      if (!slackToken || !slackChannelIds.length) {\n        return res.status(500).json({ \n          error: &#39;Slack API\u306e\u8a2d\u5b9a\u304c\u4e0d\u8db3\u3057\u3066\u3044\u307e\u3059\u3002\u74b0\u5883\u5909\u6570\u3092\u78ba\u8a8d\u3057\u3066\u304f\u3060\u3055\u3044\u3002&#39; \n        });\n      }\n      \n      const channelsToSync = global.slackChannelIds || slackChannelIds;\n      console.log(`\u8a2d\u5b9a\u3055\u308c\u3066\u3044\u308b\u5168\u30c1\u30e3\u30f3\u30cd\u30eb (${channelsToSync.length}\u4ef6) \u3092\u540c\u671f\u3057\u307e\u3059: ${channelsToSync.join(&#39;, &#39;)}`);\n      \n      \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u3054\u3068\u306e\u540c\u671f\u7d50\u679c\u3092\u4fdd\u5b58\n      const syncResults = [];\n      const errorChannels = [];\n      let totalMessageCount = 0;\n      let totalReplyCount = 0;\n      let totalReactionCount = 0;\n      \n      \/\/ API\u30ec\u30fc\u30c8\u5236\u9650\u3092\u56de\u907f\u3059\u308b\u305f\u3081\u306e\u30c7\u30a3\u30ec\u30a4\u95a2\u6570\n      const delay = ms =&gt; new Promise(resolve =&gt; setTimeout(resolve, ms));\n      \n      \/\/ \u5404\u30c1\u30e3\u30f3\u30cd\u30eb\u3092\u9806\u756a\u306b\u51e6\u7406\uff08\u30ec\u30fc\u30c8\u5236\u9650\u3092\u56de\u907f\u3059\u308b\u305f\u3081\u9806\u6b21\u51e6\u7406\uff09\n      for (const channelId of channelsToSync) {\n        try {\n          console.log(`\u30c1\u30e3\u30f3\u30cd\u30eb ${channelId} \u306e\u540c\u671f\u3092\u958b\u59cb\u3057\u307e\u3059...`);\n          \n          \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u60c5\u5831\u3092\u53d6\u5f97 - \u30ec\u30fc\u30c8\u5236\u9650\u5bfe\u7b56\n          let channelInfo;\n          try {\n            channelInfo = await slackClient.conversations.info({\n              channel: channelId\n            });\n          } catch (error) {\n            if (error.code === &#39;rate_limited&#39;) {\n              console.log(`\u30ec\u30fc\u30c8\u5236\u9650\u306b\u3088\u308a\u5f85\u6a5f\u3057\u307e\u3059: ${error.retryAfter || 5}\u79d2`);\n              await delay((error.retryAfter || 5) * 1000);\n              channelInfo = await slackClient.conversations.info({\n                channel: channelId\n              });\n            } else {\n              throw error;\n            }\n          }\n          \n          const channelName = channelInfo.channel.name || &#39;Unknown Channel&#39;;\n          console.log(`\u30c1\u30e3\u30f3\u30cd\u30eb\u540d: ${channelName}`);\n          \n          \/\/ API\u30ea\u30af\u30a8\u30b9\u30c8\u9593\u306b\u30c7\u30a3\u30ec\u30a4\u3092\u633f\u5165\u3057\u3066\u30ec\u30fc\u30c8\u5236\u9650\u3092\u56de\u907f\n          await delay(1000);\n          \n          let cursor = undefined;\n          let messageCount = 0;\n          let replyCount = 0;\n          let reactionCount = 0;\n          let hasMore = true;\n          \n          \/\/ \u30da\u30fc\u30b8\u30cd\u30fc\u30b7\u30e7\u30f3\u3092\u4f7f\u7528\u3057\u30663\u30f6\u6708\u5206\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u53d6\u5f97\n          while (hasMore) {\n            \/\/ \u30c1\u30e3\u30f3\u30cd\u30eb\u5c65\u6b74\u3092\u53d6\u5f97 - \u30ec\u30fc\u30c8\u5236\u9650\u5bfe\u7b56\n            let result;\n            try {\n              result = await slackClient.conversations.history({\n                channel: channelId,\n                limit: 100,\n                cursor: cursor,\n                oldest: threeMonthsAgoTimestamp\n              });\n            } catch (error) {\n              if (error.code === &#39;rate_limited&#39;) {\n                console.log(`\u30ec\u30fc\u30c8\u5236\u9650\u306b\u3088\u308a\u5f85\u6a5f\u3057\u307e\u3059: ${error.retryAfter || 5}\u79d2`);\n                await delay((error.retryAfter || 5) * 1000);\n                result = await slackClient.conversations.history({\n                  channel: channelId,\n                  limit: 100,\n                  cursor: cursor,\n                  oldest: threeMonthsAgoTimestamp\n                });\n              } else {\n                throw error;\n              }\n            }\n            \n            console.log(`\u30c1\u30e3\u30f3\u30cd\u30eb ${channelName} \u304b\u3089 ${result.messages.length} \u4ef6\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u53d6\u5f97`);\n            \n            \/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u4fdd\u5b58\n            for (const message of result.messages) {\n              \/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u306e\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7\u3092\u30c1\u30a7\u30c3\u30af\n              if (message.ts && parseFloat(message.ts) &gt;= threeMonthsAgoTimestamp) {\n                \/\/ \u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u4fdd\u5b58\n                const messageType = message.subtype || &#39;message&#39;;\n                const userId = message.user || &#39;unknown_user&#39;;\n                const text = message.text || &#39;&#39;;\n                const timestamp = message.ts;\n                \n                \/\/ DB\u306b\u4fdd\u5b58\n                db.run(\n                  &#39;INSERT OR REPLACE INTO slack_messages (message_id, channel_id, channel_name, user_id, text, message_type, timestamp, created_at) VALUES (?, ?, ?, ?, ?, ?, ?, ?)&#39;,\n                  [\n                    timestamp,\n                    channelId,\n                    channelName,\n                    userId,\n                    text,\n                    messageType,\n                    timestamp,\n                    Math.floor(Date.now() \/ 1000)\n                  ],\n                  function(err) {\n                    if (err) {\n                      console.error(`\u30e1\u30c3\u30bb\u30fc\u30b8\u4fdd\u5b58\u30a8\u30e9\u30fc: ${err.message}`);\n                    } else {\n                      messageCount++;\n                    }\n                  }\n                );\n                \n                \/\/ \u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u304c\u3042\u308c\u3070\u53d6\u5f97\n                if (message.thread_ts && message.reply_count &gt; 0) {\n                  console.log(`\u30b9\u30ec\u30c3\u30c9\u691c\u51fa: message_id=${message.ts}, thread_ts=${message.thread_ts}, reply_count=${message.reply_count}`);\n                  console.log(`\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u3092\u53d6\u5f97\u3057\u307e\u3059: thread_ts=${message.thread_ts}, reply_count=${message.reply_count}`);\n                  \n                  \/\/ API\u30ea\u30af\u30a8\u30b9\u30c8\u9593\u306b\u30c7\u30a3\u30ec\u30a4\u3092\u633f\u5165\u3057\u3066\u30ec\u30fc\u30c8\u5236\u9650\u3092\u56de\u907f\n                  await delay(1000);\n                  \n                  \/\/ \u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u3092\u53d6\u5f97 - \u30ec\u30fc\u30c8\u5236\u9650\u5bfe\u7b56\n                  let repliesResult;\n                  try {\n                    repliesResult = await slackClient.conversations.replies({\n                      channel: channelId,\n                      ts: message.thread_ts,\n                      limit: 100\n                    });\n                  } catch (error) {\n                    if (error.code === &#39;rate_limited&#39;) {\n                      console.log(`\u30ec\u30fc\u30c8\u5236\u9650\u306b\u3088\u308a\u5f85\u6a5f\u3057\u307e\u3059: ${error.retryAfter || 5}\u79d2`);\n                      await delay((error.retryAfter || 5) * 1000);\n                      repliesResult = await slackClient.conversations.replies({\n                        channel: channelId,\n                        ts: message.thread_ts,\n                        limit: 100\n                      });\n                    } else {\n                      throw error;\n                    }\n                  }\n                  \n                  console.log(`\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u3092 ${repliesResult.messages.length - 1} \u4ef6\u53d6\u5f97\u3057\u307e\u3057\u305f`);\n                  \n                  \/\/ \u6700\u521d\u306e\u30e1\u30c3\u30bb\u30fc\u30b8\u3092\u9664\u5916\u3057\u3066\u8fd4\u4fe1\u306e\u307f\u3092\u51e6\u7406\n                  const replies = repliesResult.messages.slice(1);\n                  \n                  for (const reply of replies) {\n                    const replyUserId = reply.user || &#39;unknown_user&#39;;\n                    const replyText = reply.text || &#39;&#39;;\n                    const replyTimestamp = reply.ts;\n                    \n                    \/\/ \u8fd4\u4fe1\u306e\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7\u3092\u30c1\u30a7\u30c3\u30af\n                    if (replyTimestamp && parseFloat(replyTimestamp) &gt;= threeMonthsAgoTimestamp) {\n                      \/\/ DB\u306b\u4fdd\u5b58\n                      db.run(\n                        &#39;INSERT OR REPLACE INTO slack_thread_replies (reply_id, thread_ts, message_id, channel_id, channel_name, user_id, text, timestamp, created_at) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)&#39;,\n                        [\n                          replyTimestamp,\n                          message.thread_ts,\n                          replyTimestamp,\n                          channelId,\n                          channelName,\n                          replyUserId,\n                          replyText,\n                          replyTimestamp,\n                          Math.floor(Date.now() \/ 1000)\n                        ],\n                        function(err) {\n                          if (err) {\n                            console.error(`\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u4fdd\u5b58\u30a8\u30e9\u30fc: ${err.message}`);\n                          } else {\n                            replyCount++;\n                          }\n                        }\n                      );\n                    }\n                  }\n                }\n                \n                \n                \/\/ \u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u304c\u3042\u308c\u3070\u51e6\u7406\n                if (message.reactions && message.reactions.length &gt; 0) {\n                  console.log(`\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u691c\u51fa: message_id=${message.ts}, reaction_count=${message.reactions.length}`);\n                  \n                  for (const reaction of message.reactions) {\n                    const emojiName = reaction.name;\n                    const userIds = reaction.users || [];\n                    \n                    console.log(`  \u7d75\u6587\u5b57: ${emojiName}, \u4ed8\u3051\u305f\u30e6\u30fc\u30b6\u30fc\u6570: ${userIds.length}`);\n                    \n                    \/\/ \u5404\u30e6\u30fc\u30b6\u30fc\u306e\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u3092\u4fdd\u5b58\n                    for (const reactUserId of userIds) {\n                      db.run(\n                        &#39;INSERT OR REPLACE INTO slack_reactions (message_id, channel_id, user_id, name, timestamp, created_at) VALUES (?, ?, ?, ?, ?, ?)&#39;,\n                        [\n                          message.ts,\n                          channelId,\n                          reactUserId,\n                          emojiName,\n                          message.ts,\n                          Math.floor(Date.now() \/ 1000)\n                        ],\n                        function(err) {\n                          if (err) {\n                            console.error(`\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u4fdd\u5b58\u30a8\u30e9\u30fc: ${err.message}`);\n                          } else {\n                            reactionCount++;\n                          }\n                        }\n                      );\n                    }\n                  }\n                }\n              }\n            }\n            \n            \/\/ \u6b21\u306e\u30da\u30fc\u30b8\u304c\u3042\u308b\u304b\u78ba\u8a8d\n            cursor = result.response_metadata?.next_cursor;\n            hasMore = !!cursor;\n            \n            \/\/ \u30da\u30fc\u30b8\u30cd\u30fc\u30b7\u30e7\u30f3\u304c\u3042\u308b\u5834\u5408\u3001API\u30ec\u30fc\u30c8\u5236\u9650\u3092\u56de\u907f\u3059\u308b\u305f\u3081\u306e\u30c7\u30a3\u30ec\u30a4\n            if (hasMore) {\n              console.log(`\u6b21\u306e\u30da\u30fc\u30b8\u304c\u3042\u308a\u307e\u3059\u3002\u6b21\u306e\u30ab\u30fc\u30bd\u30eb: ${cursor}`);\n              await delay(1000);\n            }\n          }\n          \n          totalMessageCount += messageCount;\n          totalReplyCount += replyCount;\n          totalReactionCount += reactionCount;\n          \n          console.log(`\u30c1\u30e3\u30f3\u30cd\u30eb ${channelName} \u306e\u540c\u671f\u304c\u5b8c\u4e86\u3057\u307e\u3057\u305f`);\n          syncResults.push({\n            channelId,\n            channelName,\n            messageCount,\n            replyCount,\n            reactionCount\n          });\n          \n          \/\/ \u6b21\u306e\u30c1\u30e3\u30f3\u30cd\u30eb\u3092\u51e6\u7406\u3059\u308b\u524d\u306b\u30c7\u30a3\u30ec\u30a4\n          await delay(2000);\n          \n        } catch (error) {\n          console.error(`\u30c1\u30e3\u30f3\u30cd\u30eb ${channelId} \u306e\u540c\u671f\u4e2d\u306b\u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u307e\u3057\u305f:`, error);\n          errorChannels.push({\n            channelId,\n            error: error.message\n          });\n        }\n      }\n      \n      \/\/ \u6700\u7d42\u7d50\u679c\u3092\u30ed\u30b0\u306b\u51fa\u529b\n      console.log(&#39;=== 3\u30f6\u6708\u5206\u306e\u30c7\u30fc\u30bf\u540c\u671f\u304c\u5b8c\u4e86\u3057\u307e\u3057\u305f ===&#39;);\n      console.log(`\u53d6\u5f97\u3055\u308c\u305f\u30e1\u30c3\u30bb\u30fc\u30b8\u7dcf\u6570: ${totalMessageCount}`);\n      console.log(`\u53d6\u5f97\u3055\u308c\u305f\u30b9\u30ec\u30c3\u30c9\u8fd4\u4fe1\u7dcf\u6570: ${totalReplyCount}`);\n      console.log(`\u53d6\u5f97\u3055\u308c\u305f\u30ea\u30a2\u30af\u30b7\u30e7\u30f3\u7dcf\u6570: ${totalReactionCount}`);\n      \n      if (errorChannels.length &gt; 0) {\n        console.log(`\u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u305f\u30c1\u30e3\u30f3\u30cd\u30eb\u6570: ${errorChannels.length}`);\n        errorChannels.forEach(ch =&gt; {\n          console.log(`- \u30c1\u30e3\u30f3\u30cd\u30eb ${ch.channelId}: ${ch.error}`);\n        });\n      }\n      \n      return res.json({\n        success: true,\n        message: &#39;3\u30f6\u6708\u5206\u306e\u30c7\u30fc\u30bf\u540c\u671f\u304c\u5b8c\u4e86\u3057\u307e\u3057\u305f&#39;,\n        messageCount: totalMessageCount,\n        replyCount: totalReplyCount,\n        reactionCount: totalReactionCount,\n        syncResults,\n        errorChannels\n      });\n      \n    } catch (cleanupError) {\n      console.error(&#39;\u30c7\u30fc\u30bf\u540c\u671f\u4e2d\u306b\u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u307e\u3057\u305f:&#39;, cleanupError);\n      return res.status(500).json({ error: `\u30c7\u30fc\u30bf\u540c\u671f\u30a8\u30e9\u30fc: ${cleanupError.message}` });\n    }\n  } catch (error) {\n    console.error(&#39;3\u30f6\u6708\u5206\u306e\u30c7\u30fc\u30bf\u540c\u671f\u4e2d\u306b\u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u307e\u3057\u305f:&#39;, error);\n    return res.status(500).json({ error: `API\u5b9f\u884c\u30a8\u30e9\u30fc: ${error.message}` });\n  }\n});\n\n\/\/ ... existing code ...\n\n\/\/ \u6700\u7d42\u540c\u671f\u65e5\u6642\u60c5\u5831\u3092\u8fd4\u3059API\napp.get(&#39;\/api\/last-sync-info&#39;, (req, res) =&gt; {\n  db.get(&#39;SELECT last_sync, last_sync_timestamp FROM last_sync_date ORDER BY id DESC LIMIT 1&#39;, (err, row) =&gt; {\n    if (err) {\n      console.error(&#39;\u6700\u7d42\u540c\u671f\u30c7\u30fc\u30bf\u306e\u53d6\u5f97\u306b\u5931\u6557:&#39;, err);\n      return res.status(500).json({ error: &#39;\u6700\u7d42\u540c\u671f\u60c5\u5831\u306e\u53d6\u5f97\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n    }\n    \n    if (!row) {\n      return res.json({ \n        last_sync: null, \n        last_sync_timestamp: null,\n        formatted_date: &#39;\u672a\u540c\u671f&#39;\n      });\n    }\n    \n    let formattedDate = &#39;\u672a\u540c\u671f&#39;;\n    \n    if (row.last_sync) {\n      try {\n        \/\/ DB\u5185\u306e\u30bf\u30a4\u30e0\u30b9\u30bf\u30f3\u30d7\u306f\u65e2\u306b\u65e5\u672c\u6642\u9593\u306b\u306a\u3063\u3066\u3044\u308b\u306e\u3067\u3001\u305d\u306e\u307e\u307e\u4f7f\u7528\n        const syncDate = new Date(row.last_sync);\n        const options = { \n          year: &#39;numeric&#39;,\n          month: &#39;2-digit&#39;, \n          day: &#39;2-digit&#39;,\n          hour: &#39;2-digit&#39;,\n          minute: &#39;2-digit&#39;,\n          timeZone: &#39;Asia\/Tokyo&#39; \/\/ \u660e\u793a\u7684\u306b\u65e5\u672c\u6642\u9593\u3092\u6307\u5b9a\n        };\n        formattedDate = syncDate.toLocaleString(&#39;ja-JP&#39;, options);\n        console.log(`\u6700\u7d42\u540c\u671f\u65e5\u6642\u3092\u65e5\u672c\u6642\u9593\u3067\u30d5\u30a9\u30fc\u30de\u30c3\u30c8: ${formattedDate}`);\n      } catch (e) {\n        console.error(&#39;\u65e5\u4ed8\u306e\u5909\u63db\u30a8\u30e9\u30fc:&#39;, e);\n        formattedDate = row.last_sync;\n      }\n    }\n    \n    res.json({\n      last_sync: row.last_sync,\n      last_sync_timestamp: row.last_sync_timestamp,\n      formatted_date: formattedDate\n    });\n  });\n});\n\n\/\/ ... existing code ...\n\n\/\/ \u7279\u5b9a\u30e6\u30fc\u30b6\u30fc\u306e\u8a73\u7d30\u60c5\u5831\u3092\u53d6\u5f97\u3059\u308bAPI\napp.get(&#39;\/api\/users\/:userId&#39;, async (req, res) =&gt; {\n  try {\n    const userId = req.params.userId;\n    console.log(`\u30e6\u30fc\u30b6\u30fc\u8a73\u7d30\u60c5\u5831\u30ea\u30af\u30a8\u30b9\u30c8: ${userId}`);\n    \n    \/\/ \u8a73\u7d30\u306a\u30e6\u30fc\u30b6\u30fc\u60c5\u5831\u3092\u53d6\u5f97\u3059\u308bSQL\n    const query = `\n      SELECT \n        u.user_id, \n        u.name, \n        u.real_name, \n        u.display_name, \n        u.avatar,\n        u.is_bot,\n        (SELECT COUNT(*) FROM slack_messages WHERE user_id = u.user_id) AS message_count,\n        (SELECT COUNT(*) FROM slack_thread_replies WHERE user_id = u.user_id) AS reply_count,\n        (\n          SELECT COUNT(*) FROM slack_messages \n          WHERE user_id = u.user_id \n          AND (text LIKE &#39;\/yolo%&#39; OR text LIKE &#39;\/Yolo%&#39;)\n        ) AS attendance_count,\n        (\n          SELECT COUNT(*) FROM slack_messages \n          WHERE user_id = u.user_id \n          AND (text LIKE &#39;\/yolo_office%&#39; OR text LIKE &#39;\/Yolo_office%&#39;)\n        ) AS office_count,\n        (\n          SELECT COUNT(*) FROM slack_messages \n          WHERE user_id = u.user_id \n          AND length(text) &gt;= 50\n        ) AS weekly_report_count,\n        (\n          SELECT COUNT(*) FROM slack_reactions\n          WHERE user_id = u.user_id\n        ) AS sent_reaction_count,\n        (\n          SELECT COUNT(*) FROM slack_reactions r\n          JOIN slack_messages m ON r.message_id = m.message_id\n          WHERE m.user_id = u.user_id\n        ) AS received_reaction_count\n      FROM \n        slack_users u\n      WHERE \n        u.user_id = ?\n    `;\n    \n    db.get(query, [userId], async (err, user) =&gt; {\n      if (err) {\n        console.error(&#39;\u30e6\u30fc\u30b6\u30fc\u8a73\u7d30\u60c5\u5831\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, err);\n        return res.status(500).json({ error: &#39;\u30e6\u30fc\u30b6\u30fc\u8a73\u7d30\u60c5\u5831\u306e\u53d6\u5f97\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n      }\n      \n      if (!user) {\n        return res.status(404).json({ error: &#39;\u30e6\u30fc\u30b6\u30fc\u304c\u898b\u3064\u304b\u308a\u307e\u305b\u3093&#39; });\n      }\n      \n      \/\/ \u8ffd\u52a0\u60c5\u5831\u3092\u53d6\u5f97\n      try {\n        \/\/ \u9031\u5831\u30fb\u767a\u8a00\u30ea\u30b9\u30c8\uff08\u6700\u65b020\u4ef6\uff09\u3092\u53d6\u5f97 - 50\u6587\u5b57\u4ee5\u4e0a\u306e\u9577\u6587\u6295\u7a3f\u3092\u53d6\u5f97\n        const reportsQuery = `\n          SELECT \n            m.text, \n            m.timestamp, \n            m.channel_name\n          FROM \n            slack_messages m\n          WHERE \n            m.user_id = ? \n            AND length(m.text) &gt;= 50\n          ORDER BY \n            m.timestamp DESC\n          LIMIT 20\n        `;\n        \n        \/\/ \u8fd4\u4fe1\u30ea\u30b9\u30c8\uff08\u6700\u65b020\u4ef6\uff09\u3092\u53d6\u5f97\n        const repliesQuery = `\n          SELECT \n            r.text, \n            r.timestamp, \n            r.channel_name, \n            (SELECT text FROM slack_messages WHERE message_id = r.thread_ts) AS parent_text\n          FROM \n            slack_thread_replies r\n          WHERE \n            r.user_id = ?\n          ORDER BY \n            r.timestamp DESC\n          LIMIT 20\n        `;\n        \n        \/\/ \u9001\u4fe1\u3057\u305f\u30b9\u30bf\u30f3\u30d7\uff08\u6700\u65b020\u4ef6\uff09\u3092\u53d6\u5f97\n        const sentStampsQuery = `\n          SELECT \n            r.name AS reaction, \n            r.timestamp, \n            m.text AS message_text,\n            (SELECT real_name FROM slack_users WHERE user_id = m.user_id) AS target_user\n          FROM \n            slack_reactions r\n          JOIN \n            slack_messages m ON r.message_id = m.message_id\n          WHERE \n            r.user_id = ?\n          ORDER BY \n            r.timestamp DESC\n          LIMIT 20\n        `;\n        \n        \/\/ \u53d7\u4fe1\u3057\u305f\u30b9\u30bf\u30f3\u30d7\uff08\u6700\u65b020\u4ef6\uff09\u3092\u53d6\u5f97\n        const receivedStampsQuery = `\n          SELECT \n            r.name AS reaction, \n            r.timestamp, \n            m.text AS message_text,\n            (SELECT real_name FROM slack_users WHERE user_id = r.user_id) AS from_user\n          FROM \n            slack_reactions r\n          JOIN \n            slack_messages m ON r.message_id = m.message_id\n          WHERE \n            m.user_id = ?\n          ORDER BY \n            r.timestamp DESC\n          LIMIT 20\n        `;\n        \n        \/\/ \u3059\u3079\u3066\u306e\u30af\u30a8\u30ea\u3092\u4e26\u884c\u3057\u3066\u5b9f\u884c\n        const [reports, replies, sentStamps, receivedStamps] = await Promise.all([\n          new Promise((resolve, reject) =&gt; {\n            db.all(reportsQuery, [userId], (err, rows) =&gt; {\n              if (err) {\n                console.error(&#39;\u9031\u5831\u30fb\u767a\u8a00\u5c65\u6b74\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, err);\n                resolve([]);\n              } else {\n                console.log(`\u9577\u6587\u6295\u7a3f\u5c65\u6b74 ${rows.length}\u4ef6\u53d6\u5f97:`);\n                resolve(rows);\n              }\n            });\n          }),\n          new Promise((resolve, reject) =&gt; {\n            db.all(repliesQuery, [userId], (err, rows) =&gt; {\n              if (err) {\n                console.error(&#39;\u8fd4\u4fe1\u5c65\u6b74\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, err);\n                resolve([]);\n              } else {\n                console.log(`\u8fd4\u4fe1\u5c65\u6b74 ${rows.length}\u4ef6\u53d6\u5f97`);\n                resolve(rows);\n              }\n            });\n          }),\n          new Promise((resolve, reject) =&gt; {\n            db.all(sentStampsQuery, [userId], (err, rows) =&gt; {\n              if (err) {\n                console.error(&#39;\u9001\u4fe1\u30b9\u30bf\u30f3\u30d7\u5c65\u6b74\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, err);\n                resolve([]);\n              } else {\n                console.log(`\u9001\u4fe1\u30b9\u30bf\u30f3\u30d7\u5c65\u6b74 ${rows.length}\u4ef6\u53d6\u5f97`);\n                resolve(rows);\n              }\n            });\n          }),\n          new Promise((resolve, reject) =&gt; {\n            db.all(receivedStampsQuery, [userId], (err, rows) =&gt; {\n              if (err) {\n                console.error(&#39;\u53d7\u4fe1\u30b9\u30bf\u30f3\u30d7\u5c65\u6b74\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, err);\n                resolve([]);\n              } else {\n                console.log(`\u53d7\u4fe1\u30b9\u30bf\u30f3\u30d7\u5c65\u6b74 ${rows.length}\u4ef6\u53d6\u5f97`);\n                resolve(rows);\n              }\n            });\n          })\n        ]);\n        \n        \/\/ \u7d50\u679c\u3092\u30e6\u30fc\u30b6\u30fc\u30aa\u30d6\u30b8\u30a7\u30af\u30c8\u306b\u8ffd\u52a0\n        user.reports = reports.map(report =&gt; ({\n          ...report,\n          date: new Date(parseFloat(report.timestamp) * 1000).toISOString(),\n          text_preview: report.text && report.text.length &gt; 200 ? report.text.substring(0, 200) + &#39;...&#39; : report.text\n        }));\n        \n        user.replies = replies.map(reply =&gt; ({\n          ...reply,\n          date: new Date(parseFloat(reply.timestamp) * 1000).toISOString(),\n          text_preview: reply.text && reply.text.length &gt; 100 ? reply.text.substring(0, 100) + &#39;...&#39; : reply.text,\n          parent_preview: reply.parent_text && reply.parent_text.length &gt; 50 ? reply.parent_text.substring(0, 50) + &#39;...&#39; : reply.parent_text\n        }));\n        \n        user.sent_stamps = sentStamps.map(stamp =&gt; ({\n          ...stamp,\n          date: new Date(parseFloat(stamp.timestamp) * 1000).toISOString(),\n          message_preview: stamp.message_text && stamp.message_text.length &gt; 50 ? stamp.message_text.substring(0, 50) + &#39;...&#39; : stamp.message_text\n        }));\n        \n        user.received_stamps = receivedStamps.map(stamp =&gt; ({\n          ...stamp,\n          date: new Date(parseFloat(stamp.timestamp) * 1000).toISOString(),\n          message_preview: stamp.message_text && stamp.message_text.length &gt; 50 ? stamp.message_text.substring(0, 50) + &#39;...&#39; : stamp.message_text\n        }));\n      } catch (dataError) {\n        console.error(&#39;\u8ffd\u52a0\u30c7\u30fc\u30bf\u53d6\u5f97\u30a8\u30e9\u30fc:&#39;, dataError);\n        \/\/ \u30a8\u30e9\u30fc\u304c\u767a\u751f\u3057\u3066\u3082\u3001\u57fa\u672c\u30c7\u30fc\u30bf\u306f\u8fd4\u3059\n      }\n      \n      \/\/ \u6700\u7d42\u7684\u306a\u30e6\u30fc\u30b6\u30fc\u60c5\u5831\u3092\u8fd4\u3059\n      res.json(user);\n    });\n    \n  } catch (error) {\n    console.error(&#39;\u30e6\u30fc\u30b6\u30fc\u8a73\u7d30\u53d6\u5f97\u6642\u306e\u30a8\u30e9\u30fc:&#39;, error);\n    res.status(500).json({ error: &#39;\u30e6\u30fc\u30b6\u30fc\u8a73\u7d30\u60c5\u5831\u306e\u53d6\u5f97\u306b\u5931\u6557\u3057\u307e\u3057\u305f&#39; });\n  }\n});\n<\/code><\/pre><\/div>\n\n\n\n<p><\/p>\n\n\n\n<p><\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u6311\u6226\u8005\u6c42\u30e0<\/h2>\n\n\n\n<p><a href=\"https:\/\/bravesoft.co.jp\/recruit\/\" target=\"_blank\" rel=\"noopener\" title=\"\">AI\u306e\u65b0\u6642\u4ee3\u306b\u5168\u529b\u3067\u6311\u6226\u3057\u305f\u3044\u4eba\u3092\u52df\u96c6\u4e2d\u3067\u3059\uff01<\/a><\/p>\n\n\n\n<p><\/p>\n","protected":false},"excerpt":{"rendered":"<p>\u751f\u6210AI\u306e\u9032\u5316\u304c\u6b62\u307e\u3089\u306a\u3044\uff01 \u65b0\u6280\u8853\u3092\u77e5\u308b\u306b\u306f\u624b\u3092\u52d5\u304b\u3059\u306e\u304c\u4e00\u756a\u3002\u3068\u3044\u3046\u3053\u3068\u3067\u3001GW\u306b1\u9031\u9593\u304b\u3051\u3066\u751f\u6210AI\u3067\u30a2\u30d7\u30ea\u3092\u3064\u304f\u3063\u3066\u307f\u305f\u3002\u6700\u8fd1\u306fvibe coding\uff08\u30d0\u30a4\u30d6\u30fb\u30b3\u30fc\u30c7\u30a3\u30f3\u30b0\uff09\u3068\u3044\u3046\u3089\u3057\u3044\u3002 \u305d\u306e\u7d50\u679c\u3001AI\u3068\u306e\u5bfe\u8a71&#8230;<\/p>\n","protected":false},"author":1,"featured_media":3529,"comment_status":"closed","ping_status":"open","sticky":false,"template":"","format":"standard","meta":{"jetpack_post_was_ever_published":false,"_jetpack_newsletter_access":"","_jetpack_dont_email_post_to_subs":false,"_jetpack_newsletter_tier_id":0,"_jetpack_memberships_contains_paywalled_content":false,"_jetpack_memberships_contains_paid_content":false,"footnotes":"","jetpack_publicize_message":"","jetpack_publicize_feature_enabled":true,"jetpack_social_post_already_shared":true,"jetpack_social_options":{"image_generator_settings":{"template":"highway","enabled":false},"version":2}},"categories":[8,4],"tags":[],"class_list":["post-3495","post","type-post","status-publish","format-standard","has-post-thumbnail","hentry","category-gijyutu","category-keiei"],"jetpack_publicize_connections":[],"aioseo_notices":[],"jetpack_featured_media_url":"https:\/\/i0.wp.com\/bravesoft.co.jp\/seblog\/wp-content\/uploads\/2025\/05\/BFB_KEYV-scaled.jpg?fit=2560%2C1920&ssl=1","jetpack_sharing_enabled":true,"jetpack_shortlink":"https:\/\/wp.me\/p7fAdy-Un","jetpack-related-posts":[],"_links":{"self":[{"href":"https:\/\/bravesoft.co.jp\/seblog\/wp-json\/wp\/v2\/posts\/3495"}],"collection":[{"href":"https:\/\/bravesoft.co.jp\/seblog\/wp-json\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/bravesoft.co.jp\/seblog\/wp-json\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/bravesoft.co.jp\/seblog\/wp-json\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/bravesoft.co.jp\/seblog\/wp-json\/wp\/v2\/comments?post=3495"}],"version-history":[{"count":28,"href":"https:\/\/bravesoft.co.jp\/seblog\/wp-json\/wp\/v2\/posts\/3495\/revisions"}],"predecessor-version":[{"id":3564,"href":"https:\/\/bravesoft.co.jp\/seblog\/wp-json\/wp\/v2\/posts\/3495\/revisions\/3564"}],"wp:featuredmedia":[{"embeddable":true,"href":"https:\/\/bravesoft.co.jp\/seblog\/wp-json\/wp\/v2\/media\/3529"}],"wp:attachment":[{"href":"https:\/\/bravesoft.co.jp\/seblog\/wp-json\/wp\/v2\/media?parent=3495"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/bravesoft.co.jp\/seblog\/wp-json\/wp\/v2\/categories?post=3495"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/bravesoft.co.jp\/seblog\/wp-json\/wp\/v2\/tags?post=3495"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}