Compare commits
12 Commits
93d5b6b5f3
..
master
| Author | SHA1 | Date | |
|---|---|---|---|
| ef7c24c884 | |||
| f47ca67dae | |||
| 3737edab72 | |||
| 5195317c0c | |||
| 8aa1f3addd | |||
| af4471b463 | |||
| f0afa200b1 | |||
| 503c5ef1f4 | |||
| e8dd905f27 | |||
| c5fd09d71d | |||
| 989fdce883 | |||
| fd82643477 |
@@ -27,6 +27,13 @@ const chatDB = (DB) => {
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
DB.getChatParticipants = async () => {
|
||||||
|
return DB.chatMessagesCol.distinct("senderProfileId").catch((err) => {
|
||||||
|
console.log(err);
|
||||||
|
return [];
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
DB.getRecentChatMessages = async (limit = 100) => {
|
DB.getRecentChatMessages = async (limit = 100) => {
|
||||||
const safeLimit = Math.min(Math.max(parseInt(limit, 10) || 100, 1), 200);
|
const safeLimit = Math.min(Math.max(parseInt(limit, 10) || 100, 1), 200);
|
||||||
const messages = await DB.chatMessagesCol.find({})
|
const messages = await DB.chatMessagesCol.find({})
|
||||||
|
|||||||
@@ -38,6 +38,20 @@ postDB = (DB)=>{
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
DB.addTranslation = (postid, lang, translatedText) => {
|
||||||
|
if(!DB.ObjectID.isValid(postid)) return false;
|
||||||
|
const id = DB.ObjectID(postid);
|
||||||
|
let update = {
|
||||||
|
$set:{
|
||||||
|
["translations." + lang]: translatedText
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return DB.postCols.updateOne({_id: id}, update).catch((err)=>{
|
||||||
|
console.log(err);
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
DB.newReaction = (postid, profileid, reaction) => {
|
DB.newReaction = (postid, profileid, reaction) => {
|
||||||
if(!DB.ObjectID.isValid(postid)) return false;
|
if(!DB.ObjectID.isValid(postid)) return false;
|
||||||
const id = DB.ObjectID(postid);
|
const id = DB.ObjectID(postid);
|
||||||
|
|||||||
@@ -29,6 +29,7 @@ const limiter = rateLimit({
|
|||||||
limit: 500, // Limit each IP to 100 requests per `window` (here, per 15 minutes).
|
limit: 500, // Limit each IP to 100 requests per `window` (here, per 15 minutes).
|
||||||
standardHeaders: 'draft-8', // draft-6: `RateLimit-*` headers; draft-7 & draft-8: combined `RateLimit` header
|
standardHeaders: 'draft-8', // draft-6: `RateLimit-*` headers; draft-7 & draft-8: combined `RateLimit` header
|
||||||
legacyHeaders: false, // Disable the `X-RateLimit-*` headers.
|
legacyHeaders: false, // Disable the `X-RateLimit-*` headers.
|
||||||
|
skip: (req) => req.path.startsWith("/live-captions"),
|
||||||
keyGenerator: (req) => {
|
keyGenerator: (req) => {
|
||||||
const forwarded = req.headers["x-forwarded-for"]?.split(",")[0]; // Take the first IP in the list
|
const forwarded = req.headers["x-forwarded-for"]?.split(",")[0]; // Take the first IP in the list
|
||||||
const ip = forwarded || req.ip; // Fallback to req.ip
|
const ip = forwarded || req.ip; // Fallback to req.ip
|
||||||
@@ -184,6 +185,7 @@ const songsRoute = require('./routes/songs.js');
|
|||||||
const paymentsRoute = require('./routes/payments.js');
|
const paymentsRoute = require('./routes/payments.js');
|
||||||
const bibleRoute = require('./routes/bible.js');
|
const bibleRoute = require('./routes/bible.js');
|
||||||
const chatRoute = require('./routes/chat.js');
|
const chatRoute = require('./routes/chat.js');
|
||||||
|
const liveCaptionsRoute = require('./routes/liveCaptions.js');
|
||||||
const sessionChecker = require('./middleware/sessionChecker');
|
const sessionChecker = require('./middleware/sessionChecker');
|
||||||
// -- Private Routes
|
// -- Private Routes
|
||||||
app.use('/user', sessionChecker, profileRoute);
|
app.use('/user', sessionChecker, profileRoute);
|
||||||
@@ -192,6 +194,7 @@ app.use('/payments', paymentsRoute);
|
|||||||
app.use('/bible', sessionChecker, bibleRoute);
|
app.use('/bible', sessionChecker, bibleRoute);
|
||||||
app.use('/songs', sessionChecker, songsRoute);
|
app.use('/songs', sessionChecker, songsRoute);
|
||||||
app.use('/chat', sessionChecker, chatRoute);
|
app.use('/chat', sessionChecker, chatRoute);
|
||||||
|
app.use('/live-captions', liveCaptionsRoute);
|
||||||
// -- Public Routes
|
// -- Public Routes
|
||||||
const subsplashRoute = require('./routes/subsplash.js');
|
const subsplashRoute = require('./routes/subsplash.js');
|
||||||
app.use('/subsplash', subsplashRoute);
|
app.use('/subsplash', subsplashRoute);
|
||||||
|
|||||||
@@ -489,6 +489,26 @@ const Notifications = {
|
|||||||
// sendWebNotification(requesterProfile.webSubscription, notifBody);
|
// sendWebNotification(requesterProfile.webSubscription, notifBody);
|
||||||
DB.addNotification(requesterProfile, notifBody, null, null, groupProfile._id);
|
DB.addNotification(requesterProfile, notifBody, null, null, groupProfile._id);
|
||||||
},
|
},
|
||||||
|
async youGotANewChatMessage(senderProfileId, messageText) {
|
||||||
|
const DB = await DBGetter.getDB;
|
||||||
|
const participants = await DB.getChatParticipants();
|
||||||
|
const senderProfile = await DB.getProfileCache(senderProfileId);
|
||||||
|
|
||||||
|
const tokens = [];
|
||||||
|
for (const participantProfileId of participants) {
|
||||||
|
if (participantProfileId.toString() === senderProfileId.toString()) continue;
|
||||||
|
|
||||||
|
const participantProfile = await DB.getProfileCache(participantProfileId);
|
||||||
|
if (participantProfile && Array.isArray(participantProfile.token)) {
|
||||||
|
tokens.push(...participantProfile.token);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tokens.length > 0) {
|
||||||
|
const notifBody = `${senderProfile.profile.firstName}: ${messageText.substring(0, 100)}${messageText.length > 100 ? '...' : ''}`;
|
||||||
|
sendPushNotification(tokens, notifBody, { type: 'chat' });
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -6,6 +6,8 @@
|
|||||||
"scripts": {
|
"scripts": {
|
||||||
"test": "npx mocha test/auth.test.js",
|
"test": "npx mocha test/auth.test.js",
|
||||||
"start": "node index.js",
|
"start": "node index.js",
|
||||||
|
"dev": "node --watch index.js",
|
||||||
|
"live-captions:test-sender": "node scripts/liveCaptionsTestSender.js",
|
||||||
"docker": "docker compose up -d",
|
"docker": "docker compose up -d",
|
||||||
"docker_restore": "docker-compose exec mongo mongorestore --db EMI_SOCIAL /dump/EMI_SOCIAL/",
|
"docker_restore": "docker-compose exec mongo mongorestore --db EMI_SOCIAL /dump/EMI_SOCIAL/",
|
||||||
"docker_dump": "docker-compose exec mongo mongodump --uri ${MONGO_URL} --out /dump"
|
"docker_dump": "docker-compose exec mongo mongodump --uri ${MONGO_URL} --out /dump"
|
||||||
|
|||||||
+2
-1
@@ -148,7 +148,8 @@ DB.getDB.then((DB) => {
|
|||||||
router.get("/chapters/:chapterId", async (req, res) => {
|
router.get("/chapters/:chapterId", async (req, res) => {
|
||||||
const chapterId = req.params.chapterId;
|
const chapterId = req.params.chapterId;
|
||||||
const bibleId = req.query.bibleId || defaultBibleId;
|
const bibleId = req.query.bibleId || defaultBibleId;
|
||||||
const bibles = await fetchAPI('bibles/' + bibleId + "/chapters/" + chapterId);
|
const contentType = req.query['content-type'] ? `?content-type=${req.query['content-type']}` : '';
|
||||||
|
const bibles = await fetchAPI('bibles/' + bibleId + "/chapters/" + chapterId + contentType);
|
||||||
return res.json(bibles);
|
return res.json(bibles);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ var express = require('express');
|
|||||||
var router = express.Router();
|
var router = express.Router();
|
||||||
|
|
||||||
const DB = require("../mongoDB.js");
|
const DB = require("../mongoDB.js");
|
||||||
|
const Notifications = require("../notifications.js");
|
||||||
const { getUserId, getProfileId } = require("../utils/sessionUtils.js");
|
const { getUserId, getProfileId } = require("../utils/sessionUtils.js");
|
||||||
const { normalizeLanguageCode, translateText } = require("../utils/chatTranslation.js");
|
const { normalizeLanguageCode, translateText } = require("../utils/chatTranslation.js");
|
||||||
|
|
||||||
@@ -190,6 +191,8 @@ DB.getDB.then((DB) => {
|
|||||||
return res.status(500).json({ status: "Could not save message" });
|
return res.status(500).json({ status: "Could not save message" });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Notifications.youGotANewChatMessage(profileId, text);
|
||||||
|
|
||||||
activeUsers.set(profileId + "", {
|
activeUsers.set(profileId + "", {
|
||||||
profileId: profileId + "",
|
profileId: profileId + "",
|
||||||
userId: userId + "",
|
userId: userId + "",
|
||||||
|
|||||||
@@ -0,0 +1,247 @@
|
|||||||
|
var express = require('express');
|
||||||
|
var router = express.Router();
|
||||||
|
const { rateLimit } = require("express-rate-limit");
|
||||||
|
|
||||||
|
const sessionChecker = require("../middleware/sessionChecker.js");
|
||||||
|
|
||||||
|
const MAX_BUFFER_SIZE = 300;
|
||||||
|
const DEFAULT_INITIAL_LIMIT = 40;
|
||||||
|
const MAX_INITIAL_LIMIT = 120;
|
||||||
|
const INACTIVITY_RESET_MS = 10 * 60 * 1000;
|
||||||
|
const CAPTION_META_KEYS = new Set(["sequence", "createdAt", "original", "draft", "sourceLang", "lang", "isDraft", "status", "translations"]);
|
||||||
|
|
||||||
|
const liveCaptionState = {
|
||||||
|
startedAt: Date.now(),
|
||||||
|
lastIngestAt: 0,
|
||||||
|
latestSequence: 0,
|
||||||
|
captions: [],
|
||||||
|
};
|
||||||
|
|
||||||
|
const liveCaptionsLimiter = rateLimit({
|
||||||
|
windowMs: 10 * 60 * 1000,
|
||||||
|
limit: 6000,
|
||||||
|
standardHeaders: "draft-8",
|
||||||
|
legacyHeaders: false,
|
||||||
|
keyGenerator: (req) => {
|
||||||
|
const forwarded = req.headers["x-forwarded-for"]?.split(",")[0];
|
||||||
|
const ip = forwarded || req.ip || "";
|
||||||
|
return ip.includes(":") ? ip.split(":")[0] : ip;
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
router.use(liveCaptionsLimiter);
|
||||||
|
|
||||||
|
const normalizeLang = (lang = "") => {
|
||||||
|
const value = String(lang || "").trim().toLowerCase();
|
||||||
|
if (!value) return "";
|
||||||
|
const base = value.split(",")[0].split("-")[0].trim();
|
||||||
|
return base || value;
|
||||||
|
};
|
||||||
|
|
||||||
|
const normalizeTranslations = (translations) => {
|
||||||
|
if (!translations || typeof translations !== "object" || Array.isArray(translations)) return {};
|
||||||
|
const normalized = {};
|
||||||
|
for (const [langKey, translatedText] of Object.entries(translations)) {
|
||||||
|
const lang = normalizeLang(langKey);
|
||||||
|
const text = typeof translatedText === "string" ? translatedText.trim() : "";
|
||||||
|
if (!lang || !text) continue;
|
||||||
|
normalized[lang] = text;
|
||||||
|
}
|
||||||
|
return normalized;
|
||||||
|
};
|
||||||
|
|
||||||
|
const readText = (value) => {
|
||||||
|
if (typeof value === "string") return value.trim();
|
||||||
|
return "";
|
||||||
|
};
|
||||||
|
|
||||||
|
const extractDraftText = (body = {}) => {
|
||||||
|
const directDraft = readText(body?.draft);
|
||||||
|
if (directDraft) return directDraft;
|
||||||
|
const nestedDraft = readText(body?.draft?.text);
|
||||||
|
if (nestedDraft) return nestedDraft;
|
||||||
|
const fallbackText = readText(body?.text);
|
||||||
|
if (fallbackText) return fallbackText;
|
||||||
|
return "";
|
||||||
|
};
|
||||||
|
|
||||||
|
const buildTranslationsFromFlatPayload = (payload) => {
|
||||||
|
const ignoredKeys = new Set(["original", "draft", "sourceLang", "lang", "isDraft", "status", "translations"]);
|
||||||
|
const normalized = {};
|
||||||
|
for (const [key, value] of Object.entries(payload || {})) {
|
||||||
|
if (ignoredKeys.has(key)) continue;
|
||||||
|
const lang = normalizeLang(key);
|
||||||
|
const text = typeof value === "string" ? value.trim() : "";
|
||||||
|
if (!lang || !text) continue;
|
||||||
|
normalized[lang] = text;
|
||||||
|
}
|
||||||
|
return normalized;
|
||||||
|
};
|
||||||
|
|
||||||
|
const inferSourceLangFromTranslations = (original, translations) => {
|
||||||
|
const normalizedOriginal = String(original || "").trim();
|
||||||
|
if (!normalizedOriginal) return "original";
|
||||||
|
for (const [lang, text] of Object.entries(translations || {})) {
|
||||||
|
if (String(text || "").trim() === normalizedOriginal) return normalizeLang(lang);
|
||||||
|
}
|
||||||
|
return "original";
|
||||||
|
};
|
||||||
|
|
||||||
|
const getAvailableLanguages = () => {
|
||||||
|
const langs = new Set();
|
||||||
|
for (const caption of liveCaptionState.captions) {
|
||||||
|
Object.keys(caption || {}).forEach((key) => {
|
||||||
|
if (CAPTION_META_KEYS.has(key)) return;
|
||||||
|
const lang = normalizeLang(key);
|
||||||
|
if (lang) langs.add(lang);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return Array.from(langs).filter(Boolean).sort();
|
||||||
|
};
|
||||||
|
|
||||||
|
const resetLiveCaptionState = () => {
|
||||||
|
liveCaptionState.startedAt = Date.now();
|
||||||
|
liveCaptionState.lastIngestAt = 0;
|
||||||
|
liveCaptionState.latestSequence = 0;
|
||||||
|
liveCaptionState.captions = [];
|
||||||
|
};
|
||||||
|
|
||||||
|
const maybeResetForInactivity = () => {
|
||||||
|
if (!liveCaptionState.lastIngestAt) return;
|
||||||
|
if ((Date.now() - liveCaptionState.lastIngestAt) < INACTIVITY_RESET_MS) return;
|
||||||
|
resetLiveCaptionState();
|
||||||
|
};
|
||||||
|
|
||||||
|
router.get("/stream", async (req, res) => {
|
||||||
|
try {
|
||||||
|
maybeResetForInactivity();
|
||||||
|
const sinceSequence = Number.parseInt(req.query?.sinceSequence, 10);
|
||||||
|
const requestedLimit = Number.parseInt(req.query?.limit, 10);
|
||||||
|
const initialLimit = Number.isFinite(requestedLimit)
|
||||||
|
? Math.max(1, Math.min(requestedLimit, MAX_INITIAL_LIMIT))
|
||||||
|
: DEFAULT_INITIAL_LIMIT;
|
||||||
|
|
||||||
|
let captions = [];
|
||||||
|
if (Number.isFinite(sinceSequence) && sinceSequence >= 0) {
|
||||||
|
captions = liveCaptionState.captions.filter((item) => item.sequence > sinceSequence);
|
||||||
|
} else {
|
||||||
|
captions = liveCaptionState.captions.slice(-initialLimit);
|
||||||
|
}
|
||||||
|
|
||||||
|
return res.json({
|
||||||
|
status: "ok",
|
||||||
|
latestSequence: liveCaptionState.latestSequence,
|
||||||
|
startedAt: new Date(liveCaptionState.startedAt).toISOString(),
|
||||||
|
availableLanguages: getAvailableLanguages(),
|
||||||
|
captions,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error getting live captions stream", error);
|
||||||
|
return res.status(500).json({
|
||||||
|
status: "Internal server error",
|
||||||
|
latestSequence: liveCaptionState.latestSequence,
|
||||||
|
captions: [],
|
||||||
|
availableLanguages: [],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
router.post("/ingest", async (req, res) => {
|
||||||
|
try {
|
||||||
|
// TODO: Add basic auth/API key validation before production roll-out.
|
||||||
|
const draft = extractDraftText(req.body || {});
|
||||||
|
const originalFromPayload = readText(req.body?.original);
|
||||||
|
const original = originalFromPayload || draft;
|
||||||
|
const requestedLang = normalizeLang(req.body?.lang);
|
||||||
|
const sourceLangFromRequest = normalizeLang(req.body?.sourceLang || (requestedLang && requestedLang !== "draft" ? requestedLang : ""));
|
||||||
|
const isDraft = !!draft || requestedLang === "draft" || sourceLangFromRequest === "draft" || req.body?.isDraft === true || req.body?.status === "draft";
|
||||||
|
const mapFromNested = normalizeTranslations(req.body?.translations);
|
||||||
|
const mapFromFlat = buildTranslationsFromFlatPayload(req.body);
|
||||||
|
const translations = isDraft ? {} : { ...mapFromNested, ...mapFromFlat };
|
||||||
|
const inferredSource = inferSourceLangFromTranslations(original, translations);
|
||||||
|
const sourceLang = isDraft ? "" : (sourceLangFromRequest || inferredSource);
|
||||||
|
|
||||||
|
if (!original) {
|
||||||
|
return res.status(400).json({ status: "Original text is required" });
|
||||||
|
}
|
||||||
|
if (sourceLang && sourceLang !== "original" && sourceLang !== "draft" && !translations[sourceLang]) {
|
||||||
|
translations[sourceLang] = original;
|
||||||
|
}
|
||||||
|
|
||||||
|
const sequence = liveCaptionState.latestSequence + 1;
|
||||||
|
const caption = {
|
||||||
|
sequence,
|
||||||
|
createdAt: new Date().toISOString(),
|
||||||
|
original,
|
||||||
|
sourceLang: sourceLang || undefined,
|
||||||
|
lang: isDraft ? "draft" : (sourceLang || undefined),
|
||||||
|
isDraft,
|
||||||
|
status: isDraft ? "draft" : "final",
|
||||||
|
...translations,
|
||||||
|
};
|
||||||
|
|
||||||
|
liveCaptionState.latestSequence = sequence;
|
||||||
|
liveCaptionState.lastIngestAt = Date.now();
|
||||||
|
liveCaptionState.captions.push(caption);
|
||||||
|
if (liveCaptionState.captions.length > MAX_BUFFER_SIZE) {
|
||||||
|
liveCaptionState.captions.splice(0, liveCaptionState.captions.length - MAX_BUFFER_SIZE);
|
||||||
|
}
|
||||||
|
|
||||||
|
return res.json({
|
||||||
|
status: "ok",
|
||||||
|
caption,
|
||||||
|
latestSequence: liveCaptionState.latestSequence,
|
||||||
|
availableLanguages: getAvailableLanguages(),
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error ingesting live captions", error);
|
||||||
|
return res.status(500).json({ status: "Internal server error" });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
router.post("/reset", async (_, res) => {
|
||||||
|
try {
|
||||||
|
// TODO: Add admin authorization before exposing this endpoint.
|
||||||
|
resetLiveCaptionState();
|
||||||
|
return res.json({ status: "ok" });
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error resetting live captions state", error);
|
||||||
|
return res.status(500).json({ status: "Internal server error" });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
router.get("/public/stream", async (req, res) => {
|
||||||
|
try {
|
||||||
|
maybeResetForInactivity();
|
||||||
|
const sinceSequence = Number.parseInt(req.query?.sinceSequence, 10);
|
||||||
|
const requestedLimit = Number.parseInt(req.query?.limit, 10);
|
||||||
|
const initialLimit = Number.isFinite(requestedLimit)
|
||||||
|
? Math.max(1, Math.min(requestedLimit, MAX_INITIAL_LIMIT))
|
||||||
|
: DEFAULT_INITIAL_LIMIT;
|
||||||
|
|
||||||
|
let captions = [];
|
||||||
|
if (Number.isFinite(sinceSequence) && sinceSequence >= 0) {
|
||||||
|
captions = liveCaptionState.captions.filter((item) => item.sequence > sinceSequence);
|
||||||
|
} else {
|
||||||
|
captions = liveCaptionState.captions.slice(-initialLimit);
|
||||||
|
}
|
||||||
|
|
||||||
|
return res.json({
|
||||||
|
status: "ok",
|
||||||
|
latestSequence: liveCaptionState.latestSequence,
|
||||||
|
startedAt: new Date(liveCaptionState.startedAt).toISOString(),
|
||||||
|
availableLanguages: getAvailableLanguages(),
|
||||||
|
captions,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error getting public live captions stream", error);
|
||||||
|
return res.status(500).json({
|
||||||
|
status: "Internal server error",
|
||||||
|
latestSequence: liveCaptionState.latestSequence,
|
||||||
|
captions: [],
|
||||||
|
availableLanguages: [],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = router;
|
||||||
@@ -4,6 +4,7 @@ var router = express.Router();
|
|||||||
const DB = require("./../mongoDB.js");
|
const DB = require("./../mongoDB.js");
|
||||||
const Post = require("./../def/post.js");
|
const Post = require("./../def/post.js");
|
||||||
const Notifications = require("./../notifications.js");
|
const Notifications = require("./../notifications.js");
|
||||||
|
const { translateText, normalizeLanguageCode } = require("../utils/chatTranslation.js");
|
||||||
|
|
||||||
DB.getDB.then((DB) => {
|
DB.getDB.then((DB) => {
|
||||||
|
|
||||||
@@ -481,6 +482,47 @@ DB.getDB.then((DB) => {
|
|||||||
})
|
})
|
||||||
});
|
});
|
||||||
|
|
||||||
|
router.post("/translate", async (req, res) => {
|
||||||
|
let postid = req.body.postid;
|
||||||
|
let targetLang = normalizeLanguageCode(req.body.targetLang);
|
||||||
|
|
||||||
|
// Return ack immediately
|
||||||
|
res.json({ status: "ok", message: "Translation queued" });
|
||||||
|
|
||||||
|
if (!postid || !targetLang) return;
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Get post
|
||||||
|
const posts = await DB.getPostsByTag('', null); // No good way to get one post by ID directly exposed?
|
||||||
|
// Let's use dbCols directly if needed or find it. Wait, how do we get a single post?
|
||||||
|
// I'll assume DB.getPost exists, let me check that later. Actually I will use DB.postCols directly.
|
||||||
|
const post = await DB.postCols.findOne({ _id: DB.ObjectID(postid) });
|
||||||
|
if (!post || !post.content) return;
|
||||||
|
|
||||||
|
// Strip inline tags and bible tags before translating to reduce token usage and confusion,
|
||||||
|
// or just translate the raw content and let the AI handle it? The chat translator prompt says:
|
||||||
|
// "You translate chat messages. Keep meaning, tone, emojis, names, and references. Return only the translated text."
|
||||||
|
// So it can handle tags.
|
||||||
|
|
||||||
|
// To avoid huge translations or mostly-media posts
|
||||||
|
if (post.content.length > 1000) return;
|
||||||
|
|
||||||
|
if (post.translations && post.translations[targetLang]) return;
|
||||||
|
|
||||||
|
const translation = await translateText({
|
||||||
|
text: post.content,
|
||||||
|
sourceLang: "auto",
|
||||||
|
targetLang: targetLang
|
||||||
|
});
|
||||||
|
|
||||||
|
if (translation && translation.translatedText) {
|
||||||
|
await DB.addTranslation(postid, targetLang, translation.translatedText);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error in background post translation", error);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @swagger
|
* @swagger
|
||||||
* /post/react:
|
* /post/react:
|
||||||
|
|||||||
@@ -0,0 +1,89 @@
|
|||||||
|
#!/usr/bin/env node
|
||||||
|
|
||||||
|
require("dotenv").config();
|
||||||
|
const axios = require("axios");
|
||||||
|
|
||||||
|
const baseUrl = (process.env.CAPTION_TEST_BASE_URL || process.env.BASE_URL || "http://localhost:3000").replace(/\/+$/, "");
|
||||||
|
const ingestUrl = `${baseUrl}/live-captions/ingest`;
|
||||||
|
const intervalMs = 6000;
|
||||||
|
|
||||||
|
const samples = [
|
||||||
|
{
|
||||||
|
original: "Bienvenidos a nuestro servicio de adoracion.",
|
||||||
|
es: "Bienvenidos a nuestro servicio de adoracion.",
|
||||||
|
en: "Welcome to our worship service.",
|
||||||
|
fr: "Bienvenue a notre service de louange.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
original: "Leamos juntos en el Salmo 23.",
|
||||||
|
es: "Leamos juntos en el Salmo 23.",
|
||||||
|
en: "Let us read together in Psalm 23.",
|
||||||
|
fr: "Lisons ensemble le Psaume 23.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
original: "Dios es fiel en todo tiempo.",
|
||||||
|
es: "Dios es fiel en todo tiempo.",
|
||||||
|
en: "God is faithful at all times.",
|
||||||
|
fr: "Dieu est fidele en tout temps.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
original: "Tomemos un momento para orar.",
|
||||||
|
es: "Tomemos un momento para orar.",
|
||||||
|
en: "Let us take a moment to pray.",
|
||||||
|
fr: "Prenons un moment pour prier.",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
let sampleIndex = 0;
|
||||||
|
let timer = null;
|
||||||
|
|
||||||
|
const postPayload = async (payload) => {
|
||||||
|
const kind = payload?.draft ? "draft" : "final";
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await axios.post(ingestUrl, payload, {
|
||||||
|
headers: { "Content-Type": "application/json" },
|
||||||
|
timeout: 10000,
|
||||||
|
});
|
||||||
|
const seq = response?.data?.caption?.sequence || response?.data?.latestSequence || "?";
|
||||||
|
const text = payload?.draft || payload?.original || "";
|
||||||
|
console.log(`[live-captions:test-sender] sent ${kind} sequence=${seq} text="${text}"`);
|
||||||
|
} catch (error) {
|
||||||
|
const status = error?.response?.status;
|
||||||
|
const body = error?.response?.data;
|
||||||
|
const message = error?.message || "request failed";
|
||||||
|
console.error("[live-captions:test-sender] send failed", { status, body, message });
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const sendNextSample = async () => {
|
||||||
|
const payload = samples[sampleIndex];
|
||||||
|
|
||||||
|
const draftWords = String(payload?.original || "").split(" ").filter(Boolean);
|
||||||
|
if (draftWords.length > 2) {
|
||||||
|
await postPayload({ draft: draftWords.slice(0, 2).join(" ") });
|
||||||
|
await new Promise((resolve) => setTimeout(resolve, 550));
|
||||||
|
await postPayload({ draft: draftWords.slice(0, 4).join(" ") });
|
||||||
|
await new Promise((resolve) => setTimeout(resolve, 550));
|
||||||
|
}
|
||||||
|
|
||||||
|
await postPayload(payload);
|
||||||
|
sampleIndex = (sampleIndex + 1) % samples.length;
|
||||||
|
};
|
||||||
|
|
||||||
|
const start = async () => {
|
||||||
|
console.log(`[live-captions:test-sender] posting to ${ingestUrl} every ${intervalMs / 1000}s`);
|
||||||
|
await sendNextSample();
|
||||||
|
timer = setInterval(sendNextSample, intervalMs);
|
||||||
|
};
|
||||||
|
|
||||||
|
const shutdown = () => {
|
||||||
|
if (timer) clearInterval(timer);
|
||||||
|
console.log("[live-captions:test-sender] stopped");
|
||||||
|
process.exit(0);
|
||||||
|
};
|
||||||
|
|
||||||
|
process.on("SIGINT", shutdown);
|
||||||
|
process.on("SIGTERM", shutdown);
|
||||||
|
|
||||||
|
start();
|
||||||
@@ -54,7 +54,7 @@ const translateText = async ({ text, sourceLang, targetLang }) => {
|
|||||||
content: [
|
content: [
|
||||||
{
|
{
|
||||||
type: "input_text",
|
type: "input_text",
|
||||||
text: "You translate chat messages. Keep meaning, tone, emojis, names, and references. Return only the translated text.",
|
text: "You translate chat messages and posts. Keep meaning, tone, emojis, names, and references. Do not translate structural tags starting with @ (e.g. @image:..., @youtube:..., @bible:...). Leave them exactly as they are or omit them if they do not fit the text flow. Return only the translated text.",
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
|||||||
Reference in New Issue
Block a user