backend.js 94 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618
  1. #!/usr/bin/env node
  2. "use strict";
  3. const path = require("path");
  4. const envPaths = require("../server/node_modules/env-paths");
  5. const {cli} = require("../client/oasis_client");
  6. const fs = require("fs");
  7. const os = require('os');
  8. const promisesFs = require("fs").promises;
  9. const SSBconfig = require('../server/SSB_server.js');
  10. const moment = require('../server/node_modules/moment');
  11. const FileType = require('../server/node_modules/file-type');
  12. const ssbRef = require("../server/node_modules/ssb-ref");
  13. const defaultConfig = {};
  14. const defaultConfigFile = path.join(
  15. envPaths("oasis", { suffix: "" }).config,
  16. "/default.json"
  17. );
  18. let haveConfig;
  19. try {
  20. const defaultConfigOverride = fs.readFileSync(defaultConfigFile, "utf8");
  21. Object.entries(JSON.parse(defaultConfigOverride)).forEach(([key, value]) => {
  22. defaultConfig[key] = value;
  23. });
  24. haveConfig = true;
  25. } catch (e) {
  26. if (e.code === "ENOENT") {
  27. haveConfig = false;
  28. } else {
  29. console.log(`There was a problem loading ${defaultConfigFile}`);
  30. throw e;
  31. }
  32. }
  33. const config = cli(defaultConfig, defaultConfigFile);
  34. if (config.debug) {
  35. process.env.DEBUG = "oasis,oasis:*";
  36. }
  37. //AI
  38. const { spawn } = require('child_process');
  39. function startAI() {
  40. const aiPath = path.resolve(__dirname, '../AI/ai_service.mjs');
  41. const aiProcess = spawn('node', [aiPath], {
  42. detached: false,
  43. stdio: 'ignore', //inherit for debug
  44. });
  45. aiProcess.unref();
  46. }
  47. const customStyleFile = path.join(
  48. envPaths("oasis", { suffix: "" }).config,
  49. "/custom-style.css"
  50. );
  51. let haveCustomStyle;
  52. try {
  53. fs.readFileSync(customStyleFile, "utf8");
  54. haveCustomStyle = true;
  55. } catch (e) {
  56. if (e.code === "ENOENT") {
  57. haveCustomStyle = false;
  58. } else {
  59. console.log(`There was a problem loading ${customStyleFile}`);
  60. throw e;
  61. }
  62. }
  63. const { get } = require("node:http");
  64. const debug = require("../server/node_modules/debug")("oasis");
  65. const log = (formatter, ...args) => {
  66. const isDebugEnabled = debug.enabled;
  67. debug.enabled = true;
  68. debug(formatter, ...args);
  69. debug.enabled = isDebugEnabled;
  70. };
  71. delete config._;
  72. delete config.$0;
  73. const { host } = config;
  74. const { port } = config;
  75. const url = `http://${host}:${port}`;
  76. debug("Current configuration: %O", config);
  77. debug(`You can save the above to ${defaultConfigFile} to make \
  78. these settings the default. See the readme for details.`);
  79. const { saveConfig, getConfig } = require('../configs/config-manager');
  80. const configPath = path.join(__dirname, '../configs/oasis-config.json');
  81. const oasisCheckPath = "/.well-known/oasis";
  82. process.on("uncaughtException", function (err) {
  83. if (err["code"] === "EADDRINUSE") {
  84. get(url + oasisCheckPath, (res) => {
  85. let rawData = "";
  86. res.on("data", (chunk) => {
  87. rawData += chunk;
  88. });
  89. res.on("end", () => {
  90. log(rawData);
  91. if (rawData === "oasis") {
  92. log(`Oasis is already running on host ${host} and port ${port}`);
  93. if (config.open === true) {
  94. log("Opening link to existing instance of Oasis");
  95. open(url);
  96. } else {
  97. log(
  98. "Not opening your browser because opening is disabled by your config"
  99. );
  100. }
  101. process.exit(0);
  102. } else {
  103. throw new Error(`Another server is already running at ${url}.
  104. It might be another copy of Oasis or another program on your computer.
  105. You can run Oasis on a different port number with this option:
  106. oasis --port ${config.port + 1}
  107. Alternatively, you can set the default port in ${defaultConfigFile} with:
  108. {
  109. "port": ${config.port + 1}
  110. }
  111. `);
  112. }
  113. });
  114. });
  115. } else {
  116. console.log("");
  117. console.log("Oasis traceback (share below content with devs to report!):");
  118. console.log("===========================================================");
  119. console.log(err);
  120. console.log("");
  121. }
  122. });
  123. process.argv = [];
  124. const http = require("../client/middleware");
  125. const {koaBody} = require("../server/node_modules/koa-body");
  126. const { nav, ul, li, a, form, button, div } = require("../server/node_modules/hyperaxe");
  127. const open = require("../server/node_modules/open");
  128. const pull = require("../server/node_modules/pull-stream");
  129. const koaRouter = require("../server/node_modules/@koa/router");
  130. const ssbMentions = require("../server/node_modules/ssb-mentions");
  131. const isSvg = require('../server/node_modules/is-svg');
  132. const { isFeed, isMsg, isBlob } = require("../server/node_modules/ssb-ref");
  133. const ssb = require("../client/gui");
  134. const router = new koaRouter();
  135. const extractMentions = async (text) => {
  136. const mentions = ssbMentions(text) || [];
  137. const resolvedMentions = await Promise.all(mentions.map(async (mention) => {
  138. const name = mention.name || await about.name(mention.link);
  139. return {
  140. link: mention.link,
  141. name: name || 'Anonymous',
  142. };
  143. }));
  144. return resolvedMentions;
  145. };
  146. const cooler = ssb({ offline: config.offline });
  147. // load core models (cooler)
  148. const models = require("../models/main_models");
  149. const { about, blob, friend, meta, post, vote } = models({
  150. cooler,
  151. isPublic: config.public,
  152. });
  153. const { handleBlobUpload } = require('../backend/blobHandler.js');
  154. // load plugin models (static)
  155. const exportmodeModel = require('../models/exportmode_model');
  156. const panicmodeModel = require('../models/panicmode_model');
  157. const cipherModel = require('../models/cipher_model');
  158. const legacyModel = require('../models/legacy_model');
  159. const walletModel = require('../models/wallet_model')
  160. // load plugin models (cooler)
  161. const pmModel = require('../models/privatemessages_model')({ cooler, isPublic: config.public });
  162. const bookmarksModel = require("../models/bookmarking_model")({ cooler, isPublic: config.public });
  163. const opinionsModel = require('../models/opinions_model')({ cooler, isPublic: config.public });
  164. const eventsModel = require('../models/events_model')({ cooler, isPublic: config.public });
  165. const tasksModel = require('../models/tasks_model')({ cooler, isPublic: config.public });
  166. const votesModel = require('../models/votes_model')({ cooler, isPublic: config.public });
  167. const reportsModel = require('../models/reports_model')({ cooler, isPublic: config.public });
  168. const transfersModel = require('../models/transfers_model')({ cooler, isPublic: config.public });
  169. const tagsModel = require('../models/tags_model')({ cooler, isPublic: config.public });
  170. const cvModel = require('../models/cv_model')({ cooler, isPublic: config.public });
  171. const inhabitantsModel = require('../models/inhabitants_model')({ cooler, isPublic: config.public });
  172. const feedModel = require('../models/feed_model')({ cooler, isPublic: config.public });
  173. const imagesModel = require("../models/images_model")({ cooler, isPublic: config.public });
  174. const audiosModel = require("../models/audios_model")({ cooler, isPublic: config.public });
  175. const videosModel = require("../models/videos_model")({ cooler, isPublic: config.public });
  176. const documentsModel = require("../models/documents_model")({ cooler, isPublic: config.public });
  177. const agendaModel = require("../models/agenda_model")({ cooler, isPublic: config.public });
  178. const trendingModel = require('../models/trending_model')({ cooler, isPublic: config.public });
  179. const statsModel = require('../models/stats_model')({ cooler, isPublic: config.public });
  180. const tribesModel = require('../models/tribes_model')({ cooler, isPublic: config.public });
  181. const searchModel = require('../models/search_model')({ cooler, isPublic: config.public });
  182. const activityModel = require('../models/activity_model')({ cooler, isPublic: config.public });
  183. const pixeliaModel = require('../models/pixelia_model')({ cooler, isPublic: config.public });
  184. const marketModel = require('../models/market_model')({ cooler, isPublic: config.public });
  185. const forumModel = require('../models/forum_model')({ cooler, isPublic: config.public });
  186. const blockchainModel = require('../models/blockchain_model')({ cooler, isPublic: config.public });
  187. const jobsModel = require('../models/jobs_model')({ cooler, isPublic: config.public });
  188. // starting warmup
  189. about._startNameWarmup();
  190. async function renderBlobMarkdown(text, mentions = {}, myFeedId, myUsername) {
  191. if (!text) return '';
  192. const mentionByFeed = {};
  193. Object.values(mentions).forEach(arr => {
  194. arr.forEach(m => {
  195. mentionByFeed[m.feed] = m;
  196. });
  197. });
  198. text = text.replace(/\[@([^\]]+)\]\(([^)]+)\)/g, (_, name, id) => {
  199. return `<a class="mention" href="/author/${encodeURIComponent(id)}">@${myUsername}</a>`;
  200. });
  201. const mentionRegex = /@([A-Za-z0-9_\-\.+=\/]+\.ed25519)/g;
  202. const words = text.split(' ');
  203. text = (await Promise.all(
  204. words.map(async (word) => {
  205. const match = mentionRegex.exec(word);
  206. if (match && match[1]) {
  207. const feedId = match[1];
  208. if (feedId === myFeedId) {
  209. return `<a class="mention" href="/author/${encodeURIComponent(feedId)}">@${myUsername}</a>`;
  210. }
  211. }
  212. return word;
  213. })
  214. )).join(' ');
  215. text = text
  216. .replace(/!\[image:[^\]]+\]\(([^)]+)\)/g, (_, id) =>
  217. `<img src="/blob/${encodeURIComponent(id)}" alt="image" class="post-image" />`)
  218. .replace(/\[audio:[^\]]+\]\(([^)]+)\)/g, (_, id) =>
  219. `<audio controls class="post-audio" src="/blob/${encodeURIComponent(id)}"></audio>`)
  220. .replace(/\[video:[^\]]+\]\(([^)]+)\)/g, (_, id) =>
  221. `<video controls class="post-video" src="/blob/${encodeURIComponent(id)}"></video>`)
  222. .replace(/\[pdf:[^\]]+\]\(([^)]+)\)/g, (_, id) =>
  223. `<a class="post-pdf" href="/blob/${encodeURIComponent(id)}" target="_blank">PDF</a>`);
  224. return text;
  225. }
  226. let formattedTextCache = null;
  227. const preparePreview = async function (ctx) {
  228. let text = String(ctx.request.body.text || "");
  229. const mentions = {};
  230. const rex = /(^|\s)(?!\[)@([a-zA-Z0-9\-/.=+]{3,})\b/g;
  231. let m;
  232. while ((m = rex.exec(text)) !== null) {
  233. const token = m[2];
  234. const key = token;
  235. let found = mentions[key] || [];
  236. if (/\.ed25519$/.test(token)) {
  237. const name = await about.name(token);
  238. const img = await about.image(token);
  239. found.push({
  240. feed: token,
  241. name,
  242. img,
  243. rel: { followsMe: false, following: false, blocking: false, me: false }
  244. });
  245. } else {
  246. const matches = about.named(token);
  247. for (const match of matches) {
  248. found.push(match);
  249. }
  250. }
  251. if (found.length > 0) {
  252. mentions[key] = found;
  253. }
  254. }
  255. Object.keys(mentions).forEach((key) => {
  256. let matches = mentions[key];
  257. const meaningful = matches.filter((m) => (m.rel?.followsMe || m.rel?.following) && !m.rel?.blocking);
  258. mentions[key] = meaningful.length > 0 ? meaningful : matches;
  259. });
  260. const replacer = (match, prefix, token) => {
  261. const matches = mentions[token];
  262. if (matches && matches.length === 1) {
  263. return `${prefix}[@${matches[0].name}](${matches[0].feed})`;
  264. }
  265. return match;
  266. };
  267. text = text.replace(rex, replacer);
  268. const blobMarkdown = await handleBlobUpload(ctx, "blob");
  269. if (blobMarkdown) {
  270. text += blobMarkdown;
  271. }
  272. const ssbClient = await cooler.open();
  273. const authorMeta = {
  274. id: ssbClient.id,
  275. name: await about.name(ssbClient.id),
  276. image: await about.image(ssbClient.id),
  277. };
  278. const renderedText = await renderBlobMarkdown(text, mentions, authorMeta.id, authorMeta.name);
  279. const hasBrTags = /<br\s*\/?>/.test(renderedText);
  280. const formattedText = formattedTextCache || (!hasBrTags ? renderedText.replace(/\n/g, '<br>') : renderedText);
  281. if (!formattedTextCache && !hasBrTags) {
  282. formattedTextCache = formattedText;
  283. }
  284. const contentWarning = ctx.request.body.contentWarning || '';
  285. let finalContent = formattedText;
  286. if (contentWarning && !finalContent.startsWith(contentWarning)) {
  287. finalContent = `<br>${finalContent}`;
  288. }
  289. return { authorMeta, text: renderedText, formattedText: finalContent, mentions };
  290. };
  291. // set koaMiddleware maxSize: 50 MiB (voted by community at: 09/04/2025)
  292. const megabyte = Math.pow(2, 20);
  293. const maxSize = 50 * megabyte;
  294. // koaMiddleware to manage files
  295. const homeDir = os.homedir();
  296. const blobsPath = path.join(homeDir, '.ssb', 'blobs', 'tmp');
  297. const koaBodyMiddleware = koaBody({
  298. multipart: true,
  299. formidable: {
  300. uploadDir: blobsPath,
  301. keepExtensions: true,
  302. maxFieldsSize: maxSize,
  303. hash: 'sha256',
  304. },
  305. parsedMethods: ['POST'],
  306. });
  307. const resolveCommentComponents = async function (ctx) {
  308. let parentId;
  309. try {
  310. parentId = decodeURIComponent(ctx.params.message);
  311. } catch {
  312. parentId = ctx.params.message;
  313. }
  314. const parentMessage = await post.get(parentId);
  315. if (!parentMessage || !parentMessage.value) {
  316. throw new Error("Invalid parentMessage or missing 'value'");
  317. }
  318. const myFeedId = await meta.myFeedId();
  319. const hasRoot =
  320. typeof parentMessage?.value?.content?.root === "string" &&
  321. ssbRef.isMsg(parentMessage.value.content.root);
  322. const hasFork =
  323. typeof parentMessage?.value?.content?.fork === "string" &&
  324. ssbRef.isMsg(parentMessage.value.content.fork);
  325. const rootMessage = hasRoot
  326. ? hasFork
  327. ? parentMessage
  328. : await post.get(parentMessage.value.content.root)
  329. : parentMessage;
  330. const messages = await post.topicComments(rootMessage.key);
  331. messages.push(rootMessage);
  332. let contentWarning;
  333. if (ctx.request.body) {
  334. const rawContentWarning = String(ctx.request.body.contentWarning || "").trim();
  335. contentWarning = rawContentWarning.length > 0 ? rawContentWarning : undefined;
  336. }
  337. return { messages, myFeedId, parentMessage, contentWarning };
  338. };
  339. // import views (core)
  340. const { authorView, previewCommentView, commentView, editProfileView, extendedView, latestView, likesView, threadView, hashtagView, mentionsView, popularView, previewView, privateView, publishCustomView, publishView, previewSubtopicView, subtopicView, imageSearchView, setLanguage, topicsView, summaryView, threadsView } = require("../views/main_views");
  341. // import views (modules)
  342. const { activityView } = require("../views/activity_view");
  343. const { cvView, createCVView } = require("../views/cv_view");
  344. const { indexingView } = require("../views/indexing_view");
  345. const { pixeliaView } = require("../views/pixelia_view");
  346. const { statsView } = require("../views/stats_view");
  347. const { tribesView, tribeDetailView, tribesInvitesView, tribeView, renderInvitePage } = require("../views/tribes_view");
  348. const { agendaView } = require("../views/agenda_view");
  349. const { documentView, singleDocumentView } = require("../views/document_view");
  350. const { inhabitantsView, inhabitantsProfileView } = require("../views/inhabitants_view");
  351. const { walletViewRender, walletView, walletHistoryView, walletReceiveView, walletSendFormView, walletSendConfirmView, walletSendResultView, walletErrorView } = require("../views/wallet_view");
  352. const { pmView } = require("../views/pm_view");
  353. const { tagsView } = require("../views/tags_view");
  354. const { videoView, singleVideoView } = require("../views/video_view");
  355. const { audioView, singleAudioView } = require("../views/audio_view");
  356. const { eventView, singleEventView } = require("../views/event_view");
  357. const { invitesView } = require("../views/invites_view");
  358. const { modulesView } = require("../views/modules_view");
  359. const { reportView, singleReportView } = require("../views/report_view");
  360. const { taskView, singleTaskView } = require("../views/task_view");
  361. const { voteView } = require("../views/vote_view");
  362. const { bookmarkView, singleBookmarkView } = require("../views/bookmark_view");
  363. const { feedView, feedCreateView } = require("../views/feed_view");
  364. const { legacyView } = require("../views/legacy_view");
  365. const { opinionsView } = require("../views/opinions_view");
  366. const { peersView } = require("../views/peers_view");
  367. const { searchView } = require("../views/search_view");
  368. const { transferView, singleTransferView } = require("../views/transfer_view");
  369. const { cipherView } = require("../views/cipher_view");
  370. const { imageView, singleImageView } = require("../views/image_view");
  371. const { settingsView } = require("../views/settings_view");
  372. const { trendingView } = require("../views/trending_view");
  373. const { marketView, singleMarketView } = require("../views/market_view");
  374. const { aiView } = require("../views/AI_view");
  375. const { forumView, singleForumView } = require("../views/forum_view");
  376. const { renderBlockchainView, renderSingleBlockView } = require("../views/blockchain_view");
  377. const { jobsView, singleJobsView, renderJobForm } = require("../views/jobs_view");
  378. let sharp;
  379. try {
  380. sharp = require("sharp");
  381. } catch (e) {
  382. // Optional dependency
  383. }
  384. const readmePath = path.join(__dirname, "..", ".." ,"README.md");
  385. const packagePath = path.join(__dirname, "..", "server", "package.json");
  386. const readme = fs.readFileSync(readmePath, "utf8");
  387. const version = JSON.parse(fs.readFileSync(packagePath, "utf8")).version;
  388. const nullImageId = '&0000000000000000000000000000000000000000000=.sha256';
  389. const getAvatarUrl = (image) => {
  390. if (!image || image === nullImageId) {
  391. return '/assets/images/default-avatar.png';
  392. }
  393. return `/image/256/${encodeURIComponent(image)}`;
  394. };
  395. router
  396. .param("imageSize", (imageSize, ctx, next) => {
  397. const size = Number(imageSize);
  398. const isInteger = size % 1 === 0;
  399. const overMinSize = size > 2;
  400. const underMaxSize = size <= 256;
  401. ctx.assert(
  402. isInteger && overMinSize && underMaxSize,
  403. 400,
  404. "Invalid image size"
  405. );
  406. return next();
  407. })
  408. .param("blobId", (blobId, ctx, next) => {
  409. ctx.assert(ssbRef.isBlob(blobId), 400, "Invalid blob link");
  410. return next();
  411. })
  412. .param("message", (message, ctx, next) => {
  413. ctx.assert(ssbRef.isMsg(message), 400, "Invalid message link");
  414. return next();
  415. })
  416. .param("feed", (message, ctx, next) => {
  417. ctx.assert(ssbRef.isFeedId(message), 400, "Invalid feed link");
  418. return next();
  419. })
  420. //GET backend routes
  421. .get("/", async (ctx) => {
  422. ctx.redirect("/activity"); // default view when starting Oasis
  423. })
  424. .get("/robots.txt", (ctx) => {
  425. ctx.body = "User-agent: *\nDisallow: /";
  426. })
  427. .get(oasisCheckPath, (ctx) => {
  428. ctx.body = "oasis";
  429. })
  430. .get('/stats', async ctx => {
  431. const filter = ctx.query.filter || 'ALL';
  432. const stats = await statsModel.getStats(filter);
  433. ctx.body = statsView(stats, filter);
  434. })
  435. .get("/public/popular/:period", async (ctx) => {
  436. const { period } = ctx.params;
  437. const popularMod = ctx.cookies.get("popularMod") || 'on';
  438. if (popularMod !== 'on') {
  439. ctx.redirect('/modules');
  440. return;
  441. }
  442. const i18n = require("../client/assets/translations/i18n");
  443. const lang = ctx.cookies.get('lang') || 'en';
  444. const translations = i18n[lang] || i18n['en'];
  445. const publicPopular = async ({ period }) => {
  446. const messages = await post.popular({ period });
  447. const prefix = nav(
  448. div({ class: "filters" },
  449. ul(
  450. li(
  451. form({ method: "GET", action: "/public/popular/day" },
  452. button({ type: "submit", class: "filter-btn" }, translations.day)
  453. )
  454. ),
  455. li(
  456. form({ method: "GET", action: "/public/popular/week" },
  457. button({ type: "submit", class: "filter-btn" }, translations.week)
  458. )
  459. ),
  460. li(
  461. form({ method: "GET", action: "/public/popular/month" },
  462. button({ type: "submit", class: "filter-btn" }, translations.month)
  463. )
  464. ),
  465. li(
  466. form({ method: "GET", action: "/public/popular/year" },
  467. button({ type: "submit", class: "filter-btn" }, translations.year)
  468. )
  469. )
  470. )
  471. )
  472. );
  473. return popularView({
  474. messages,
  475. prefix,
  476. });
  477. };
  478. ctx.body = await publicPopular({ period });
  479. })
  480. // modules
  481. .get("/modules", async (ctx) => {
  482. const configMods = getConfig().modules;
  483. const modules = [
  484. 'popular', 'topics', 'summaries', 'latest', 'threads', 'multiverse', 'invites', 'wallet',
  485. 'legacy', 'cipher', 'bookmarks', 'videos', 'docs', 'audios', 'tags', 'images', 'trending',
  486. 'events', 'tasks', 'market', 'tribes', 'governance', 'reports', 'opinions', 'transfers',
  487. 'feed', 'pixelia', 'agenda', 'ai', 'forum', 'jobs'
  488. ];
  489. const moduleStates = modules.reduce((acc, mod) => {
  490. acc[`${mod}Mod`] = configMods[`${mod}Mod`];
  491. return acc;
  492. }, {});
  493. ctx.body = modulesView(moduleStates);
  494. })
  495. // AI
  496. .get('/ai', async (ctx) => {
  497. const aiMod = ctx.cookies.get("aiMod") || 'on';
  498. if (aiMod !== 'on') {
  499. ctx.redirect('/modules');
  500. return;
  501. }
  502. startAI();
  503. const historyPath = path.join(__dirname, '..', '..', 'src', 'configs', 'AI-history.json');
  504. let chatHistory = [];
  505. try {
  506. const fileData = fs.readFileSync(historyPath, 'utf-8');
  507. chatHistory = JSON.parse(fileData);
  508. } catch (e) {
  509. chatHistory = [];
  510. }
  511. const config = getConfig();
  512. const userPrompt = config.ai?.prompt?.trim() || '';
  513. ctx.body = aiView(chatHistory, userPrompt);
  514. })
  515. // pixelArt
  516. .get('/pixelia', async (ctx) => {
  517. const pixeliaMod = ctx.cookies.get("pixeliaMod") || 'on';
  518. if (pixeliaMod !== 'on') {
  519. ctx.redirect('/modules');
  520. return;
  521. }
  522. const pixelArt = await pixeliaModel.listPixels();
  523. ctx.body = pixeliaView(pixelArt);
  524. })
  525. // blockexplorer
  526. .get('/blockexplorer', async (ctx) => {
  527. const userId = SSBconfig.config.keys.id;
  528. const query = ctx.query;
  529. const filter = query.filter || 'recent';
  530. const blockchainData = await blockchainModel.listBlockchain(filter, userId);
  531. ctx.body = renderBlockchainView(blockchainData, filter, userId);
  532. })
  533. .get('/blockexplorer/block/:id', async (ctx) => {
  534. const blockId = ctx.params.id;
  535. const block = await blockchainModel.getBlockById(blockId);
  536. ctx.body = renderSingleBlockView(block);
  537. })
  538. .get("/public/latest", async (ctx) => {
  539. const latestMod = ctx.cookies.get("latestMod") || 'on';
  540. if (latestMod !== 'on') {
  541. ctx.redirect('/modules');
  542. return;
  543. }
  544. const messages = await post.latest();
  545. ctx.body = await latestView({ messages });
  546. })
  547. .get("/public/latest/extended", async (ctx) => {
  548. const extendedMod = ctx.cookies.get("extendedMod") || 'on';
  549. if (extendedMod !== 'on') {
  550. ctx.redirect('/modules');
  551. return;
  552. }
  553. const messages = await post.latestExtended();
  554. ctx.body = await extendedView({ messages });
  555. })
  556. .get("/public/latest/topics", async (ctx) => {
  557. const topicsMod = ctx.cookies.get("topicsMod") || 'on';
  558. if (topicsMod !== 'on') {
  559. ctx.redirect('/modules');
  560. return;
  561. }
  562. const messages = await post.latestTopics();
  563. const channels = await post.channels();
  564. const list = channels.map((c) => {
  565. return li(a({ href: `/hashtag/${c}` }, `#${c}`));
  566. });
  567. const prefix = nav(ul(list));
  568. ctx.body = await topicsView({ messages, prefix });
  569. })
  570. .get("/public/latest/summaries", async (ctx) => {
  571. const summariesMod = ctx.cookies.get("summariesMod") || 'on';
  572. if (summariesMod !== 'on') {
  573. ctx.redirect('/modules');
  574. return;
  575. }
  576. const messages = await post.latestSummaries();
  577. ctx.body = await summaryView({ messages });
  578. })
  579. .get("/public/latest/threads", async (ctx) => {
  580. const threadsMod = ctx.cookies.get("threadsMod") || 'on';
  581. if (threadsMod !== 'on') {
  582. ctx.redirect('/modules');
  583. return;
  584. }
  585. const messages = await post.latestThreads();
  586. ctx.body = await threadsView({ messages });
  587. })
  588. .get("/author/:feed", async (ctx) => {
  589. const { feed } = ctx.params;
  590. const gt = Number(ctx.request.query["gt"] || -1);
  591. const lt = Number(ctx.request.query["lt"] || -1);
  592. if (lt > 0 && gt > 0 && gt >= lt)
  593. throw new Error("Given search range is empty");
  594. const author = async (feedId) => {
  595. const description = await about.description(feedId);
  596. const name = await about.name(feedId);
  597. const image = await about.image(feedId);
  598. const messages = await post.fromPublicFeed(feedId, gt, lt);
  599. const firstPost = await post.firstBy(feedId);
  600. const lastPost = await post.latestBy(feedId);
  601. const relationship = await friend.getRelationship(feedId);
  602. const avatarUrl = getAvatarUrl(image);
  603. return authorView({
  604. feedId,
  605. messages,
  606. firstPost,
  607. lastPost,
  608. name,
  609. description,
  610. avatarUrl,
  611. relationship,
  612. });
  613. };
  614. ctx.body = await author(feed);
  615. })
  616. .get("/search", async (ctx) => {
  617. const query = ctx.query.query || '';
  618. if (!query) {
  619. return ctx.body = await searchView({ messages: [], query, types: [] });
  620. }
  621. const results = await searchModel.search({ query, types: [] });
  622. const groupedResults = Object.entries(results).reduce((acc, [type, msgs]) => {
  623. acc[type] = msgs.map(msg => {
  624. if (!msg.value || !msg.value.content) {
  625. return {};
  626. }
  627. return {
  628. ...msg,
  629. content: msg.value.content,
  630. author: msg.value.content.author || 'Unknown',
  631. };
  632. });
  633. return acc;
  634. }, {});
  635. ctx.body = await searchView({ results: groupedResults, query, types: [] });
  636. })
  637. .get('/images', async (ctx) => {
  638. const imagesMod = ctx.cookies.get("imagesMod") || 'on';
  639. if (imagesMod !== 'on') {
  640. ctx.redirect('/modules');
  641. return;
  642. }
  643. const filter = ctx.query.filter || 'all'
  644. const images = await imagesModel.listAll(filter);
  645. ctx.body = await imageView(images, filter, null);
  646. })
  647. .get('/images/edit/:id', async ctx => {
  648. const imageId = ctx.params.id;
  649. const images = await imagesModel.listAll('all');
  650. ctx.body = await imageView(images, 'edit', imageId);
  651. })
  652. .get('/images/:imageId', async ctx => {
  653. const imageId = ctx.params.imageId;
  654. const filter = ctx.query.filter || 'all';
  655. const image = await imagesModel.getImageById(imageId);
  656. ctx.body = await singleImageView(image, filter);
  657. })
  658. .get('/audios', async (ctx) => {
  659. const audiosMod = ctx.cookies.get("audiosMod") || 'on';
  660. if (audiosMod !== 'on') {
  661. ctx.redirect('/modules');
  662. return;
  663. }
  664. const filter = ctx.query.filter || 'all';
  665. const audios = await audiosModel.listAll(filter);
  666. ctx.body = await audioView(audios, filter, null);
  667. })
  668. .get('/audios/edit/:id', async (ctx) => {
  669. const audiosMod = ctx.cookies.get("audiosMod") || 'on';
  670. if (audiosMod !== 'on') {
  671. ctx.redirect('/modules');
  672. return;
  673. }
  674. const audio = await audiosModel.getAudioById(ctx.params.id);
  675. ctx.body = await audioView([audio], 'edit', ctx.params.id);
  676. })
  677. .get('/audios/:audioId', async ctx => {
  678. const audioId = ctx.params.audioId;
  679. const filter = ctx.query.filter || 'all';
  680. const audio = await audiosModel.getAudioById(audioId);
  681. ctx.body = await singleAudioView(audio, filter);
  682. })
  683. .get('/videos', async (ctx) => {
  684. const filter = ctx.query.filter || 'all';
  685. const videos = await videosModel.listAll(filter);
  686. ctx.body = await videoView(videos, filter, null);
  687. })
  688. .get('/videos/edit/:id', async (ctx) => {
  689. const video = await videosModel.getVideoById(ctx.params.id);
  690. ctx.body = await videoView([video], 'edit', ctx.params.id);
  691. })
  692. .get('/videos/:videoId', async ctx => {
  693. const videoId = ctx.params.videoId;
  694. const filter = ctx.query.filter || 'all';
  695. const video = await videosModel.getVideoById(videoId);
  696. ctx.body = await singleVideoView(video, filter);
  697. })
  698. .get('/documents', async (ctx) => {
  699. const filter = ctx.query.filter || 'all';
  700. const documents = await documentsModel.listAll(filter);
  701. ctx.body = await documentView(documents, filter, null);
  702. })
  703. .get('/documents/edit/:id', async (ctx) => {
  704. const document = await documentsModel.getDocumentById(ctx.params.id);
  705. ctx.body = await documentView([document], 'edit', ctx.params.id);
  706. })
  707. .get('/documents/:documentId', async ctx => {
  708. const documentId = ctx.params.documentId;
  709. const filter = ctx.query.filter || 'all';
  710. const document = await documentsModel.getDocumentById(documentId);
  711. ctx.body = await singleDocumentView(document, filter);
  712. })
  713. .get('/cv', async ctx => {
  714. const cv = await cvModel.getCVByUserId()
  715. ctx.body = await cvView(cv)
  716. })
  717. .get('/cv/create', async ctx => {
  718. ctx.body = await createCVView()
  719. })
  720. .get('/cv/edit/:id', async ctx => {
  721. const cv = await cvModel.getCVByUserId()
  722. ctx.body = await createCVView(cv, true)
  723. })
  724. .get('/pm', async ctx => {
  725. const { recipients = '' } = ctx.query;
  726. ctx.body = await pmView(recipients);
  727. })
  728. .get("/inbox", async (ctx) => {
  729. const inboxMod = ctx.cookies.get("inboxMod") || 'on';
  730. if (inboxMod !== 'on') {
  731. ctx.redirect('/modules');
  732. return;
  733. }
  734. const inboxMessages = async () => {
  735. const messages = await post.inbox();
  736. return privateView({ messages });
  737. };
  738. ctx.body = await inboxMessages();
  739. })
  740. .get('/tags', async ctx => {
  741. const filter = ctx.query.filter || 'all'
  742. const tags = await tagsModel.listTags(filter)
  743. ctx.body = await tagsView(tags, filter)
  744. })
  745. .get('/reports', async ctx => {
  746. const filter = ctx.query.filter || 'all';
  747. const reports = await reportsModel.listAll(filter);
  748. ctx.body = await reportView(reports, filter, null);
  749. })
  750. .get('/reports/edit/:id', async ctx => {
  751. const report = await reportsModel.getReportById(ctx.params.id);
  752. ctx.body = await reportView([report], 'edit', ctx.params.id);
  753. })
  754. .get('/reports/:reportId', async ctx => {
  755. const reportId = ctx.params.reportId;
  756. const filter = ctx.query.filter || 'all';
  757. const report = await reportsModel.getReportById(reportId, filter);
  758. ctx.body = await singleReportView(report, filter);
  759. })
  760. .get('/trending', async (ctx) => {
  761. const filter = ctx.query.filter || 'RECENT';
  762. const trendingItems = await trendingModel.listTrending(filter);
  763. const items = trendingItems.filtered || [];
  764. const categories = trendingModel.categories;
  765. ctx.body = await trendingView(items, filter, categories);
  766. })
  767. .get('/agenda', async (ctx) => {
  768. const filter = ctx.query.filter || 'all';
  769. const data = await agendaModel.listAgenda(filter);
  770. ctx.body = await agendaView(data, filter);
  771. })
  772. .get("/hashtag/:hashtag", async (ctx) => {
  773. const { hashtag } = ctx.params;
  774. const messages = await post.fromHashtag(hashtag);
  775. ctx.body = await hashtagView({ hashtag, messages });
  776. })
  777. .get('/inhabitants', async (ctx) => {
  778. const filter = ctx.query.filter || 'all';
  779. const query = {
  780. search: ctx.query.search || ''
  781. };
  782. if (['CVs', 'MATCHSKILLS'].includes(filter)) {
  783. query.location = ctx.query.location || '';
  784. query.language = ctx.query.language || '';
  785. query.skills = ctx.query.skills || '';
  786. }
  787. const userId = SSBconfig.config.keys.id;
  788. const inhabitants = await inhabitantsModel.listInhabitants({
  789. filter,
  790. ...query
  791. });
  792. ctx.body = await inhabitantsView(inhabitants, filter, query, userId);
  793. })
  794. .get('/inhabitant/:id', async (ctx) => {
  795. const id = ctx.params.id;
  796. const about = await inhabitantsModel.getLatestAboutById(id);
  797. const cv = await inhabitantsModel.getCVByUserId(id);
  798. const feed = await inhabitantsModel.getFeedByUserId(id);
  799. const currentUserId = SSBconfig.config.keys.id;
  800. ctx.body = await inhabitantsProfileView({ about, cv, feed }, currentUserId);
  801. })
  802. .get('/tribes', async ctx => {
  803. const filter = ctx.query.filter || 'all';
  804. const search = ctx.query.search || '';
  805. const tribes = await tribesModel.listAll();
  806. let filteredTribes = tribes;
  807. if (search) {
  808. filteredTribes = tribes.filter(tribe =>
  809. tribe.title.toLowerCase().includes(search.toLowerCase())
  810. );
  811. }
  812. ctx.body = await tribesView(filteredTribes, filter, null, ctx.query);
  813. })
  814. .get('/tribes/create', async ctx => {
  815. ctx.body = await tribesView([], 'create', null)
  816. })
  817. .get('/tribes/edit/:id', async ctx => {
  818. const tribe = await tribesModel.getTribeById(ctx.params.id)
  819. ctx.body = await tribesView([tribe], 'edit', ctx.params.id)
  820. })
  821. .get('/tribe/:tribeId', koaBody(), async ctx => {
  822. const tribeId = ctx.params.tribeId;
  823. const tribe = await tribesModel.getTribeById(tribeId);
  824. const userId = SSBconfig.config.keys.id;
  825. const query = ctx.query;
  826. if (!query.feedFilter) {
  827. query.feedFilter = 'TOP';
  828. }
  829. if (tribe.isAnonymous === false && !tribe.members.includes(userId)) {
  830. ctx.status = 403;
  831. ctx.body = { message: 'You cannot access to this tribe!' };
  832. return;
  833. }
  834. if (!tribe.members.includes(userId)) {
  835. ctx.status = 403;
  836. ctx.body = { message: 'You cannot access to this tribe!' };
  837. return;
  838. }
  839. ctx.body = await tribeView(tribe, userId, query);
  840. })
  841. .get('/activity', async ctx => {
  842. const filter = ctx.query.filter || 'recent';
  843. const actions = await activityModel.listFeed(filter);
  844. const userId = SSBconfig.config.keys.id;
  845. ctx.body = activityView(actions, filter, userId);
  846. })
  847. .get("/profile", async (ctx) => {
  848. const myFeedId = await meta.myFeedId();
  849. const gt = Number(ctx.request.query["gt"] || -1);
  850. const lt = Number(ctx.request.query["lt"] || -1);
  851. if (lt > 0 && gt > 0 && gt >= lt)
  852. throw new Error("Given search range is empty");
  853. const description = await about.description(myFeedId);
  854. const name = await about.name(myFeedId);
  855. const image = await about.image(myFeedId);
  856. const messages = await post.fromPublicFeed(myFeedId, gt, lt);
  857. const firstPost = await post.firstBy(myFeedId);
  858. const lastPost = await post.latestBy(myFeedId);
  859. const avatarUrl = getAvatarUrl(image);
  860. ctx.body = await authorView({
  861. feedId: myFeedId,
  862. messages,
  863. firstPost,
  864. lastPost,
  865. name,
  866. description,
  867. avatarUrl,
  868. relationship: { me: true },
  869. });
  870. })
  871. .get("/profile/edit", async (ctx) => {
  872. const myFeedId = await meta.myFeedId();
  873. const description = await about.description(myFeedId);
  874. const name = await about.name(myFeedId);
  875. ctx.body = await editProfileView({
  876. name,
  877. description,
  878. });
  879. })
  880. .post("/profile/edit", koaBody({ multipart: true }), async (ctx) => {
  881. const name = String(ctx.request.body.name);
  882. const description = String(ctx.request.body.description);
  883. const image = await promisesFs.readFile(ctx.request.files.image.filepath);
  884. ctx.body = await post.publishProfileEdit({
  885. name,
  886. description,
  887. image,
  888. });
  889. ctx.redirect("/profile");
  890. })
  891. .get("/publish/custom", async (ctx) => {
  892. ctx.body = await publishCustomView();
  893. })
  894. .get("/json/:message", async (ctx) => {
  895. if (config.public) {
  896. throw new Error(
  897. "Sorry, many actions are unavailable when Oasis is running in public mode. Please run Oasis in the default mode and try again."
  898. );
  899. }
  900. const { message } = ctx.params;
  901. ctx.type = "application/json";
  902. const json = async (message) => {
  903. const json = await meta.get(message);
  904. return JSON.stringify(json, null, 2);
  905. };
  906. ctx.body = await json(message);
  907. })
  908. .get("/blob/:blobId", async (ctx) => {
  909. const { blobId } = ctx.params;
  910. const id = blobId.startsWith('&') ? blobId : `&${blobId}`;
  911. const buffer = await blob.getResolved({ blobId });
  912. let fileType;
  913. try {
  914. fileType = await FileType.fromBuffer(buffer);
  915. } catch {
  916. fileType = null;
  917. }
  918. let mime = fileType?.mime || "application/octet-stream";
  919. if (mime === "application/octet-stream" && buffer.slice(0, 4).toString() === "%PDF") {
  920. mime = "application/pdf";
  921. }
  922. ctx.set("Content-Type", mime);
  923. ctx.set("Content-Disposition", `inline; filename="${blobId}"`);
  924. ctx.set("Cache-Control", "public, max-age=31536000, immutable");
  925. ctx.body = buffer;
  926. })
  927. .get("/image/:imageSize/:blobId", async (ctx) => {
  928. const { blobId, imageSize } = ctx.params;
  929. const size = Number(imageSize);
  930. const fallbackPixel = Buffer.from(
  931. "iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAQAAAC1HAwCAAAAC0lEQVR42mNk+A8AAQUBAScY42YAAAAASUVORK5CYII=",
  932. "base64"
  933. );
  934. const fakeImage = () => {
  935. if (typeof sharp !== "function") {
  936. return Promise.resolve(fallbackPixel);
  937. }
  938. return sharp({
  939. create: {
  940. width: size,
  941. height: size,
  942. channels: 4,
  943. background: { r: 0, g: 0, b: 0, alpha: 0.5 },
  944. },
  945. }).png().toBuffer();
  946. };
  947. try {
  948. const buffer = await blob.getResolved({ blobId });
  949. if (!buffer) {
  950. ctx.set("Content-Type", "image/png");
  951. ctx.body = await fakeImage();
  952. return;
  953. }
  954. const fileType = await FileType.fromBuffer(buffer);
  955. const mimeType = fileType?.mime || "application/octet-stream";
  956. ctx.set("Content-Type", mimeType);
  957. if (typeof sharp === "function") {
  958. ctx.body = await sharp(buffer)
  959. .resize(size, size)
  960. .png()
  961. .toBuffer();
  962. } else {
  963. ctx.body = buffer;
  964. }
  965. } catch (err) {
  966. ctx.set("Content-Type", "image/png");
  967. ctx.body = await fakeImage();
  968. }
  969. })
  970. .get("/settings", async (ctx) => {
  971. const theme = ctx.cookies.get("theme") || "Dark-SNH";
  972. const config = getConfig();
  973. const aiPrompt = config.ai?.prompt || "";
  974. const getMeta = async ({ theme, aiPrompt }) => {
  975. return settingsView({
  976. theme,
  977. version: version.toString(),
  978. aiPrompt
  979. });
  980. };
  981. ctx.body = await getMeta({ theme, aiPrompt });
  982. })
  983. .get("/peers", async (ctx) => {
  984. const theme = ctx.cookies.get("theme") || config.theme;
  985. const getMeta = async () => {
  986. const allPeers = await meta.peers();
  987. const connected = allPeers.filter(([, data]) => data.state === "connected");
  988. const offline = allPeers.filter(([, data]) => data.state !== "connected");
  989. const enrich = async (peers) => {
  990. return await Promise.all(
  991. peers.map(async ([address, data]) => {
  992. const feedId = data.key || data.id;
  993. const name = await about.name(feedId);
  994. return [
  995. address,
  996. {
  997. ...data,
  998. key: feedId,
  999. name: name || feedId,
  1000. },
  1001. ];
  1002. })
  1003. );
  1004. };
  1005. const connectedPeers = await enrich(connected);
  1006. const offlinePeers = await enrich(offline);
  1007. return peersView({
  1008. connectedPeers,
  1009. peers: offlinePeers,
  1010. });
  1011. };
  1012. ctx.body = await getMeta();
  1013. })
  1014. .get("/invites", async (ctx) => {
  1015. const theme = ctx.cookies.get("theme") || config.theme;
  1016. const invitesMod = ctx.cookies.get("invitesMod") || 'on';
  1017. if (invitesMod !== 'on') {
  1018. ctx.redirect('/modules');
  1019. return;
  1020. }
  1021. const getMeta = async ({ theme }) => {
  1022. return invitesView({});
  1023. };
  1024. ctx.body = await getMeta({ theme });
  1025. })
  1026. .get("/likes/:feed", async (ctx) => {
  1027. const { feed } = ctx.params;
  1028. const likes = async ({ feed }) => {
  1029. const pendingMessages = post.likes({ feed });
  1030. const pendingName = about.name(feed);
  1031. return likesView({
  1032. messages: await pendingMessages,
  1033. feed,
  1034. name: await pendingName,
  1035. });
  1036. };
  1037. ctx.body = await likes({ feed });
  1038. })
  1039. .get("/mentions", async (ctx) => {
  1040. const { messages, myFeedId } = await post.mentionsMe();
  1041. ctx.body = await mentionsView({ messages, myFeedId });
  1042. })
  1043. .get('/opinions', async (ctx) => {
  1044. const filter = ctx.query.filter || 'RECENT';
  1045. const opinions = await opinionsModel.listOpinions(filter);
  1046. ctx.body = await opinionsView(opinions, filter);
  1047. })
  1048. .get('/feed', async ctx => {
  1049. const filter = ctx.query.filter || 'ALL';
  1050. const feeds = await feedModel.listFeeds(filter);
  1051. ctx.body = feedView(feeds, filter);
  1052. })
  1053. .get('/feed/create', async ctx => {
  1054. ctx.body = feedCreateView();
  1055. })
  1056. .get('/forum', async ctx => {
  1057. const forumMod = ctx.cookies.get("forumMod") || 'on';
  1058. if (forumMod !== 'on') {
  1059. ctx.redirect('/modules');
  1060. return;
  1061. }
  1062. const filter = ctx.query.filter || 'hot';
  1063. const forums = await forumModel.listAll(filter);
  1064. ctx.body = await forumView(forums, filter);
  1065. })
  1066. .get('/forum/:forumId', async ctx => {
  1067. const rawId = ctx.params.forumId
  1068. const msg = await forumModel.getMessageById(rawId)
  1069. const isReply = Boolean(msg.root)
  1070. const forumId = isReply ? msg.root : rawId
  1071. const highlightCommentId = isReply ? rawId : null
  1072. const forum = await forumModel.getForumById(forumId)
  1073. const messagesData = await forumModel.getMessagesByForumId(forumId)
  1074. ctx.body = await singleForumView(
  1075. forum,
  1076. messagesData,
  1077. ctx.query.filter,
  1078. highlightCommentId
  1079. )
  1080. })
  1081. .get('/legacy', async (ctx) => {
  1082. const legacyMod = ctx.cookies.get("legacyMod") || 'on';
  1083. if (legacyMod !== 'on') {
  1084. ctx.redirect('/modules');
  1085. return;
  1086. }
  1087. try {
  1088. ctx.body = await legacyView();
  1089. } catch (error) {
  1090. ctx.body = { error: error.message };
  1091. }
  1092. })
  1093. .get('/bookmarks', async (ctx) => {
  1094. const bookmarksMod = ctx.cookies.get("bookmarksMod") || 'on';
  1095. if (bookmarksMod !== 'on') {
  1096. ctx.redirect('/modules');
  1097. return;
  1098. }
  1099. const filter = ctx.query.filter || 'all';
  1100. const bookmarks = await bookmarksModel.listAll(null, filter);
  1101. ctx.body = await bookmarkView(bookmarks, filter, null);
  1102. })
  1103. .get('/bookmarks/edit/:id', async (ctx) => {
  1104. const bookmarksMod = ctx.cookies.get("bookmarksMod") || 'on';
  1105. if (bookmarksMod !== 'on') {
  1106. ctx.redirect('/modules');
  1107. return;
  1108. }
  1109. const bookmarkId = ctx.params.id;
  1110. const bookmark = await bookmarksModel.getBookmarkById(bookmarkId);
  1111. if (bookmark.opinions_inhabitants && bookmark.opinions_inhabitants.length > 0) {
  1112. ctx.flash = { message: "This bookmark has received votes and cannot be updated." };
  1113. ctx.redirect(`/bookmarks?filter=mine`);
  1114. }
  1115. ctx.body = await bookmarkView([bookmark], 'edit', bookmarkId);
  1116. })
  1117. .get('/bookmarks/:bookmarkId', async ctx => {
  1118. const bookmarkId = ctx.params.bookmarkId;
  1119. const filter = ctx.query.filter || 'all';
  1120. const bookmark = await bookmarksModel.getBookmarkById(bookmarkId);
  1121. ctx.body = await singleBookmarkView(bookmark, filter);
  1122. })
  1123. .get('/tasks', async ctx=>{
  1124. const filter = ctx.query.filter||'all';
  1125. const tasks = await tasksModel.listAll(filter);
  1126. ctx.body = await taskView(tasks,filter,null);
  1127. })
  1128. .get('/tasks/edit/:id', async ctx=>{
  1129. const id = ctx.params.id;
  1130. const task = await tasksModel.getTaskById(id);
  1131. ctx.body = await taskView(task,'edit',id);
  1132. })
  1133. .get('/tasks/:taskId', async ctx => {
  1134. const taskId = ctx.params.taskId;
  1135. const filter = ctx.query.filter || 'all';
  1136. const task = await tasksModel.getTaskById(taskId, filter);
  1137. ctx.body = await taskView([task], filter, taskId);
  1138. })
  1139. .get('/events', async (ctx) => {
  1140. const eventsMod = ctx.cookies.get("eventsMod") || 'on';
  1141. if (eventsMod !== 'on') {
  1142. ctx.redirect('/modules');
  1143. return;
  1144. }
  1145. const filter = ctx.query.filter || 'all';
  1146. const events = await eventsModel.listAll(null, filter);
  1147. ctx.body = await eventView(events, filter, null);
  1148. })
  1149. .get('/events/edit/:id', async (ctx) => {
  1150. const eventsMod = ctx.cookies.get("eventsMod") || 'on';
  1151. if (eventsMod !== 'on') {
  1152. ctx.redirect('/modules');
  1153. return;
  1154. }
  1155. const eventId = ctx.params.id;
  1156. const event = await eventsModel.getEventById(eventId);
  1157. ctx.body = await eventView([event], 'edit', eventId);
  1158. })
  1159. .get('/events/:eventId', async ctx => {
  1160. const eventId = ctx.params.eventId;
  1161. const filter = ctx.query.filter || 'all';
  1162. const event = await eventsModel.getEventById(eventId);
  1163. ctx.body = await singleEventView(event, filter);
  1164. })
  1165. .get('/votes', async ctx => {
  1166. const filter = ctx.query.filter || 'all';
  1167. const voteList = await votesModel.listAll(filter);
  1168. ctx.body = await voteView(voteList, filter, null);
  1169. })
  1170. .get('/votes/edit/:id', async ctx => {
  1171. const id = ctx.params.id;
  1172. const vote = await votesModel.getVoteById(id);
  1173. ctx.body = await voteView([vote], 'edit', id);
  1174. })
  1175. .get('/votes/:voteId', async ctx => {
  1176. const voteId = ctx.params.voteId;
  1177. const vote = await votesModel.getVoteById(voteId);
  1178. ctx.body = await voteView(vote);
  1179. })
  1180. .get('/market', async ctx => {
  1181. const marketMod = ctx.cookies.get("marketMod") || 'on';
  1182. if (marketMod !== 'on') {
  1183. ctx.redirect('/modules');
  1184. return;
  1185. }
  1186. const filter = ctx.query.filter || 'all';
  1187. const marketItems = await marketModel.listAllItems(filter);
  1188. ctx.body = await marketView(marketItems, filter, null);
  1189. })
  1190. .get('/market/edit/:id', async ctx => {
  1191. const id = ctx.params.id;
  1192. const marketItem = await marketModel.getItemById(id);
  1193. ctx.body = await marketView([marketItem], 'edit', marketItem);
  1194. })
  1195. .get('/market/:itemId', async ctx => {
  1196. const itemId = ctx.params.itemId;
  1197. const filter = ctx.query.filter || 'all';
  1198. const item = await marketModel.getItemById(itemId);
  1199. ctx.body = await singleMarketView(item, filter);
  1200. })
  1201. .get('/jobs', async (ctx) => {
  1202. const jobsMod = ctx.cookies.get("jobsMod") || 'on';
  1203. if (jobsMod !== 'on') {
  1204. ctx.redirect('/modules');
  1205. return;
  1206. }
  1207. const filter = ctx.query.filter || 'ALL';
  1208. const query = {
  1209. search: ctx.query.search || '',
  1210. };
  1211. if (filter === 'CV') {
  1212. query.location = ctx.query.location || '';
  1213. query.language = ctx.query.language || '';
  1214. query.skills = ctx.query.skills || '';
  1215. const inhabitants = await inhabitantsModel.listInhabitants({
  1216. filter: 'CVs',
  1217. ...query
  1218. });
  1219. ctx.body = await jobsView(inhabitants, filter, query);
  1220. return;
  1221. }
  1222. const jobs = await jobsModel.listJobs(filter, ctx.state.user?.id, query);
  1223. ctx.body = await jobsView(jobs, filter, query);
  1224. })
  1225. .get('/jobs/edit/:id', async (ctx) => {
  1226. const id = ctx.params.id;
  1227. const job = await jobsModel.getJobById(id);
  1228. ctx.body = await jobsView([job], 'EDIT');
  1229. })
  1230. .get('/jobs/:jobId', async (ctx) => {
  1231. const jobId = ctx.params.jobId;
  1232. const filter = ctx.query.filter || 'ALL';
  1233. const job = await jobsModel.getJobById(jobId);
  1234. ctx.body = await singleJobsView(job, filter);
  1235. })
  1236. .get('/cipher', async (ctx) => {
  1237. const cipherMod = ctx.cookies.get("cipherMod") || 'on';
  1238. if (cipherMod !== 'on') {
  1239. ctx.redirect('/modules');
  1240. return;
  1241. }
  1242. try {
  1243. ctx.body = await cipherView();
  1244. } catch (error) {
  1245. ctx.body = { error: error.message };
  1246. }
  1247. })
  1248. .get("/thread/:message", async (ctx) => {
  1249. const { message } = ctx.params;
  1250. const thread = async (message) => {
  1251. const messages = await post.fromThread(message);
  1252. return threadView({ messages });
  1253. };
  1254. ctx.body = await thread(message);
  1255. })
  1256. .get("/subtopic/:message", async (ctx) => {
  1257. const { message } = ctx.params;
  1258. const rootMessage = await post.get(message);
  1259. const myFeedId = await meta.myFeedId();
  1260. debug("%O", rootMessage);
  1261. const messages = [rootMessage];
  1262. ctx.body = await subtopicView({ messages, myFeedId });
  1263. })
  1264. .get("/publish", async (ctx) => {
  1265. ctx.body = await publishView();
  1266. })
  1267. .get("/comment/:message", async (ctx) => {
  1268. const { messages, myFeedId, parentMessage } =
  1269. await resolveCommentComponents(ctx);
  1270. ctx.body = await commentView({ messages, myFeedId, parentMessage });
  1271. })
  1272. .get("/wallet", async (ctx) => {
  1273. const { url, user, pass } = getConfig().wallet;
  1274. const walletMod = ctx.cookies.get("walletMod") || 'on';
  1275. if (walletMod !== 'on') {
  1276. ctx.redirect('/modules');
  1277. return;
  1278. }
  1279. try {
  1280. const balance = await walletModel.getBalance(url, user, pass);
  1281. const address = await walletModel.getAddress(url, user, pass);
  1282. ctx.body = await walletView(balance, address);
  1283. } catch (error) {
  1284. ctx.body = await walletErrorView(error);
  1285. }
  1286. })
  1287. .get("/wallet/history", async (ctx) => {
  1288. const { url, user, pass } = getConfig().wallet;
  1289. try {
  1290. const balance = await walletModel.getBalance(url, user, pass);
  1291. const transactions = await walletModel.listTransactions(url, user, pass);
  1292. const address = await walletModel.getAddress(url, user, pass);
  1293. ctx.body = await walletHistoryView(balance, transactions, address);
  1294. } catch (error) {
  1295. ctx.body = await walletErrorView(error);
  1296. }
  1297. })
  1298. .get("/wallet/receive", async (ctx) => {
  1299. const { url, user, pass } = getConfig().wallet;
  1300. try {
  1301. const balance = await walletModel.getBalance(url, user, pass);
  1302. const address = await walletModel.getAddress(url, user, pass);
  1303. ctx.body = await walletReceiveView(balance, address);
  1304. } catch (error) {
  1305. ctx.body = await walletErrorView(error);
  1306. }
  1307. })
  1308. .get("/wallet/send", async (ctx) => {
  1309. const { url, user, pass, fee } = getConfig().wallet;
  1310. try {
  1311. const balance = await walletModel.getBalance(url, user, pass);
  1312. const address = await walletModel.getAddress(url, user, pass);
  1313. ctx.body = await walletSendFormView(balance, null, null, fee, null, address);
  1314. } catch (error) {
  1315. ctx.body = await walletErrorView(error);
  1316. }
  1317. })
  1318. .get('/transfers', async ctx => {
  1319. const filter = ctx.query.filter || 'all'
  1320. const list = await transfersModel.listAll(filter)
  1321. ctx.body = await transferView(list, filter, null)
  1322. })
  1323. .get('/transfers/edit/:id', async ctx => {
  1324. const tr = await transfersModel.getTransferById(ctx.params.id)
  1325. ctx.body = await transferView([tr], 'edit', ctx.params.id)
  1326. })
  1327. .get('/transfers/:transferId', async ctx => {
  1328. const transferId = ctx.params.transferId;
  1329. const filter = ctx.query.filter || 'all';
  1330. const transfer = await transfersModel.getTransferById(transferId);
  1331. ctx.body = await singleTransferView(transfer, filter);
  1332. })
  1333. //POST backend routes
  1334. .post('/ai', koaBody(), async (ctx) => {
  1335. const axios = require('../server/node_modules/axios').default;
  1336. const { input } = ctx.request.body;
  1337. if (!input) {
  1338. ctx.status = 400;
  1339. ctx.body = { error: 'No input provided' };
  1340. return;
  1341. }
  1342. const config = getConfig();
  1343. const userPrompt = config.ai?.prompt?.trim() || "Provide an informative and precise response.";
  1344. const response = await axios.post('http://localhost:4001/ai', { input });
  1345. const aiResponse = response.data.answer;
  1346. const historyPath = path.join(__dirname, '..', '..', 'src', 'configs', 'AI-history.json');
  1347. let chatHistory = [];
  1348. try {
  1349. const fileData = fs.readFileSync(historyPath, 'utf-8');
  1350. chatHistory = JSON.parse(fileData);
  1351. } catch (e) {
  1352. chatHistory = [];
  1353. }
  1354. chatHistory.unshift({
  1355. prompt: userPrompt,
  1356. question: input,
  1357. answer: aiResponse,
  1358. timestamp: Date.now()
  1359. });
  1360. chatHistory = chatHistory.slice(0, 20);
  1361. fs.writeFileSync(historyPath, JSON.stringify(chatHistory, null, 2), 'utf-8');
  1362. ctx.body = aiView(chatHistory, userPrompt);
  1363. })
  1364. .post('/ai/clear', async (ctx) => {
  1365. const fs = require('fs');
  1366. const path = require('path');
  1367. const { getConfig } = require('../configs/config-manager.js');
  1368. const historyPath = path.join(__dirname, '..', '..', 'src', 'configs', 'AI-history.json');
  1369. fs.writeFileSync(historyPath, '[]', 'utf-8');
  1370. const config = getConfig();
  1371. const userPrompt = config.ai?.prompt?.trim() || '';
  1372. ctx.body = aiView([], userPrompt);
  1373. })
  1374. .post('/pixelia/paint', koaBody(), async (ctx) => {
  1375. const { x, y, color } = ctx.request.body;
  1376. if (x < 1 || x > 50 || y < 1 || y > 200) {
  1377. const errorMessage = 'Coordinates are wrong!';
  1378. const pixelArt = await pixeliaModel.listPixels();
  1379. ctx.body = pixeliaView(pixelArt, errorMessage);
  1380. return;
  1381. }
  1382. await pixeliaModel.paintPixel(x, y, color);
  1383. ctx.redirect('/pixelia');
  1384. })
  1385. .post('/pm', koaBody(), async ctx => {
  1386. const { recipients, subject, text } = ctx.request.body;
  1387. const recipientsArr = recipients.split(',').map(s => s.trim()).filter(Boolean);
  1388. await pmModel.sendMessage(recipientsArr, subject, text);
  1389. ctx.redirect('/pm');
  1390. })
  1391. .post('/inbox/delete/:id', koaBody(), async ctx => {
  1392. const { id } = ctx.params;
  1393. await pmModel.deleteMessageById(id);
  1394. ctx.redirect('/inbox');
  1395. })
  1396. .post("/search", koaBody(), async (ctx) => {
  1397. const body = ctx.request.body;
  1398. const query = body.query || "";
  1399. let types = body.type || [];
  1400. if (typeof types === "string") types = [types];
  1401. if (!Array.isArray(types)) types = [];
  1402. if (!query) {
  1403. return ctx.body = await searchView({ messages: [], query, types });
  1404. }
  1405. const results = await searchModel.search({ query, types });
  1406. const groupedResults = Object.entries(results).reduce((acc, [type, msgs]) => {
  1407. acc[type] = msgs.map(msg => {
  1408. if (!msg.value || !msg.value.content) {
  1409. return {};
  1410. }
  1411. return {
  1412. ...msg,
  1413. content: msg.value.content,
  1414. author: msg.value.content.author || 'Unknown',
  1415. };
  1416. });
  1417. return acc;
  1418. }, {});
  1419. ctx.body = await searchView({ results: groupedResults, query, types });
  1420. })
  1421. .post("/subtopic/preview/:message",
  1422. koaBody({ multipart: true }),
  1423. async (ctx) => {
  1424. const { message } = ctx.params;
  1425. const rootMessage = await post.get(message);
  1426. const myFeedId = await meta.myFeedId();
  1427. const rawContentWarning = String(ctx.request.body.contentWarning).trim();
  1428. const contentWarning =
  1429. rawContentWarning.length > 0 ? rawContentWarning : undefined;
  1430. const messages = [rootMessage];
  1431. const previewData = await preparePreview(ctx);
  1432. ctx.body = await previewSubtopicView({
  1433. messages,
  1434. myFeedId,
  1435. previewData,
  1436. contentWarning,
  1437. });
  1438. }
  1439. )
  1440. .post("/subtopic/:message", koaBody(), async (ctx) => {
  1441. const { message } = ctx.params;
  1442. const text = String(ctx.request.body.text);
  1443. const rawContentWarning = String(ctx.request.body.contentWarning).trim();
  1444. const contentWarning =
  1445. rawContentWarning.length > 0 ? rawContentWarning : undefined;
  1446. const publishSubtopic = async ({ message, text }) => {
  1447. const mentions = extractMentions(text);
  1448. const parent = await post.get(message);
  1449. return post.subtopic({
  1450. parent,
  1451. message: { text, mentions, contentWarning },
  1452. });
  1453. };
  1454. ctx.body = await publishSubtopic({ message, text });
  1455. ctx.redirect(`/thread/${encodeURIComponent(message)}`);
  1456. })
  1457. .post("/comment/preview/:message", koaBody({ multipart: true }), async (ctx) => {
  1458. const { messages, contentWarning, myFeedId, parentMessage } = await resolveCommentComponents(ctx);
  1459. const previewData = await preparePreview(ctx);
  1460. ctx.body = await previewCommentView({
  1461. messages,
  1462. myFeedId,
  1463. contentWarning,
  1464. previewData,
  1465. parentMessage,
  1466. });
  1467. })
  1468. .post("/comment/:message", koaBody(), async (ctx) => {
  1469. let decodedMessage;
  1470. try {
  1471. decodedMessage = decodeURIComponent(ctx.params.message);
  1472. } catch {
  1473. decodedMessage = ctx.params.message;
  1474. }
  1475. const text = String(ctx.request.body.text);
  1476. const rawContentWarning = String(ctx.request.body.contentWarning);
  1477. const contentWarning =
  1478. rawContentWarning.length > 0 ? rawContentWarning : undefined;
  1479. let mentions = extractMentions(text);
  1480. if (!Array.isArray(mentions)) mentions = [];
  1481. const parent = await meta.get(decodedMessage);
  1482. ctx.body = await post.comment({
  1483. parent,
  1484. message: {
  1485. text,
  1486. mentions,
  1487. contentWarning
  1488. },
  1489. });
  1490. ctx.redirect(`/thread/${encodeURIComponent(parent.key)}`);
  1491. })
  1492. .post("/publish/preview", koaBody({multipart: true, formidable: { multiples: false }, urlencoded: true }), async (ctx) => {
  1493. const rawContentWarning = ctx.request.body.contentWarning?.toString().trim() || "";
  1494. const contentWarning = rawContentWarning.length > 0 ? rawContentWarning : undefined;
  1495. const previewData = await preparePreview(ctx);
  1496. ctx.body = await previewView({ previewData, contentWarning });
  1497. })
  1498. .post("/publish", koaBody({ multipart: true, urlencoded: true, formidable: { multiples: false } }), async (ctx) => {
  1499. const text = ctx.request.body.text?.toString().trim() || "";
  1500. const rawContentWarning = ctx.request.body.contentWarning?.toString().trim() || "";
  1501. const contentWarning = rawContentWarning.length > 0 ? rawContentWarning : undefined;
  1502. let mentions = [];
  1503. try {
  1504. mentions = JSON.parse(ctx.request.body.mentions || "[]");
  1505. } catch (e) {
  1506. mentions = await extractMentions(text);
  1507. }
  1508. await post.root({ text, mentions, contentWarning });
  1509. ctx.redirect("/public/latest");
  1510. })
  1511. .post("/publish/custom", koaBody(), async (ctx) => {
  1512. const text = String(ctx.request.body.text);
  1513. const obj = JSON.parse(text);
  1514. ctx.body = await post.publishCustom(obj);
  1515. ctx.redirect(`/public/latest`);
  1516. })
  1517. .post("/follow/:feed", koaBody(), async (ctx) => {
  1518. const { feed } = ctx.params;
  1519. const referer = new URL(ctx.request.header.referer);
  1520. ctx.body = await friend.follow(feed);
  1521. ctx.redirect(referer.href);
  1522. })
  1523. .post("/unfollow/:feed", koaBody(), async (ctx) => {
  1524. const { feed } = ctx.params;
  1525. const referer = new URL(ctx.request.header.referer);
  1526. ctx.body = await friend.unfollow(feed);
  1527. ctx.redirect(referer.href);
  1528. })
  1529. .post("/block/:feed", koaBody(), async (ctx) => {
  1530. const { feed } = ctx.params;
  1531. const referer = new URL(ctx.request.header.referer);
  1532. ctx.body = await friend.block(feed);
  1533. ctx.redirect(referer.href);
  1534. })
  1535. .post("/unblock/:feed", koaBody(), async (ctx) => {
  1536. const { feed } = ctx.params;
  1537. const referer = new URL(ctx.request.header.referer);
  1538. ctx.body = await friend.unblock(feed);
  1539. ctx.redirect(referer.href);
  1540. })
  1541. .post("/like/:message", koaBody(), async (ctx) => {
  1542. const { message } = ctx.params;
  1543. const messageKey = message;
  1544. const voteValue = Number(ctx.request.body.voteValue);
  1545. const encoded = {
  1546. message: encodeURIComponent(message),
  1547. };
  1548. const referer = new URL(ctx.request.header.referer);
  1549. referer.hash = `centered-footer-${encoded.message}`;
  1550. const like = async ({ messageKey, voteValue }) => {
  1551. const value = Number(voteValue);
  1552. const message = await post.get(messageKey);
  1553. const isPrivate = message.value.meta.private === true;
  1554. const messageRecipients = isPrivate ? message.value.content.recps : [];
  1555. const normalized = messageRecipients.map((recipient) => {
  1556. if (typeof recipient === "string") {
  1557. return recipient;
  1558. }
  1559. if (typeof recipient.link === "string") {
  1560. return recipient.link;
  1561. }
  1562. return null;
  1563. });
  1564. const recipients = normalized.length > 0 ? normalized : undefined;
  1565. return vote.publish({ messageKey, value, recps: recipients });
  1566. };
  1567. ctx.body = await like({ messageKey, voteValue });
  1568. ctx.redirect(referer.href);
  1569. })
  1570. .post('/forum/create', koaBody(), async ctx => {
  1571. const { category, title, text } = ctx.request.body;
  1572. await forumModel.createForum(category, title, text);
  1573. ctx.redirect('/forum');
  1574. })
  1575. .post('/forum/:id/message', koaBody(), async ctx => {
  1576. const forumId = ctx.params.id;
  1577. const { message, parentId } = ctx.request.body;
  1578. const userId = SSBconfig.config.keys.id;
  1579. const newMessage = { text: message, author: userId, timestamp: new Date().toISOString() };
  1580. await forumModel.addMessageToForum(forumId, newMessage, parentId);
  1581. ctx.redirect(`/forum/${encodeURIComponent(forumId)}`);
  1582. })
  1583. .post('/forum/:forumId/vote', koaBody(), async ctx => {
  1584. const { forumId } = ctx.params;
  1585. const { target, value } = ctx.request.body;
  1586. await forumModel.voteContent(target, parseInt(value, 10));
  1587. const back = ctx.get('referer') || `/forum/${encodeURIComponent(forumId)}`;
  1588. ctx.redirect(back);
  1589. })
  1590. .post('/forum/delete/:id', koaBody(), async ctx => {
  1591. await forumModel.deleteForumById(ctx.params.id);
  1592. ctx.redirect('/forum');
  1593. })
  1594. .post('/legacy/export', koaBody(), async (ctx) => {
  1595. const password = ctx.request.body.password;
  1596. if (!password || password.length < 32) {
  1597. ctx.redirect('/legacy');
  1598. return;
  1599. }
  1600. try {
  1601. const encryptedFilePath = await legacyModel.exportData({ password });
  1602. ctx.body = {
  1603. message: 'Data exported successfully!',
  1604. file: encryptedFilePath
  1605. };
  1606. ctx.redirect('/legacy');
  1607. } catch (error) {
  1608. ctx.status = 500;
  1609. ctx.body = { error: `Error: ${error.message}` };
  1610. ctx.redirect('/legacy');
  1611. }
  1612. })
  1613. .post('/legacy/import', koaBody({
  1614. multipart: true,
  1615. formidable: {
  1616. keepExtensions: true,
  1617. uploadDir: '/tmp',
  1618. }
  1619. }), async (ctx) => {
  1620. const uploadedFile = ctx.request.files?.uploadedFile;
  1621. const password = ctx.request.body.importPassword;
  1622. if (!uploadedFile) {
  1623. ctx.body = { error: 'No file uploaded' };
  1624. ctx.redirect('/legacy');
  1625. return;
  1626. }
  1627. if (!password || password.length < 32) {
  1628. ctx.body = { error: 'Password is too short or missing.' };
  1629. ctx.redirect('/legacy');
  1630. return;
  1631. }
  1632. try {
  1633. await legacyModel.importData({ filePath: uploadedFile.filepath, password });
  1634. ctx.body = { message: 'Data imported successfully!' };
  1635. ctx.redirect('/legacy');
  1636. } catch (error) {
  1637. ctx.body = { error: error.message };
  1638. ctx.redirect('/legacy');
  1639. }
  1640. })
  1641. .post('/trending/:contentId/:category', async (ctx) => {
  1642. const { contentId, category } = ctx.params;
  1643. const voterId = SSBconfig?.keys?.id;
  1644. const target = await trendingModel.getMessageById(contentId);
  1645. if (target?.content?.opinions_inhabitants?.includes(voterId)) {
  1646. ctx.flash = { message: 'You have already opined.' };
  1647. ctx.redirect('/trending');
  1648. return;
  1649. }
  1650. await trendingModel.createVote(contentId, category);
  1651. ctx.redirect('/trending');
  1652. })
  1653. .post('/opinions/:contentId/:category', async (ctx) => {
  1654. const { contentId, category } = ctx.params;
  1655. const voterId = SSBconfig?.keys?.id;
  1656. const target = await opinionsModel.getMessageById(contentId);
  1657. if (target?.content?.opinions_inhabitants?.includes(voterId)) {
  1658. ctx.flash = { message: 'You have already opined.' };
  1659. ctx.redirect('/opinions');
  1660. return;
  1661. }
  1662. await opinionsModel.createVote(contentId, category);
  1663. ctx.redirect('/opinions');
  1664. })
  1665. .post('/agenda/discard/:itemId', async (ctx) => {
  1666. const { itemId } = ctx.params;
  1667. await agendaModel.discardItem(itemId);
  1668. ctx.redirect('/agenda');
  1669. })
  1670. .post('/agenda/restore/:itemId', async (ctx) => {
  1671. const { itemId } = ctx.params;
  1672. await agendaModel.restoreItem(itemId);
  1673. ctx.redirect('/agenda?filter=discarded');
  1674. })
  1675. .post('/feed/create', koaBody(), async ctx => {
  1676. const { text } = ctx.request.body || {};
  1677. await feedModel.createFeed(text.trim());
  1678. ctx.redirect('/feed');
  1679. })
  1680. .post('/feed/opinions/:feedId/:category', async ctx => {
  1681. const { feedId, category } = ctx.params;
  1682. await opinionsModel.createVote(feedId, category);
  1683. ctx.redirect('/feed');
  1684. })
  1685. .post('/feed/refeed/:id', koaBody(), async ctx => {
  1686. await feedModel.createRefeed(ctx.params.id);
  1687. ctx.redirect('/feed');
  1688. })
  1689. .post('/bookmarks/create', koaBody(), async (ctx) => {
  1690. const { url, tags, description, category, lastVisit } = ctx.request.body;
  1691. const formattedLastVisit = lastVisit ? moment(lastVisit).isBefore(moment()) ? moment(lastVisit).toISOString() : moment().toISOString() : moment().toISOString();
  1692. await bookmarksModel.createBookmark(url, tags, description, category, formattedLastVisit);
  1693. ctx.redirect('/bookmarks');
  1694. })
  1695. .post('/bookmarks/update/:id', koaBody(), async (ctx) => {
  1696. const { url, tags, description, category, lastVisit } = ctx.request.body;
  1697. const bookmarkId = ctx.params.id;
  1698. const formattedLastVisit = lastVisit
  1699. ? moment(lastVisit).isBefore(moment())
  1700. ? moment(lastVisit).toISOString()
  1701. : moment().toISOString()
  1702. : moment().toISOString();
  1703. const bookmark = await bookmarksModel.getBookmarkById(bookmarkId);
  1704. if (bookmark.opinions_inhabitants && bookmark.opinions_inhabitants.length > 0) {
  1705. ctx.flash = { message: "This bookmark has received votes and cannot be updated." };
  1706. ctx.redirect(`/bookmarks?filter=mine`);
  1707. }
  1708. await bookmarksModel.updateBookmarkById(bookmarkId, {
  1709. url,
  1710. tags,
  1711. description,
  1712. category,
  1713. lastVisit: formattedLastVisit,
  1714. createdAt: bookmark.createdAt,
  1715. author: bookmark.author,
  1716. });
  1717. ctx.redirect('/bookmarks?filter=mine');
  1718. })
  1719. .post('/bookmarks/delete/:id', koaBody(), async (ctx) => {
  1720. const bookmarkId = ctx.params.id;
  1721. await bookmarksModel.deleteBookmarkById(bookmarkId);
  1722. ctx.redirect('/bookmarks?filter=mine');
  1723. })
  1724. .post('/bookmarks/opinions/:bookmarkId/:category', async (ctx) => {
  1725. const { bookmarkId, category } = ctx.params;
  1726. const voterId = SSBconfig?.keys?.id;
  1727. const bookmark = await bookmarksModel.getBookmarkById(bookmarkId);
  1728. if (bookmark.opinions_inhabitants && bookmark.opinions_inhabitants.includes(voterId)) {
  1729. ctx.flash = { message: "You have already opined." };
  1730. ctx.redirect('/bookmarks');
  1731. return;
  1732. }
  1733. await opinionsModel.createVote(bookmarkId, category, 'bookmark');
  1734. ctx.redirect('/bookmarks');
  1735. })
  1736. .post('/images/create', koaBody({ multipart: true }), async ctx => {
  1737. const blob = await handleBlobUpload(ctx, 'image');
  1738. const { tags, title, description, meme } = ctx.request.body;
  1739. await imagesModel.createImage(blob, tags, title, description, meme);
  1740. ctx.redirect('/images');
  1741. })
  1742. .post('/images/update/:id', koaBody({ multipart: true }), async ctx => {
  1743. const { tags, title, description, meme } = ctx.request.body;
  1744. const blob = ctx.request.files?.image ? await handleBlobUpload(ctx, 'image') : null;
  1745. const match = blob?.match(/\(([^)]+)\)/);
  1746. const blobId = match ? match[1] : blob;
  1747. await imagesModel.updateImageById(ctx.params.id, blobId, tags, title, description, meme);
  1748. ctx.redirect('/images?filter=mine');
  1749. })
  1750. .post('/images/delete/:id', koaBody(), async ctx => {
  1751. await imagesModel.deleteImageById(ctx.params.id);
  1752. ctx.redirect('/images?filter=mine');
  1753. })
  1754. .post('/images/opinions/:imageId/:category', async ctx => {
  1755. const { imageId, category } = ctx.params;
  1756. const voterId = SSBconfig?.keys?.id;
  1757. const image = await imagesModel.getImageById(imageId);
  1758. if (image.opinions_inhabitants && image.opinions_inhabitants.includes(voterId)) {
  1759. ctx.flash = { message: "You have already opined." };
  1760. ctx.redirect('/images');
  1761. return;
  1762. }
  1763. await imagesModel.createOpinion(imageId, category, 'image');
  1764. ctx.redirect('/images');
  1765. })
  1766. .post('/audios/create', koaBody({ multipart: true }), async (ctx) => {
  1767. const audioBlob = await handleBlobUpload(ctx, 'audio');
  1768. const { tags, title, description } = ctx.request.body;
  1769. await audiosModel.createAudio(audioBlob, tags, title, description);
  1770. ctx.redirect('/audios');
  1771. })
  1772. .post('/audios/update/:id', koaBody({ multipart: true }), async (ctx) => {
  1773. const { tags, title, description } = ctx.request.body;
  1774. const blob = ctx.request.files?.audio ? await handleBlobUpload(ctx, 'audio') : null;
  1775. await audiosModel.updateAudioById(ctx.params.id, blob, tags, title, description);
  1776. ctx.redirect('/audios?filter=mine');
  1777. })
  1778. .post('/audios/delete/:id', koaBody(), async (ctx) => {
  1779. await audiosModel.deleteAudioById(ctx.params.id);
  1780. ctx.redirect('/audios?filter=mine');
  1781. })
  1782. .post('/audios/opinions/:audioId/:category', async (ctx) => {
  1783. const { audioId, category } = ctx.params;
  1784. const voterId = SSBconfig?.keys?.id;
  1785. const audio = await audiosModel.getAudioById(audioId);
  1786. if (audio.opinions_inhabitants?.includes(voterId)) {
  1787. ctx.flash = { message: "You have already opined." };
  1788. ctx.redirect('/audios');
  1789. return;
  1790. }
  1791. await audiosModel.createOpinion(audioId, category);
  1792. ctx.redirect('/audios');
  1793. })
  1794. .post('/videos/create', koaBody({ multipart: true }), async (ctx) => {
  1795. const videoBlob = await handleBlobUpload(ctx, 'video');
  1796. const { tags, title, description } = ctx.request.body;
  1797. await videosModel.createVideo(videoBlob, tags, title, description);
  1798. ctx.redirect('/videos');
  1799. })
  1800. .post('/videos/update/:id', koaBody({ multipart: true }), async (ctx) => {
  1801. const { tags, title, description } = ctx.request.body;
  1802. const blob = ctx.request.files?.video ? await handleBlobUpload(ctx, 'video') : null;
  1803. await videosModel.updateVideoById(ctx.params.id, blob, tags, title, description);
  1804. ctx.redirect('/videos?filter=mine');
  1805. })
  1806. .post('/videos/delete/:id', koaBody(), async (ctx) => {
  1807. await videosModel.deleteVideoById(ctx.params.id);
  1808. ctx.redirect('/videos?filter=mine');
  1809. })
  1810. .post('/videos/opinions/:videoId/:category', async (ctx) => {
  1811. const { videoId, category } = ctx.params;
  1812. const voterId = SSBconfig?.keys?.id;
  1813. const video = await videosModel.getVideoById(videoId);
  1814. if (video.opinions_inhabitants?.includes(voterId)) {
  1815. ctx.flash = { message: "You have already opined." };
  1816. ctx.redirect('/videos');
  1817. return;
  1818. }
  1819. await videosModel.createOpinion(videoId, category);
  1820. ctx.redirect('/videos');
  1821. })
  1822. .post('/documents/create', koaBody({ multipart: true }), async (ctx) => {
  1823. const docBlob = await handleBlobUpload(ctx, 'document');
  1824. const { tags, title, description } = ctx.request.body;
  1825. await documentsModel.createDocument(docBlob, tags, title, description);
  1826. ctx.redirect('/documents');
  1827. })
  1828. .post('/documents/update/:id', koaBody({ multipart: true }), async (ctx) => {
  1829. const { tags, title, description } = ctx.request.body;
  1830. const blob = ctx.request.files?.document ? await handleBlobUpload(ctx, 'document') : null;
  1831. await documentsModel.updateDocumentById(ctx.params.id, blob, tags, title, description);
  1832. ctx.redirect('/documents?filter=mine');
  1833. })
  1834. .post('/documents/delete/:id', koaBody(), async (ctx) => {
  1835. await documentsModel.deleteDocumentById(ctx.params.id);
  1836. ctx.redirect('/documents?filter=mine');
  1837. })
  1838. .post('/documents/opinions/:documentId/:category', async (ctx) => {
  1839. const { documentId, category } = ctx.params;
  1840. const voterId = SSBconfig?.keys?.id;
  1841. const document = await documentsModel.getDocumentById(documentId);
  1842. if (document.opinions_inhabitants?.includes(voterId)) {
  1843. ctx.flash = { message: "You have already opined." };
  1844. ctx.redirect('/documents');
  1845. return;
  1846. }
  1847. await documentsModel.createOpinion(documentId, category);
  1848. ctx.redirect('/documents');
  1849. })
  1850. .post('/cv/upload', koaBody({ multipart: true }), async ctx => {
  1851. const photoUrl = await handleBlobUpload(ctx, 'image')
  1852. await cvModel.createCV(ctx.request.body, photoUrl)
  1853. ctx.redirect('/cv')
  1854. })
  1855. .post('/cv/update/:id', koaBody({ multipart: true }), async ctx => {
  1856. const photoUrl = await handleBlobUpload(ctx, 'image')
  1857. await cvModel.updateCV(ctx.params.id, ctx.request.body, photoUrl)
  1858. ctx.redirect('/cv')
  1859. })
  1860. .post('/cv/delete/:id', async ctx => {
  1861. await cvModel.deleteCVById(ctx.params.id)
  1862. ctx.redirect('/cv')
  1863. })
  1864. .post('/cipher/encrypt', koaBody(), async (ctx) => {
  1865. const { text, password } = ctx.request.body;
  1866. if (password.length < 32) {
  1867. ctx.body = { error: 'Password is too short or missing.' };
  1868. ctx.redirect('/cipher');
  1869. return;
  1870. }
  1871. const { encryptedText, iv, salt, authTag } = cipherModel.encryptData(text, password);
  1872. const view = await cipherView(encryptedText, "", iv, password);
  1873. ctx.body = view;
  1874. })
  1875. .post('/cipher/decrypt', koaBody(), async (ctx) => {
  1876. const { encryptedText, password } = ctx.request.body;
  1877. if (password.length < 32) {
  1878. ctx.body = { error: 'Password is too short or missing.' };
  1879. ctx.redirect('/cipher');
  1880. return;
  1881. }
  1882. const decryptedText = cipherModel.decryptData(encryptedText, password);
  1883. const view = await cipherView("", decryptedText, "", password);
  1884. ctx.body = view;
  1885. })
  1886. .post('/tribes/create', koaBody({ multipart: true }), async ctx => {
  1887. const { title, description, location, tags, isLARP, isAnonymous, inviteMode } = ctx.request.body;
  1888. // Block L.A.R.P. creation
  1889. if (isLARP === 'true' || isLARP === true) {
  1890. ctx.status = 400;
  1891. ctx.body = { error: "L.A.R.P. tribes cannot be created." };
  1892. return;
  1893. }
  1894. const image = await handleBlobUpload(ctx, 'image');
  1895. await tribesModel.createTribe(
  1896. title,
  1897. description,
  1898. image,
  1899. location,
  1900. tags,
  1901. isLARP === 'true',
  1902. isAnonymous === 'true',
  1903. inviteMode
  1904. );
  1905. ctx.redirect('/tribes');
  1906. })
  1907. .post('/tribes/update/:id', koaBody({ multipart: true }), async ctx => {
  1908. const { title, description, location, isLARP, isAnonymous, inviteMode, tags } = ctx.request.body;
  1909. // Block L.A.R.P. creation
  1910. if (isLARP === 'true' || isLARP === true) {
  1911. ctx.status = 400;
  1912. ctx.body = { error: "L.A.R.P. tribes cannot be updated." };
  1913. return;
  1914. }
  1915. const parsedTags = tags ? tags.split(',').map(t => t.trim()).filter(Boolean) : [];
  1916. const image = await handleBlobUpload(ctx, 'image');
  1917. await tribesModel.updateTribeById(ctx.params.id, {
  1918. title,
  1919. description,
  1920. image,
  1921. location,
  1922. tags: parsedTags,
  1923. isLARP: isLARP === 'true',
  1924. isAnonymous: isAnonymous === 'true',
  1925. inviteMode
  1926. });
  1927. ctx.redirect('/tribes?filter=mine');
  1928. })
  1929. .post('/tribes/delete/:id', async ctx => {
  1930. await tribesModel.deleteTribeById(ctx.params.id)
  1931. ctx.redirect('/tribes?filter=mine')
  1932. })
  1933. .post('/tribes/generate-invite', koaBody(), async ctx => {
  1934. const { tribeId } = ctx.request.body;
  1935. const inviteCode = await tribesModel.generateInvite(tribeId);
  1936. ctx.body = await renderInvitePage(inviteCode);
  1937. })
  1938. .post('/tribes/join-code', koaBody(), async ctx => {
  1939. const { inviteCode } = ctx.request.body
  1940. await tribesModel.joinByInvite(inviteCode)
  1941. ctx.redirect('/tribes?filter=membership')
  1942. })
  1943. .post('/tribes/leave/:id', koaBody(), async ctx => {
  1944. await tribesModel.leaveTribe(ctx.params.id)
  1945. ctx.redirect('/tribes?filter=membership')
  1946. })
  1947. .post('/tribes/:id/message', koaBody(), async ctx => {
  1948. const tribeId = ctx.params.id;
  1949. const message = ctx.request.body.message;
  1950. await tribesModel.postMessage(tribeId, message);
  1951. ctx.redirect(ctx.headers.referer);
  1952. })
  1953. .post('/tribes/:id/refeed/:msgId', koaBody(), async ctx => {
  1954. const tribeId = ctx.params.id;
  1955. const msgId = ctx.params.msgId;
  1956. await tribesModel.refeed(tribeId, msgId);
  1957. ctx.redirect(ctx.headers.referer);
  1958. })
  1959. .post('/tribe/:id/message', koaBody(), async ctx => {
  1960. const tribeId = ctx.params.id;
  1961. const message = ctx.request.body.message;
  1962. await tribesModel.postMessage(tribeId, message);
  1963. ctx.redirect('/tribes?filter=mine')
  1964. })
  1965. .post('/panic/remove', koaBody(), async (ctx) => {
  1966. const { exec } = require('child_process');
  1967. try {
  1968. await panicmodeModel.removeSSB();
  1969. ctx.body = {
  1970. message: 'Your blockchain has been succesfully deleted!'
  1971. };
  1972. exec('pkill -f "node SSB_server.js start"');
  1973. setTimeout(() => {
  1974. process.exit(0);
  1975. }, 1000);
  1976. } catch (error) {
  1977. ctx.body = {
  1978. error: 'Error deleting your blockchain: ' + error.message
  1979. };
  1980. }
  1981. })
  1982. .post('/export/create', async (ctx) => {
  1983. try {
  1984. const outputPath = path.join(os.homedir(), 'ssb_exported.zip');
  1985. await exportmodeModel.exportSSB(outputPath);
  1986. ctx.set('Content-Type', 'application/zip');
  1987. ctx.set('Content-Disposition', `attachment; filename=ssb_exported.zip`);
  1988. ctx.body = fs.createReadStream(outputPath);
  1989. ctx.res.on('finish', () => {
  1990. fs.unlinkSync(outputPath);
  1991. });
  1992. } catch (error) {
  1993. ctx.body = {
  1994. error: 'Error exporting your blockchain: ' + error.message
  1995. };
  1996. }
  1997. })
  1998. .post('/tasks/create', koaBody(), async (ctx) => {
  1999. const { title, description, startTime, endTime, priority, location, tags, isPublic } = ctx.request.body;
  2000. await tasksModel.createTask(title, description, startTime, endTime, priority, location, tags, isPublic);
  2001. ctx.redirect('/tasks?filter=mine');
  2002. })
  2003. .post('/tasks/update/:id', koaBody(), async (ctx) => {
  2004. const { title, description, startTime, endTime, priority, location, tags, isPublic } = ctx.request.body;
  2005. const taskId = ctx.params.id;
  2006. const parsedTags = Array.isArray(tags)
  2007. ? tags.filter(Boolean)
  2008. : (typeof tags === 'string' ? tags.split(',').map(t => t.trim()).filter(Boolean) : []);
  2009. await tasksModel.updateTaskById(taskId, {
  2010. title,
  2011. description,
  2012. startTime,
  2013. endTime,
  2014. priority,
  2015. location,
  2016. tags: parsedTags,
  2017. isPublic
  2018. });
  2019. ctx.redirect('/tasks?filter=mine');
  2020. })
  2021. .post('/tasks/assign/:id', koaBody(), async (ctx) => {
  2022. const taskId = ctx.params.id;
  2023. await tasksModel.toggleAssignee(taskId);
  2024. ctx.redirect('/tasks');
  2025. })
  2026. .post('/tasks/delete/:id', koaBody(), async (ctx) => {
  2027. const taskId = ctx.params.id;
  2028. await tasksModel.deleteTaskById(taskId);
  2029. ctx.redirect('/tasks?filter=mine');
  2030. })
  2031. .post('/tasks/status/:id', koaBody(), async (ctx) => {
  2032. const taskId = ctx.params.id;
  2033. const { status } = ctx.request.body;
  2034. await tasksModel.updateTaskStatus(taskId, status);
  2035. ctx.redirect('/tasks?filter=mine');
  2036. })
  2037. .post('/reports/create', koaBody({ multipart: true }), async ctx => {
  2038. const { title, description, category, tags, severity } = ctx.request.body;
  2039. const image = await handleBlobUpload(ctx, 'image');
  2040. await reportsModel.createReport(title, description, category, image, tags, severity);
  2041. ctx.redirect('/reports');
  2042. })
  2043. .post('/reports/update/:id', koaBody({ multipart: true }), async ctx => {
  2044. const { title, description, category, tags, severity } = ctx.request.body;
  2045. const image = await handleBlobUpload(ctx, 'image');
  2046. await reportsModel.updateReportById(ctx.params.id, {
  2047. title, description, category, image, tags, severity
  2048. });
  2049. ctx.redirect('/reports?filter=mine');
  2050. })
  2051. .post('/reports/delete/:id', async ctx => {
  2052. await reportsModel.deleteReportById(ctx.params.id);
  2053. ctx.redirect('/reports?filter=mine');
  2054. })
  2055. .post('/reports/confirm/:id', async ctx => {
  2056. await reportsModel.confirmReportById(ctx.params.id);
  2057. ctx.redirect('/reports');
  2058. })
  2059. .post('/reports/status/:id', koaBody(), async (ctx) => {
  2060. const reportId = ctx.params.id;
  2061. const { status } = ctx.request.body;
  2062. await reportsModel.updateReportById(reportId, { status });
  2063. ctx.redirect('/reports?filter=mine');
  2064. })
  2065. .post('/events/create', koaBody(), async (ctx) => {
  2066. const { title, description, date, location, price, url, attendees, tags, isPublic } = ctx.request.body;
  2067. await eventsModel.createEvent(title, description, date, location, price, url, attendees, tags, isPublic);
  2068. ctx.redirect('/events?filter=mine');
  2069. })
  2070. .post('/events/update/:id', koaBody(), async (ctx) => {
  2071. const { title, description, date, location, price, url, attendees, tags, isPublic } = ctx.request.body;
  2072. const eventId = ctx.params.id;
  2073. const event = await eventsModel.getEventById(eventId);
  2074. await eventsModel.updateEventById(eventId, {
  2075. title,
  2076. description,
  2077. date,
  2078. location,
  2079. price,
  2080. url,
  2081. attendees,
  2082. tags,
  2083. isPublic,
  2084. createdAt: event.createdAt,
  2085. organizer: event.organizer,
  2086. });
  2087. ctx.redirect('/events?filter=mine');
  2088. })
  2089. .post('/events/attend/:id', koaBody(), async (ctx) => {
  2090. const eventId = ctx.params.id;
  2091. await eventsModel.toggleAttendee(eventId);
  2092. ctx.redirect('/events');
  2093. })
  2094. .post('/events/delete/:id', koaBody(), async (ctx) => {
  2095. const eventId = ctx.params.id;
  2096. await eventsModel.deleteEventById(eventId);
  2097. ctx.redirect('/events?filter=mine');
  2098. })
  2099. .post('/votes/create', koaBody(), async ctx => {
  2100. const { question, deadline, options, tags = '' } = ctx.request.body;
  2101. const defaultOptions = ['YES', 'NO', 'ABSTENTION', 'CONFUSED', 'FOLLOW_MAJORITY', 'NOT_INTERESTED'];
  2102. const parsedOptions = options
  2103. ? options.split(',').map(o => o.trim()).filter(Boolean)
  2104. : defaultOptions;
  2105. const parsedTags = tags.split(',').map(t => t.trim()).filter(Boolean);
  2106. await votesModel.createVote(question, deadline, parsedOptions, parsedTags);
  2107. ctx.redirect('/votes');
  2108. })
  2109. .post('/votes/update/:id', koaBody(), async ctx => {
  2110. const id = ctx.params.id;
  2111. const { question, deadline, options, tags = '' } = ctx.request.body;
  2112. const parsedOptions = options
  2113. ? options.split(',').map(o => o.trim()).filter(Boolean)
  2114. : undefined;
  2115. const parsedTags = tags ? tags.split(',').map(t => t.trim()).filter(Boolean) : [];
  2116. await votesModel.updateVoteById(id, { question, deadline, options: parsedOptions, tags: parsedTags });
  2117. ctx.redirect('/votes?filter=mine');
  2118. })
  2119. .post('/votes/delete/:id', koaBody(), async ctx => {
  2120. const id = ctx.params.id;
  2121. await votesModel.deleteVoteById(id);
  2122. ctx.redirect('/votes?filter=mine');
  2123. })
  2124. .post('/votes/vote/:id', koaBody(), async ctx => {
  2125. const id = ctx.params.id;
  2126. const { choice } = ctx.request.body;
  2127. await votesModel.voteOnVote(id, choice);
  2128. ctx.redirect('/votes?filter=open');
  2129. })
  2130. .post('/votes/opinions/:voteId/:category', async (ctx) => {
  2131. const { voteId, category } = ctx.params;
  2132. const voterId = SSBconfig?.keys?.id;
  2133. const vote = await votesModel.getVoteById(voteId);
  2134. if (vote.opinions_inhabitants && vote.opinions_inhabitants.includes(voterId)) {
  2135. ctx.flash = { message: "You have already opined." };
  2136. ctx.redirect('/votes');
  2137. return;
  2138. }
  2139. await votesModel.createOpinion(voteId, category);
  2140. ctx.redirect('/votes');
  2141. })
  2142. .post('/market/create', koaBody({ multipart: true }), async ctx => {
  2143. const { item_type, title, description, price, tags, item_status, deadline, includesShipping, stock } = ctx.request.body;
  2144. const image = await handleBlobUpload(ctx, 'image');
  2145. if (!stock || stock <= 0) {
  2146. ctx.throw(400, 'Stock must be a positive number.');
  2147. }
  2148. await marketModel.createItem(item_type, title, description, image, price, tags, item_status, deadline, includesShipping, stock);
  2149. ctx.redirect('/market');
  2150. })
  2151. .post('/market/update/:id', koaBody({ multipart: true }), async ctx => {
  2152. const id = ctx.params.id;
  2153. const { item_type, title, description, price, tags = '', item_status, deadline, includesShipping, stock } = ctx.request.body;
  2154. const parsedTags = tags.split(',').map(t => t.trim()).filter(Boolean);
  2155. if (stock < 0) {
  2156. ctx.throw(400, 'Stock cannot be negative.');
  2157. }
  2158. const updatedData = {
  2159. item_type,
  2160. title,
  2161. description,
  2162. price,
  2163. item_status,
  2164. deadline,
  2165. includesShipping,
  2166. tags: parsedTags,
  2167. stock
  2168. };
  2169. const image = await handleBlobUpload(ctx, 'image');
  2170. updatedData.image = image;
  2171. await marketModel.updateItemById(id, updatedData);
  2172. ctx.redirect('/market?filter=mine');
  2173. })
  2174. .post('/market/delete/:id', koaBody(), async ctx => {
  2175. const id = ctx.params.id;
  2176. await marketModel.deleteItemById(id);
  2177. ctx.redirect('/market?filter=mine');
  2178. })
  2179. .post('/market/sold/:id', koaBody(), async ctx => {
  2180. const id = ctx.params.id;
  2181. const marketItem = await marketModel.getItemById(id);
  2182. if (marketItem.stock <= 0) {
  2183. ctx.throw(400, 'No stock left to mark as sold.');
  2184. }
  2185. if (marketItem.status !== 'SOLD') {
  2186. await marketModel.setItemAsSold(id);
  2187. await marketModel.decrementStock(id);
  2188. }
  2189. ctx.redirect('/market?filter=mine');
  2190. })
  2191. .post('/market/buy/:id', koaBody(), async ctx => {
  2192. const id = ctx.params.id;
  2193. const marketItem = await marketModel.getItemById(id);
  2194. if (marketItem.item_type === 'exchange') {
  2195. if (marketItem.status !== 'SOLD') {
  2196. const buyerId = ctx.request.body.buyerId;
  2197. const { price, title, seller } = marketItem;
  2198. const subject = `Your item "${title}" has been sold`;
  2199. const text = `The item with title: "${title}" has been sold. The buyer with OASIS ID: ${buyerId} purchased it for: $${price}.`;
  2200. await pmModel.sendMessage([seller], subject, text);
  2201. await marketModel.setItemAsSold(id);
  2202. }
  2203. }
  2204. await marketModel.decrementStock(id);
  2205. ctx.redirect('/inbox?filter=sent');
  2206. })
  2207. .post('/market/bid/:id', koaBody(), async ctx => {
  2208. const id = ctx.params.id;
  2209. const userId = SSBconfig.config.keys.id;
  2210. const { bidAmount } = ctx.request.body;
  2211. const marketItem = await marketModel.getItemById(id);
  2212. await marketModel.addBidToAuction(id, userId, bidAmount);
  2213. if (marketItem.stock > 0 && marketItem.status === 'SOLD') {
  2214. await marketModel.decrementStock(id);
  2215. }
  2216. ctx.redirect('/market?filter=auctions');
  2217. })
  2218. .post('/jobs/create', koaBody({ multipart: true }), async (ctx) => {
  2219. const {
  2220. job_type,
  2221. title,
  2222. description,
  2223. requirements,
  2224. languages,
  2225. job_time,
  2226. tasks,
  2227. location,
  2228. vacants,
  2229. salary
  2230. } = ctx.request.body;
  2231. const imageBlob = ctx.request.files?.image
  2232. ? await handleBlobUpload(ctx, 'image')
  2233. : null;
  2234. await jobsModel.createJob({
  2235. job_type,
  2236. title,
  2237. description,
  2238. requirements,
  2239. languages,
  2240. job_time,
  2241. tasks,
  2242. location,
  2243. vacants: vacants ? parseInt(vacants, 10) : 1,
  2244. salary: salary != null ? parseFloat(salary) : 0,
  2245. image: imageBlob
  2246. });
  2247. ctx.redirect('/jobs?filter=MINE');
  2248. })
  2249. .post('/jobs/update/:id', koaBody({ multipart: true }), async (ctx) => {
  2250. const id = ctx.params.id;
  2251. const {
  2252. job_type,
  2253. title,
  2254. description,
  2255. requirements,
  2256. languages,
  2257. job_time,
  2258. tasks,
  2259. location,
  2260. vacants,
  2261. salary
  2262. } = ctx.request.body;
  2263. const imageBlob = ctx.request.files?.image
  2264. ? await handleBlobUpload(ctx, 'image')
  2265. : undefined;
  2266. await jobsModel.updateJob(id, {
  2267. job_type,
  2268. title,
  2269. description,
  2270. requirements,
  2271. languages,
  2272. job_time,
  2273. tasks,
  2274. location,
  2275. vacants: vacants ? parseInt(vacants, 10) : undefined,
  2276. salary: salary != null && salary !== '' ? parseFloat(salary) : undefined,
  2277. image: imageBlob
  2278. });
  2279. ctx.redirect('/jobs?filter=MINE');
  2280. })
  2281. .post('/jobs/delete/:id', koaBody(), async (ctx) => {
  2282. const id = ctx.params.id;
  2283. await jobsModel.deleteJob(id);
  2284. ctx.redirect('/jobs?filter=MINE');
  2285. })
  2286. .post('/jobs/status/:id', koaBody(), async (ctx) => {
  2287. const id = ctx.params.id;
  2288. const { status } = ctx.request.body;
  2289. await jobsModel.updateJobStatus(id, String(status).toUpperCase());
  2290. ctx.redirect('/jobs?filter=MINE');
  2291. })
  2292. .post('/jobs/subscribe/:id', koaBody(), async (ctx) => {
  2293. const id = ctx.params.id;
  2294. await jobsModel.subscribeToJob(id, config.keys.id);
  2295. ctx.redirect('/jobs');
  2296. })
  2297. .post('/jobs/unsubscribe/:id', koaBody(), async (ctx) => {
  2298. const id = ctx.params.id;
  2299. await jobsModel.unsubscribeFromJob(id, config.keys.id);
  2300. ctx.redirect('/jobs');
  2301. })
  2302. // UPDATE OASIS
  2303. .post("/update", koaBody(), async (ctx) => {
  2304. const util = require("node:util");
  2305. const exec = util.promisify(require("node:child_process").exec);
  2306. async function updateTool() {
  2307. const { stdout, stderr } = await exec("git reset --hard && git pull");
  2308. console.log("oasis@version: updating Oasis...");
  2309. console.log(stdout);
  2310. console.log(stderr);
  2311. const { stdout: shOut, stderr: shErr } = await exec("sh install.sh");
  2312. console.log("oasis@version: running install.sh...");
  2313. console.log(shOut);
  2314. console.error(shErr);
  2315. }
  2316. await updateTool();
  2317. const referer = new URL(ctx.request.header.referer);
  2318. ctx.redirect(referer.href);
  2319. })
  2320. .post("/settings/theme", koaBody(), async (ctx) => {
  2321. const theme = String(ctx.request.body.theme || "").trim();
  2322. const currentConfig = getConfig();
  2323. currentConfig.themes.current = theme || "Dark-SNH";
  2324. fs.writeFileSync(configPath, JSON.stringify(currentConfig, null, 2));
  2325. ctx.cookies.set("theme", currentConfig.themes.current);
  2326. ctx.redirect("/settings");
  2327. })
  2328. .post("/language", koaBody(), async (ctx) => {
  2329. const language = String(ctx.request.body.language);
  2330. ctx.cookies.set("language", language);
  2331. const referer = new URL(ctx.request.header.referer);
  2332. ctx.redirect(referer.href);
  2333. })
  2334. .post("/settings/conn/start", koaBody(), async (ctx) => {
  2335. await meta.connStart();
  2336. ctx.redirect("/peers");
  2337. })
  2338. .post("/settings/conn/stop", koaBody(), async (ctx) => {
  2339. await meta.connStop();
  2340. ctx.redirect("/peers");
  2341. })
  2342. .post("/settings/conn/sync", koaBody(), async (ctx) => {
  2343. await meta.sync();
  2344. ctx.redirect("/peers");
  2345. })
  2346. .post("/settings/conn/restart", koaBody(), async (ctx) => {
  2347. await meta.connRestart();
  2348. ctx.redirect("/peers");
  2349. })
  2350. .post("/settings/invite/accept", koaBody(), async (ctx) => {
  2351. try {
  2352. const invite = String(ctx.request.body.invite);
  2353. await meta.acceptInvite(invite);
  2354. } catch (e) {
  2355. }
  2356. ctx.redirect("/invites");
  2357. })
  2358. .post("/settings/ssb-logstream", koaBody(), async (ctx) => {
  2359. const logLimit = parseInt(ctx.request.body.ssb_log_limit, 10);
  2360. if (!isNaN(logLimit) && logLimit > 0 && logLimit <= 100000) {
  2361. const configData = fs.readFileSync(configPath, 'utf8');
  2362. const config = JSON.parse(configData);
  2363. if (!config.ssbLogStream) config.ssbLogStream = {};
  2364. config.ssbLogStream.limit = logLimit;
  2365. fs.writeFileSync(configPath, JSON.stringify(config, null, 2));
  2366. }
  2367. ctx.redirect("/settings");
  2368. })
  2369. .post("/settings/rebuild", async (ctx) => {
  2370. meta.rebuild();
  2371. ctx.redirect("/settings");
  2372. })
  2373. .post("/save-modules", koaBody(), async (ctx) => {
  2374. const modules = [
  2375. 'popular', 'topics', 'summaries', 'latest', 'threads', 'multiverse', 'invites', 'wallet',
  2376. 'legacy', 'cipher', 'bookmarks', 'videos', 'docs', 'audios', 'tags', 'images', 'trending',
  2377. 'events', 'tasks', 'market', 'tribes', 'governance', 'reports', 'opinions', 'transfers',
  2378. 'feed', 'pixelia', 'agenda', 'ai', 'forum', 'jobs'
  2379. ];
  2380. const currentConfig = getConfig();
  2381. modules.forEach(mod => {
  2382. const modKey = `${mod}Mod`;
  2383. const formKey = `${mod}Form`;
  2384. const modValue = ctx.request.body[formKey] === 'on' ? 'on' : 'off';
  2385. currentConfig.modules[modKey] = modValue;
  2386. });
  2387. saveConfig(currentConfig);
  2388. ctx.redirect(`/modules`);
  2389. })
  2390. .post("/settings/ai", koaBody(), async (ctx) => {
  2391. const aiPrompt = String(ctx.request.body.ai_prompt || "").trim();
  2392. if (aiPrompt.length > 128) {
  2393. ctx.status = 400;
  2394. ctx.body = "Prompt too long. Must be 128 characters or fewer.";
  2395. return;
  2396. }
  2397. const currentConfig = getConfig();
  2398. currentConfig.ai = currentConfig.ai || {};
  2399. currentConfig.ai.prompt = aiPrompt;
  2400. saveConfig(currentConfig);
  2401. const referer = new URL(ctx.request.header.referer);
  2402. ctx.redirect("/settings");
  2403. })
  2404. .post('/transfers/create',
  2405. koaBody(),
  2406. async ctx => {
  2407. const { to, concept, amount, deadline, tags } = ctx.request.body
  2408. await transfersModel.createTransfer(to, concept, amount, deadline, tags)
  2409. ctx.redirect('/transfers')
  2410. })
  2411. .post('/transfers/update/:id',
  2412. koaBody(),
  2413. async ctx => {
  2414. const { to, concept, amount, deadline, tags } = ctx.request.body
  2415. await transfersModel.updateTransferById(
  2416. ctx.params.id, to, concept, amount, deadline, tags
  2417. )
  2418. ctx.redirect('/transfers?filter=mine')
  2419. })
  2420. .post('/transfers/confirm/:id', async ctx => {
  2421. await transfersModel.confirmTransferById(ctx.params.id)
  2422. ctx.redirect('/transfers')
  2423. })
  2424. .post('/transfers/delete/:id', async ctx => {
  2425. await transfersModel.deleteTransferById(ctx.params.id)
  2426. ctx.redirect('/transfers?filter=mine')
  2427. })
  2428. .post('/transfers/opinions/:transferId/:category', async ctx => {
  2429. const { transferId, category } = ctx.params
  2430. const voterId = SSBconfig?.keys?.id;
  2431. const t = await transfersModel.getTransferById(transferId)
  2432. if (t.opinions_inhabitants.includes(voterId)) {
  2433. ctx.flash = { message: 'You have already opined.' }
  2434. ctx.redirect('/transfers')
  2435. return
  2436. }
  2437. await opinionsModel.createVote(transferId, category, 'transfer')
  2438. ctx.redirect('/transfers')
  2439. })
  2440. .post("/settings/wallet", koaBody(), async (ctx) => {
  2441. const url = String(ctx.request.body.wallet_url);
  2442. const user = String(ctx.request.body.wallet_user);
  2443. const pass = String(ctx.request.body.wallet_pass);
  2444. const fee = String(ctx.request.body.wallet_fee);
  2445. const currentConfig = getConfig();
  2446. if (url) currentConfig.wallet.url = url;
  2447. if (user) currentConfig.wallet.user = user;
  2448. if (pass) currentConfig.wallet.pass = pass;
  2449. if (fee) currentConfig.wallet.fee = fee;
  2450. saveConfig(currentConfig);
  2451. const referer = new URL(ctx.request.header.referer);
  2452. ctx.redirect(referer.href);
  2453. })
  2454. .post("/wallet/send", koaBody(), async (ctx) => {
  2455. const action = String(ctx.request.body.action);
  2456. const destination = String(ctx.request.body.destination);
  2457. const amount = Number(ctx.request.body.amount);
  2458. const fee = Number(ctx.request.body.fee);
  2459. const { url, user, pass } = getConfig().wallet;
  2460. let balance = null
  2461. try {
  2462. balance = await walletModel.getBalance(url, user, pass);
  2463. } catch (error) {
  2464. ctx.body = await walletErrorView(error);
  2465. }
  2466. switch (action) {
  2467. case 'confirm':
  2468. const validation = await walletModel.validateSend(url, user, pass, destination, amount, fee);
  2469. if (validation.isValid) {
  2470. try {
  2471. ctx.body = await walletSendConfirmView(balance, destination, amount, fee);
  2472. } catch (error) {
  2473. ctx.body = await walletErrorView(error);
  2474. }
  2475. } else {
  2476. try {
  2477. const statusMessages = {
  2478. type: 'error',
  2479. title: 'validation_errors',
  2480. messages: validation.errors,
  2481. }
  2482. ctx.body = await walletSendFormView(balance, destination, amount, fee, statusMessages);
  2483. } catch (error) {
  2484. ctx.body = await walletErrorView(error);
  2485. }
  2486. }
  2487. break;
  2488. case 'send':
  2489. try {
  2490. const txId = await walletModel.sendToAddress(url, user, pass, destination, amount);
  2491. ctx.body = await walletSendResultView(balance, destination, amount, txId);
  2492. } catch (error) {
  2493. ctx.body = await walletErrorView(error);
  2494. }
  2495. break;
  2496. }
  2497. });
  2498. const routes = router.routes();
  2499. const middleware = [
  2500. async (ctx, next) => {
  2501. if (config.public && ctx.method !== "GET") {
  2502. throw new Error(
  2503. "Sorry, many actions are unavailable when Oasis is running in public mode. Please run Oasis in the default mode and try again."
  2504. );
  2505. }
  2506. await next();
  2507. },
  2508. async (ctx, next) => {
  2509. const selectedLanguage = ctx.cookies.get("language") || "en";
  2510. setLanguage(selectedLanguage);
  2511. await next();
  2512. },
  2513. async (ctx, next) => {
  2514. const ssb = await cooler.open();
  2515. const status = await ssb.status();
  2516. const values = Object.values(status.sync.plugins);
  2517. const totalCurrent = Object.values(status.sync.plugins).reduce(
  2518. (acc, cur) => acc + cur,
  2519. 0
  2520. );
  2521. const totalTarget = status.sync.since * values.length;
  2522. const left = totalTarget - totalCurrent;
  2523. const percent = Math.floor((totalCurrent / totalTarget) * 1000) / 10;
  2524. const megabyte = 1024 * 1024;
  2525. if (left > megabyte) {
  2526. ctx.response.body = indexingView({ percent });
  2527. } else {
  2528. try {
  2529. await next();
  2530. } catch (err) {
  2531. ctx.status = err.status || 500;
  2532. ctx.body = { message: err.message || 'Internal Server Error' };
  2533. }
  2534. }
  2535. },
  2536. routes,
  2537. ];
  2538. const { allowHost } = config;
  2539. const app = http({ host, port, middleware, allowHost });
  2540. app._close = () => {
  2541. nameWarmup.close();
  2542. cooler.close();
  2543. };
  2544. module.exports = app;
  2545. if (config.open === true) {
  2546. open(url);
  2547. }