From 9cf095e766d056b04939b9b0a10ef3c0fa346465 Mon Sep 17 00:00:00 2001 From: elaineshijie Date: Tue, 12 Nov 2024 20:01:21 +0800 Subject: [PATCH 1/4] chore: refactor run service --- run-service/controller.js | 475 --------------------------- run-service/controller/controller.js | 219 ++++++++++++ run-service/{ => router}/router.js | 6 +- run-service/server.js | 26 +- run-service/{ => toRemove}/test.js | 0 run-service/{ => toRemove}/test2.js | 0 run-service/utils/index.js | 187 +++++++++++ run-service/{ => utils}/redis.js | 15 +- user-service/package-lock.json | 2 +- 9 files changed, 438 insertions(+), 492 deletions(-) delete mode 100644 run-service/controller.js create mode 100644 run-service/controller/controller.js rename run-service/{ => router}/router.js (81%) rename run-service/{ => toRemove}/test.js (100%) rename run-service/{ => toRemove}/test2.js (100%) create mode 100644 run-service/utils/index.js rename run-service/{ => utils}/redis.js (76%) diff --git a/run-service/controller.js b/run-service/controller.js deleted file mode 100644 index 22d98b2931..0000000000 --- a/run-service/controller.js +++ /dev/null @@ -1,475 +0,0 @@ -import { v4 as uuidv4 } from "uuid"; -import axios from "axios"; -import printRedisMemory from "./redis.js"; -import { questionServiceUrl, redisClient } from "./server.js"; -import UnauthorizedError from "./errors/UnauthorisedError.js"; -import ConflictError from "./errors/ConflictError.js"; -import ServiceUnavailableError from "./errors/ServiceUnavailable.js"; -import NotFoundError from "./errors/NotFoundError.js"; -import BaseError from "./errors/BaseError.js"; - -// Add function to create channelId connection when session starts -const startSession = async (req, res) => { - const { firstUserId, secondUserId } = req.body; - console.log("Starting session between", firstUserId, "and", secondUserId); - // sort both users by id - const [userA, userB] = [firstUserId, secondUserId].sort(); - try { - // Check if a session already exists by checking redis store - const sessionKey = `session:${userA}-${userB}`; - - console.log("✅✅✅✅ REDIS CONTENTS UPON REQUESTION TO START SESSION ✅✅✅✅") - printRedisMemory() - // Get session data from Redis - const channelData = await redisClient.hGetAll(sessionKey); - if (channelData && channelData.channelId) { - console.log("Session data found in Redis:", channelData.channelId); - // Remove channel data from Redis - //await redisClient.del(sessionKey); - return res.status(200).json({ - statusCode: 200, - message: `Unique channelId found for ${userA} and ${userB}`, - data: { channelId: channelData.channelId, userA, userB }, - }); - } else { - // Create a new channelId - - const channelId = uuidv4(); - console.log("Creating session in Redis:", channelId); - // Store channelId in Redis - - // create a key-channelId - await redisClient.hSet(sessionKey, { - channelId, - userA, - userB, - }); - await redisClient.hSet(`channelId:${channelId}`, { - sessionKey, - [firstUserId]: "disconnected", - [secondUserId]: "disconnected", - }); - console.log("stored session with sessionkey", sessionKey); - // NOTE: Maybe both users have to join session within x minutes, otherwise this will fail - await redisClient.expire(sessionKey, 600); - // TODO: test expiry - console.log("created session with sessionkey", sessionKey); - return res.status(200).json({ - statusCode: 200, - message: `New channelId created for ${userA} and ${userB}`, - data: { channelId, userA, userB }, - }); - } - } catch (error) { - console.error("Error creating sessionId:", error.message); - return res - .status(500) - .json({ statusCode: 500, message: "Failed to start session" }); - } -}; - -// Add function to create SSE connection with unique channelId -const subscribeToChannel = async (req, res) => { - const { channelId } = req.params; - const { userId, otherUserId } = req.query; - const [userA, userB] = [userId, otherUserId].sort(); - - const subscriber = redisClient.duplicate(); - await subscriber.connect(); - - // Set response headers for SSE - res.setHeader("Content-Type", "text/event-stream"); - res.setHeader("Cache-Control", "no-cache"); - res.setHeader("Connection", "keep-alive"); - - // TODO: handle pub sub failures? Check in store whether there is an existing result? - - // Response to indicate start (on connect) - res.write( - `data: ${JSON.stringify({ - statusCode: 201, - message: "Starting session...", - })}\n\n` - ); - - const getCurrentChannelData = await redisClient.hGetAll( - `channel:${channelId}` - ); - - await redisClient.hSet(`channel:${channelId}`, { - ...getCurrentChannelData, - [userId]: "connected", - }); - - const toLog = await redisClient.hGetAll(`channel:${channelId}`); - console.log("User connected to channel:", toLog); - // NOTE: EXECUTING RESPONSES WILL BE SENT TO BOTH USERS - subscriber.subscribe(`channel:${channelId}`, (message) => { - const update = JSON.parse(message); - - console.log("Received update from Redis:", update.statusCode); - if (update.statusCode === 200) { - // remove the channel data from Redis - // (async () => { - // await redisClient.del(`channel:${channelId}`); - // })(); - - // final one: set timeout of 8 seconds - // comment out later - // setTimeout(() => { - res.write(`data: ${JSON.stringify(update)}\n\n`); - // }, 8000); - - // res.write(`data: ${JSON.stringify(update)}\n\n`); - } else if (update.statusCode === 206) { - - // temporarily: set a random timeout from 1 to 5 seconds - // comment out later - const timeout = Math.floor(Math.random() * 5000) + 1000; - // setTimeout(() => { - res.write(`data: ${JSON.stringify(update)}\n\n`); - // }, timeout); - - // res.write(`data: ${JSON.stringify(update)}\n\n`); - - } else { - res.write(`data: ${JSON.stringify(update)}\n\n`); - } - }); - - req.on("close", async () => { - console.log("Receiving client's closing connection response"); - // delete channel data from Redis - const channelData = await redisClient.hGetAll(`channel:${channelId}`); - if (channelData && channelData[otherUserId] === "disconnected") { - await redisClient.del(`channel:${channelId}`); - await redisClient.del(`session:${userA}-${userB}`); - } else { - await redisClient.hSet(`channel:${channelId}`, { - ...channelData, - [userId]: "disconnected", - }); - } - await subscriber.unsubscribe(); - await subscriber.disconnect(); - res.end(); - }); -}; - -const executeTest = async (req, res) => { - try { - console.log("executing test started"); - const questionId = req.params.questionId; - const { codeAttempt, channelId, firstUserId, secondUserId } = req.body; - - // Check if another execution is in progress - const initialChannelData = await redisClient.hGetAll( - `channel:${channelId}` - ); - console.log("Existing job data:", initialChannelData); - if (initialChannelData && initialChannelData.status === "processing") { - console.log( - "Another execution is already in progress. Try again later error thrown" - ); - throw new ConflictError( - "Another execution is already in progress. Try again later." - ); - return; - } - - // If another client is not subscribed to the channel, throw an error - // if (initialChannelData) { - // if ( - // initialChannelData[firstUserId] !== "connected" || - // initialChannelData[secondUserId] !== "connected" - // ) { - // console.log( - // "Another user is not connected to the channel. Try again later error thrown" - // ); - // throw new ConflictError( - // "Another user is not connected to the channel. Try again later." - // ); - // return; - // } - // } - - // Retrieve testcases for the question - console.log("Executing test cases for questionId:", questionId); - const testcases = await getTestcases(questionId); - if (!testcases) { - throw new NotFoundError("Testcases not found for the question"); - } - - for (let i = 0; i < testcases.length; i++) { - console.log("Testcase:", testcases[i]._id); - console.log("isPublic:", testcases[i].isPublic); - } - console.log("Testcases retrieved sucessfully"); - console.log({ channelId }) - // Indicate that the test cases are being processed - initial message to indicate start of execution - const channelData = await redisClient.hGetAll(`channel:${channelId}`); - if (channelData && channelData.status !== "processing") { - await redisClient.hSet(`channel:${channelId}`, { - status: "processing", - questionId, - codeAttempt, - data: JSON.stringify([]), - }); - } - const testCaseCount = testcases.length; - - const toLog = await redisClient.hGetAll(`channel:${channelId}`); - console.log("Channel data after setting status:", toLog); - - // Start processing test cases - processTestcases(channelId, testcases, codeAttempt, questionId); - - // Respond to client with test case count - res.status(200).json({ - statusCode: 200, - message: `Executing test cases for questionId: ${questionId}`, - data: { testCaseCount: testCaseCount } - }); - } catch (error) { - console.log("ERROR: ", error) - if (error instanceof BaseError) { - console.log("Error while executing testcase:", error.message); - return res - .status(error.statusCode) - .json({ statusCode: error.statusCode, message: error.message }); - } - res.status(500).json({ error: "Failed to execute test cases" }); - } -}; - -// TODO: Refactor code below -// Function to execute a single test case -async function runTestcase(testcase, code) { - try { - // Append test code to main code - const normalizedCode = code.trim().replace(/\r?\n/g, "\n"); - const testCode = testcase.testCode.trim().replace(/\r?\n/g, "\n"); - - // Combine the normalized code and test code - const sourceCode = `${normalizedCode}\n\n${testCode}`; - const requestBody = { - source_code: sourceCode, - language_id: 71, - expected_output: testcase.expectedOutput, - }; - - const headers = { - "X-Auth-Token": process.env.X_AUTH_TOKEN, - "X-Auth-User": process.env.X_AUTH_USER, - }; - - // Send POST request to the judge0 API - const response = await axios.post(process.env.JUDGE0_API_URL, requestBody, { - headers, - }); - - //console.log("Response from API:", response.data); - const responseData = response.data; - - let outputFinal = responseData.stdout ? responseData.stdout : ""; - let testCaseDetailsFinal = { - input: responseData.input || "No input description", - expectedOutput: testcase.expectedOutput || "No expected output", - testCaseId: testcase._id, - }; - - // Remove output and question details if the testcase is not public - if (!testcase.isPublic) { - console.log("Removing output and question details for private testcase"); - outputFinal = null; - testCaseDetailsFinal = { - testCaseId: testcase._id, - input: "Hidden", - expectedOutput: "Hidden", - }; - } - - const testCaseResult = { - stderr: responseData.stderr, - isPassed: responseData.status.id === 3 ? true : false, - stdout: outputFinal, - testCaseDetails: testCaseDetailsFinal, - memory: responseData.memory || 0, - time: responseData.time || "0", - }; - return testCaseResult; - } catch (error) { - console.error("Error executing test case:", error.message); - - if (error.response) { - switch (error.response.status) { - case 404: - // Service not found error -> means not available? - throw new ServiceUnavailableError( - "Execution Service not found. Testcase execution failed." - ); - case 401: - throw new UnauthorizedError( - "Authentication failed. Testcase failed to execute." - ); - case 503: - throw new ServiceUnavailableError( - 503, - "Queue is full. Testcase execution failed." - ); - default: - throw new BaseError( - error.response.status, - "Error executing test case." - ); - } - } else { - throw new BaseError(500, "Error executing test case."); - } - } -} - -async function processTestcases(channelId, testcases, code, questionId) { - // const results = []; - let hasError = false; - console.log("============Starting testcases execution=========="); - await redisClient.publish( - `channel:${channelId}`, - JSON.stringify({ statusCode: 202, message: `Executing test cases of question ${questionId}` }) - ); - - - const results = []; - - - // Map each test case to a promise - const promises = testcases.map(async (testcase) => { - try { - const result = await runTestcase(testcase, code); - results.push(result); - - console.log("result", result); - - // Publish only the latest test case result - await redisClient.publish( - `channel:${channelId}`, - JSON.stringify({ - statusCode: 206, - message: `Testcase ${testcase._id} executed successfully`, - data: { result }, - }) - ); - - return result; // Return the result for Promise.all - } catch (error) { - console.log("Error while executing testcase:", testcase._id); - - const errorMessage = { - testcaseId: testcase._id, - message: error.message, - }; - results.push(errorMessage); - hasError = true; - - console.log("==========Error while executing testcase:", errorMessage.testcaseId); - console.log("==========Error Message:", errorMessage.message); - - // Publish only the latest error result - await redisClient.publish( - `channel:${channelId}`, - JSON.stringify({ statusCode: 500, message: errorMessage }) - ); - - throw errorMessage; // Throw to handle failure in Promise.all - } - }); - - // for (let i = 0; i < testcases.length; i++) { - // try { - // const result = await runTestcase(testcases[i], code); - // results.push(result); - - // console.log("result", result); - // // Publish only the latest test case result - // await redisClient.publish( - // `channel:${channelId}`, - // JSON.stringify({ - // statusCode: 206, - // message: `Testcase ${testcases[i]._id} executed successfully`, - // data: { result }, - // }) - // ); - // } catch (error) { - // console.log("Error while executing testcase:", testcases[i]._id); - // const errorMessage = { - // testcaseId: testcases[i]._id, - // message: error.message, - // }; - // results.push(errorMessage); - // hasError = true; // Set error flag to true - - // console.log( - // "==========Error while executing testcase:", - // errorMessage.testcaseId - // ); - // console.log("==========Error Message:", errorMessage.error); - // // Publish only the latest error result - // await redisClient.publish( - // `channel:${channelId}`, - // JSON.stringify({ statusCode: 500, message: errorMessage }) - // ); - // break; - // } - // } - - try { - await Promise.all(promises); - - console.log("Completed testcases execution for questionId:", questionId); - - await redisClient.hSet(`channel:${channelId}`, { - status: hasError ? "error" : "completed", - data: JSON.stringify(results), - questionId: questionId, - codeAttempt: code, - }); - - - // Publish the "complete" status only if no errors occurred - if (!hasError) { - await redisClient.publish( - `channel:${channelId}`, - JSON.stringify({ - statusCode: 200, - data: { results: results, questionId, code }, - }) - ); - } - } catch (error) { - console.log("Some testcases failed:", error); - - hasError = true; - - - } - - - - printRedisMemory(); -} - -// TODO: Refactor this function -async function getTestcases(questionId) { - try { - const response = await axios.get(`${questionServiceUrl}/${questionId}`); - const testcases = response.data.data.testCase; - return testcases; - } catch (error) { - console.error("Error retrieving testcases:", error.message); - throw new ServiceUnavailableError( - "Connection to question service failed. Testcase execution failed." - ); - } -} - -export { executeTest, subscribeToChannel, startSession }; diff --git a/run-service/controller/controller.js b/run-service/controller/controller.js new file mode 100644 index 0000000000..20577246f3 --- /dev/null +++ b/run-service/controller/controller.js @@ -0,0 +1,219 @@ +import { v4 as uuidv4 } from "uuid"; + +import { printRedisMemory, setChannelData } from "../utils/redis.js"; +import { questionServiceUrl, redisClient } from "../server.js"; +import UnauthorizedError from "../errors/UnauthorisedError.js"; +import ConflictError from "../errors/ConflictError.js"; +import ServiceUnavailableError from "../errors/ServiceUnavailable.js"; +import NotFoundError from "../errors/NotFoundError.js"; +import BaseError from "../errors/BaseError.js"; +import { getTestcases, processTestcases } from "../utils/index.js"; + +// Add function to create channelId connection when session starts +const startSession = async (req, res) => { + const { firstUserId, secondUserId } = req.body; + // sort both users by id + const [userA, userB] = [firstUserId, secondUserId].sort(); + try { + // Check if a session already exists by checking redis store + const sessionKey = `session:${userA}-${userB}`; + + console.log( + "✅✅✅✅ REDIS CONTENTS UPON REQUESTION TO START SESSION ✅✅✅✅" + ); + printRedisMemory(); + // Get session data from Redis + const channelData = await redisClient.hGetAll(sessionKey); + if (channelData && channelData.channelId) { + console.log( + "============== Session data found in Redis: ==============", + channelData.channelId + ); + // Remove channel data from Redis + return res.status(200).json({ + statusCode: 200, + message: `Unique channelId found for ${userA} and ${userB}`, + data: { channelId: channelData.channelId, userA, userB }, + }); + } else { + // Create a new channelId + + const channelId = uuidv4(); + console.log("Creating session in Redis:", channelId); + const sessionData = { channelId, userA, userB }; + setChannelData(sessionKey, sessionData); + // NOTE: Maybe both users have to join session within x minutes, otherwise this will fail + await redisClient.expire(sessionKey, 600); + console.log( + "============ Created session with sessionkey ============", + sessionKey + ); + return res.status(200).json({ + statusCode: 200, + message: `New channelId created for ${userA} and ${userB}`, + data: { channelId, userA, userB }, + }); + } + } catch (error) { + console.error("Error creating sessionId:", error.message); + return res + .status(500) + .json({ statusCode: 500, message: "Failed to start session" }); + } +}; + +// Add function to create SSE connection with unique channelId +const subscribeToChannel = async (req, res) => { + const { channelId } = req.params; + const { userId, otherUserId } = req.query; + const [userA, userB] = [userId, otherUserId].sort(); + + const subscriber = redisClient.duplicate(); + await subscriber.connect(); + + // Set response headers for SSE + res.setHeader("Content-Type", "text/event-stream"); + res.setHeader("Cache-Control", "no-cache"); + res.setHeader("Connection", "keep-alive"); + + // Response to indicate start (on connect) + res.write( + `data: ${JSON.stringify({ + statusCode: 201, + message: "Starting session...", + })}\n\n` + ); + + const channelKey = `channel:${channelId}`; + + const setUserConnectedData = { [userId]: "connected" }; + setChannelData(channelKey, setUserConnectedData); + + subscriber.subscribe(`channel:${channelId}`, (message) => { + const update = JSON.parse(message); + + if (update.statusCode === 200) { + // final one: set timeout of 8 seconds + // comment out later + // setTimeout(() => { + res.write(`data: ${JSON.stringify(update)}\n\n`); + // }, 8000); + } else if (update.statusCode === 206) { + res.write(`data: ${JSON.stringify(update)}\n\n`); + } else { + res.write(`data: ${JSON.stringify(update)}\n\n`); + } + }); + + req.on("close", async () => { + console.log("Receiving client's closing connection response"); + // delete channel data from Redis + const channelData = await redisClient.hGetAll(`channel:${channelId}`); + if (channelData && channelData[otherUserId] === "disconnected") { + await redisClient.del(`channel:${channelId}`); + await redisClient.del(`session:${userA}-${userB}`); + } else { + const setUserDisconnectedData = { [userId]: "disconnected" }; + setChannelData(channelKey, setUserDisconnectedData); + } + await subscriber.unsubscribe(); + await subscriber.disconnect(); + printRedisMemory(); + res.end(); + }); +}; + +const executeTest = async (req, res) => { + try { + printRedisMemory(); + console.log("executing test started"); + const questionId = req.params.questionId; + const { codeAttempt, channelId, firstUserId, secondUserId } = req.body; + + const channelKey = `channel:${channelId}`; + // Check if another execution is in progress + const initialChannelData = await redisClient.hGetAll( + `channel:${channelId}` + ); + console.log("Existing job data:", initialChannelData); + if (initialChannelData && initialChannelData.status === "processing") { + console.log( + "Another execution is already in progress. Try again later error thrown" + ); + throw new ConflictError( + "Another execution is already in progress. Try again later." + ); + return; + } + + // The code below no longer cause issues for the current implementation + // but if test cases are executed as long as one user choose to leave room, + // without execute test cases, running test cases from frontend to save + // attempt may cause issues + // Delete when necessary + // If another client is not subscribed to the channel, throw an error + if (initialChannelData) { + if ( + initialChannelData[firstUserId] !== "connected" || + initialChannelData[secondUserId] !== "connected" + ) { + console.log( + "Another user is not connected to the channel. Try again later error thrown" + ); + throw new ConflictError( + "Another user is not connected to the channel. Try again later." + ); + return; + } + } + + // Retrieve testcases for the question + console.log("Executing test cases for questionId:", questionId); + const testcases = await getTestcases(questionId); + if (!testcases) { + throw new NotFoundError("Testcases not found for the question"); + } + + for (let i = 0; i < testcases.length; i++) { + console.log("Testcase:", testcases[i]._id); + console.log("isPublic:", testcases[i].isPublic); + } + console.log("Testcases retrieved sucessfully"); + console.log({ channelId }); + // Indicate that the test cases are being processed - initial message to indicate start of execution + const channelData = await redisClient.hGetAll(`channel:${channelId}`); + if (channelData && channelData.status !== "processing") { + const setStatusData = { + status: "processing", + questionId, + codeAttempt, + data: JSON.stringify([]), + }; + setChannelData(channelKey, setStatusData); + } + const testCaseCount = testcases.length; + + const toLog = await redisClient.hGetAll(`channel:${channelId}`); + console.log("Channel data after setting status:", toLog); + + // Start processing test cases + processTestcases(channelId, testcases, codeAttempt, questionId); + + // Respond to client with test case count + res.status(200).json({ + statusCode: 200, + message: `Executing test cases for questionId: ${questionId}`, + data: { testCaseCount: testCaseCount }, + }); + } catch (error) { + console.log("ERROR: ", error); + if (error instanceof BaseError) { + return res + .status(error.statusCode) + .json({ statusCode: error.statusCode, message: error.message }); + } + res.status(500).json({ error: "Failed to execute test cases" }); + } +}; + +export { executeTest, subscribeToChannel, startSession }; diff --git a/run-service/router.js b/run-service/router/router.js similarity index 81% rename from run-service/router.js rename to run-service/router/router.js index cf2ac52ac2..9ebab34608 100644 --- a/run-service/router.js +++ b/run-service/router/router.js @@ -1,5 +1,9 @@ import express from "express"; -import { executeTest, startSession, subscribeToChannel } from "./controller.js"; +import { + executeTest, + startSession, + subscribeToChannel, +} from "../controller/controller.js"; const router = express.Router(); // POST route to initiate a session diff --git a/run-service/server.js b/run-service/server.js index 83859e849e..65c8475d70 100644 --- a/run-service/server.js +++ b/run-service/server.js @@ -1,9 +1,9 @@ -import express from 'express'; -import { createClient } from 'redis'; -import dotenv from 'dotenv'; -import router from './router.js'; -import corsMiddleware from './middlewares/cors.js'; -import loggingMiddleware from './middlewares/logging.js'; +import express from "express"; +import { createClient } from "redis"; +import dotenv from "dotenv"; +import router from "./router/router.js"; +import corsMiddleware from "./middlewares/cors.js"; +import loggingMiddleware from "./middlewares/logging.js"; import errorHandler from "./middlewares/errorHandler.js"; const app = express(); @@ -15,23 +15,23 @@ const questionServiceUrl = process.env.QUESTION_SVC_DEV; // TODO: Replace with actual Redis connection instance const redisClient = createClient({ - url: process.env.REDIS_URL_DEV ?? 'redis://localhost:6379', + url: process.env.REDIS_URL_DEV ?? "redis://localhost:6379", }); -redisClient.on('error', (err) => console.log('Redis Client Error', err)); -redisClient.on('connect', () => console.log('Connected to Redis')); -redisClient.on('ready', () => console.log('Redis Client Ready')); +redisClient.on("error", (err) => console.log("Redis Client Error", err)); +redisClient.on("connect", () => console.log("Connected to Redis")); +redisClient.on("ready", () => console.log("Redis Client Ready")); await redisClient.connect(); app.use(loggingMiddleware); app.use(corsMiddleware); -app.use('/api/run-service', router); +app.use("/api/run-service", router); // Test Route for Health Checks -app.get('/healthz', (req, res) => { +app.get("/healthz", (req, res) => { res .status(200) - .json({ message: 'Connected to /healthz route of run-service' }); + .json({ message: "Connected to /healthz route of run-service" }); }); app.use(errorHandler); diff --git a/run-service/test.js b/run-service/toRemove/test.js similarity index 100% rename from run-service/test.js rename to run-service/toRemove/test.js diff --git a/run-service/test2.js b/run-service/toRemove/test2.js similarity index 100% rename from run-service/test2.js rename to run-service/toRemove/test2.js diff --git a/run-service/utils/index.js b/run-service/utils/index.js new file mode 100644 index 0000000000..9bb6a8fad0 --- /dev/null +++ b/run-service/utils/index.js @@ -0,0 +1,187 @@ +import axios from "axios"; +import ServiceUnavailableError from "../errors/ServiceUnavailable.js"; +import { questionServiceUrl, redisClient } from "../server.js"; +import {printRedisMemory} from "./redis.js"; + +async function getTestcases(questionId) { + try { + const response = await axios.get(`${questionServiceUrl}/${questionId}`); + const testcases = response.data.data.testCase; + return testcases; + } catch (error) { + console.error("Error retrieving testcases:", error.message); + throw new ServiceUnavailableError( + "Connection to question service failed. Testcase execution failed." + ); + } +} +// Function to execute a single test case +async function runTestcase(testcase, code) { + try { + // Append test code to main code + const normalizedCode = code.trim().replace(/\r?\n/g, "\n"); + const testCode = testcase.testCode.trim().replace(/\r?\n/g, "\n"); + + // Combine the normalized code and test code + const sourceCode = `${normalizedCode}\n\n${testCode}`; + const requestBody = { + source_code: sourceCode, + language_id: 71, + expected_output: testcase.expectedOutput, + }; + + const headers = { + "X-Auth-Token": process.env.X_AUTH_TOKEN, + "X-Auth-User": process.env.X_AUTH_USER, + }; + + // Send POST request to the judge0 API + const response = await axios.post(process.env.JUDGE0_API_URL, requestBody, { + headers, + }); + + const responseData = response.data; + + let outputFinal = responseData.stdout ? responseData.stdout : ""; + let testCaseDetailsFinal = { + input: responseData.input || "No input description", + expectedOutput: testcase.expectedOutput || "No expected output", + testCaseId: testcase._id, + }; + + // Remove output and question details if the testcase is not public + if (!testcase.isPublic) { + console.log("Removing output and question details for private testcase"); + outputFinal = null; + testCaseDetailsFinal = { + testCaseId: testcase._id, + input: "Hidden", + expectedOutput: "Hidden", + }; + } + + const testCaseResult = { + stderr: responseData.stderr, + isPassed: responseData.status.id === 3 ? true : false, + stdout: outputFinal, + testCaseDetails: testCaseDetailsFinal, + memory: responseData.memory || 0, + time: responseData.time || "0", + }; + return testCaseResult; + } catch (error) { + console.error("Error executing test case:", error.message); + + if (error.response) { + switch (error.response.status) { + case 404: + // Service not found error -> means not available? + throw new ServiceUnavailableError( + "Execution Service not found. Testcase execution failed." + ); + case 401: + throw new UnauthorizedError( + "Authentication failed. Testcase failed to execute." + ); + case 503: + throw new ServiceUnavailableError( + 503, + "Queue is full. Testcase execution failed." + ); + default: + throw new BaseError( + error.response.status, + "Error executing test case." + ); + } + } else { + throw new BaseError(500, "Error executing test case."); + } + } +} + +async function processTestcases(channelId, testcases, code, questionId) { + // const results = []; + let hasError = false; + console.log("============Starting testcases execution=========="); + await redisClient.publish( + `channel:${channelId}`, + JSON.stringify({ + statusCode: 202, + message: `Executing test cases of question ${questionId}`, + }) + ); + + const results = []; + + // Map each test case to a promise + const promises = testcases.map(async (testcase) => { + try { + const result = await runTestcase(testcase, code); + results.push(result); + + console.log("result", result); + + // Publish only the latest test case result + await redisClient.publish( + `channel:${channelId}`, + JSON.stringify({ + statusCode: 206, + message: `Testcase ${testcase._id} executed successfully`, + data: { result }, + }) + ); + + return result; // Return the result for Promise.all + } catch (error) { + console.log("Error while executing testcase:", testcase._id); + + const errorMessage = { + testcaseId: testcase._id, + message: error.message, + }; + results.push(errorMessage); + hasError = true; + + // Publish only the latest error result + await redisClient.publish( + `channel:${channelId}`, + JSON.stringify({ statusCode: 500, message: errorMessage }) + ); + + throw errorMessage; // Throw to handle failure in Promise.all + } + }); + + try { + await Promise.all(promises); + + console.log("Completed testcases execution for questionId:", questionId); + + await redisClient.hSet(`channel:${channelId}`, { + status: hasError ? "error" : "completed", + data: JSON.stringify(results), + questionId: questionId, + codeAttempt: code, + }); + + // Publish the "complete" status only if no errors occurred + if (!hasError) { + await redisClient.publish( + `channel:${channelId}`, + JSON.stringify({ + statusCode: 200, + data: { results: results, questionId, code }, + }) + ); + } + } catch (error) { + console.log("Some testcases failed:", error); + + hasError = true; + } + + printRedisMemory(); + } + +export { getTestcases, processTestcases }; diff --git a/run-service/redis.js b/run-service/utils/redis.js similarity index 76% rename from run-service/redis.js rename to run-service/utils/redis.js index 4cb95b02f3..7ac602eb2a 100644 --- a/run-service/redis.js +++ b/run-service/utils/redis.js @@ -1,4 +1,4 @@ -import { redisClient } from "./server.js"; +import { redisClient } from "../server.js"; // Function to print Redis memory contents async function printRedisMemory() { @@ -37,4 +37,15 @@ async function printRedisMemory() { console.error("Error printing Redis memory:", error); } } -export default printRedisMemory; \ No newline at end of file + +// function to set channel data in redis +const setChannelData = async (key, data) => { + const channelData = await redisClient.hGetAll(`${key}`); + if (channelData) { + await redisClient.hSet(`${key}`, { + ...channelData, + ...data, + }); + } +}; +export { printRedisMemory, setChannelData }; diff --git a/user-service/package-lock.json b/user-service/package-lock.json index 9704e55f7b..e3925aab9c 100644 --- a/user-service/package-lock.json +++ b/user-service/package-lock.json @@ -3092,4 +3092,4 @@ } } } -} \ No newline at end of file +} From 03ef13bfc1425a6465a1270128b9adbabf6b73ac Mon Sep 17 00:00:00 2001 From: elaineshijie Date: Tue, 12 Nov 2024 23:39:01 +0800 Subject: [PATCH 2/4] chore: clean up code --- run-service/controller/controller.js | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/run-service/controller/controller.js b/run-service/controller/controller.js index 20577246f3..bd08774c3f 100644 --- a/run-service/controller/controller.js +++ b/run-service/controller/controller.js @@ -1,10 +1,8 @@ import { v4 as uuidv4 } from "uuid"; import { printRedisMemory, setChannelData } from "../utils/redis.js"; -import { questionServiceUrl, redisClient } from "../server.js"; -import UnauthorizedError from "../errors/UnauthorisedError.js"; +import { redisClient } from "../server.js"; import ConflictError from "../errors/ConflictError.js"; -import ServiceUnavailableError from "../errors/ServiceUnavailable.js"; import NotFoundError from "../errors/NotFoundError.js"; import BaseError from "../errors/BaseError.js"; import { getTestcases, processTestcases } from "../utils/index.js"; From 4343a7f8503b38aaa1f5f37e25fe08e93d54880f Mon Sep 17 00:00:00 2001 From: elaineshijie Date: Tue, 12 Nov 2024 23:43:14 +0800 Subject: [PATCH 3/4] fix: fix comments --- run-service/controller/controller.js | 7 +++---- 1 file changed, 3 insertions(+), 4 deletions(-) diff --git a/run-service/controller/controller.js b/run-service/controller/controller.js index bd08774c3f..0749ef7cff 100644 --- a/run-service/controller/controller.js +++ b/run-service/controller/controller.js @@ -145,10 +145,9 @@ const executeTest = async (req, res) => { } // The code below no longer cause issues for the current implementation - // but if test cases are executed as long as one user choose to leave room, - // without execute test cases, running test cases from frontend to save - // attempt may cause issues - // Delete when necessary + // but when as long as one user choose to leave room, without executing test cases, + // running test cases from frontend to save attempt with test cases may cause issues + // Delete the code below when necessary // If another client is not subscribed to the channel, throw an error if (initialChannelData) { if ( From 662afd63e61b454944324730bf4f97c687c93920 Mon Sep 17 00:00:00 2001 From: elaineshijie Date: Wed, 13 Nov 2024 13:11:38 +0800 Subject: [PATCH 4/4] chore: remove checks for disconnect --- run-service/controller/controller.js | 22 +++------------------- 1 file changed, 3 insertions(+), 19 deletions(-) diff --git a/run-service/controller/controller.js b/run-service/controller/controller.js index 0749ef7cff..fb39a19008 100644 --- a/run-service/controller/controller.js +++ b/run-service/controller/controller.js @@ -86,6 +86,9 @@ const subscribeToChannel = async (req, res) => { const setUserConnectedData = { [userId]: "connected" }; setChannelData(channelKey, setUserConnectedData); + console.log("User connected to channel:", userId); + const toLog = await redisClient.hGetAll(`channel:${channelId}`); + console.log(`Channel data after setting connected for user:${userId}`, toLog); subscriber.subscribe(`channel:${channelId}`, (message) => { const update = JSON.parse(message); @@ -144,25 +147,6 @@ const executeTest = async (req, res) => { return; } - // The code below no longer cause issues for the current implementation - // but when as long as one user choose to leave room, without executing test cases, - // running test cases from frontend to save attempt with test cases may cause issues - // Delete the code below when necessary - // If another client is not subscribed to the channel, throw an error - if (initialChannelData) { - if ( - initialChannelData[firstUserId] !== "connected" || - initialChannelData[secondUserId] !== "connected" - ) { - console.log( - "Another user is not connected to the channel. Try again later error thrown" - ); - throw new ConflictError( - "Another user is not connected to the channel. Try again later." - ); - return; - } - } // Retrieve testcases for the question console.log("Executing test cases for questionId:", questionId);