diff --git a/.github/workflows/CI_Tests_and_Report.yaml b/.github/workflows/CI_Tests_and_Report.yaml index a295a3d03..87a7563ac 100644 --- a/.github/workflows/CI_Tests_and_Report.yaml +++ b/.github/workflows/CI_Tests_and_Report.yaml @@ -133,14 +133,14 @@ jobs: - name: Send sanity POST request and store response env: # <-- api url for sanity test -->/<---- repository_id --->/<------ group_id ------> - API_URL: localhost:8080/api/sanity/test/632893aad3bd45536c41b684/6400a70f5eda22409c4d2ed9 + API_URL: localhost:8080/api/sanity/test/${{ secrets.SANITY_REPO_ID }}/${{ secrets.SANITY_GROUP_ID }} run: | curl -X POST -H 'Content-Type: application/json' -d '{"email": "${{ secrets.SEED_EMAIL }}", "password": "${{ secrets.SEED_PW }}", "stayLoggedIn": true, "repository": "Seed-Test", "source": "db"}' "$API_URL" > sanityReport.txt - name: print sanity run: | echo $(cat sanityReport.txt) - + - name: Get passed and total sanity run: | passed=$(awk '/Steps:/ { match($0, /[0-9]+ passed/); print substr($0, RSTART, RLENGTH-7) }' sanityReport.txt) @@ -193,4 +193,4 @@ jobs: sanityStepsFailed: ${{ steps.passed-total-sanity.outputs.failedSteps }} sanityStepsSkipped: ${{ steps.passed-total-sanity.outputs.skippedSteps }} workflowLink: "${{ github.server_url }}/${{ github.repository }}/actions/runs/${{ github.run_id }}" - webhook: ${{ secrets.MS_TEAMS_WEBHOOK_URI }} \ No newline at end of file + webhook: ${{ secrets.MS_TEAMS_WEBHOOK_URI }} diff --git a/backend/features/step_definitions/stepdefs.js b/backend/features/step_definitions/stepdefs.js index 1237f07ca..1d265b2f2 100644 --- a/backend/features/step_definitions/stepdefs.js +++ b/backend/features/step_definitions/stepdefs.js @@ -11,7 +11,7 @@ require('geckodriver'); const firefox = require('../../node_modules/selenium-webdriver/firefox'); const chrome = require('../../node_modules/selenium-webdriver/chrome'); const edge = require('../../node_modules/selenium-webdriver/edge'); -const moment = require('../../node_modules/moment'); +const { applySpecialCommands } = require('../../src/serverHelper'); let driver; const firefoxOptions = new firefox.Options(); @@ -77,7 +77,7 @@ Before(async function () { edgeOptions.setMobileEmulation({ deviceName: currentParameters.emulator }); break; case 'firefox': - // no way to do it ? + // no way to do it ? } if (currentParameters.oneDriver) { @@ -268,10 +268,10 @@ When('The site should wait for {string} milliseconds', async function (ms) { When('I insert {string} into the field {string}', async function fillTextField(value, label) { const world = this; const identifiers = [`//input[@id='${label}']`, `//input[contains(@id,'${label}')]`, `//textarea[@id='${label}']`, `//textarea[contains(@id,'${label}')]`, - `//textarea[@*='${label}']`, `//textarea[contains(@*='${label}')]`, `//*[@id='${label}']`, `//input[@type='text' and @*='${label}']`, - `//label[contains(text(),'${label}')]/following::input[@type='text']`, `${label}`]; + `//textarea[@*='${label}']`, `//textarea[contains(@*='${label}')]`, `//*[@id='${label}']`, `//input[@type='text' and @*='${label}']`, + `//label[contains(text(),'${label}')]/following::input[@type='text']`, `${label}`]; - if (value.includes('@@')) value = calcDate(value); + if (value.includes('@@')) value = applyDateCommand(value); const promises = []; for (const idString of identifiers) promises.push( @@ -293,154 +293,6 @@ When('I insert {string} into the field {string}', async function fillTextField(v await driver.sleep(100 + currentParameters.waitTime); }); -function calcDate(value) { - // Regex that matches the start: e.g @@Date, @@Day @@Month, @@Day,23 - // works only with PCRE2. JS uses EMCAScript - // const start_regex = /^((@@Date)|((@@Day,\d{1,2}|@@Day)|(@@Month,\d{1,2}|@@Month)|(@@Year,\d{4}|@@Year))(?!\1)(((@@Month,\d{1,2}|@@Month)|(@@Year,\d{4}|@@Year)|(@@Day,\d{1,2}|@@Day))(?!\2))?(((@@Year,\d{4}|@@Year)|(@@Day,\d{1,2}|@@Day)|(@@Month,\d{1,2}|@@Month))(?!\3))?)|(^\s*$)/ - - // Regex that matches the middle: e.g. +@@Day,2-@@Month,4 .... - const mid_regex = /(^((\+|\-)@@(\d+),(Day|Month|Year))*)|(^\s*$)/; - // Regex that matches the format end: e.g @@format:DDMMYY€€ - const end_regex = /(^(@@format:\w*€€)*)|(^\s*$)/; - - function getStart(str) { - let endIndex = str.length; - const symbols = ['+', '-', '@@format']; - - symbols.forEach((symbol) => { - const symbolIndex = str.indexOf(symbol); - if (symbolIndex !== -1 && symbolIndex < endIndex) endIndex = symbolIndex; - }); - return str.substring(0, endIndex); - } - - function getMid(str) { - let endIndex = str.length; - const symbols = ['@@format']; - - symbols.forEach((symbol) => { - const symbolIndex = str.indexOf(symbol); - if (symbolIndex !== -1 && symbolIndex < endIndex) endIndex = symbolIndex; - }); - return str.substring(0, endIndex); - } - const start = getStart(value).replace(' ', ''); - const mid = getMid(value.replace(start, '')).replace(' ', ''); - const end = mid.replace(mid, '').trim(); - - // check if the start part is written correctly - const dates = start.split(/@@Date/); - const substrings = [/@@Day,\d{1,2}|@@Day/, /@@Month,\d{1,2}|@@Month/, /@@Year,\d{4}|@@Year/]; - const substringsErr = ['@@Day', '@@Month', '@@Year']; - // check if @@Date has been used - if (dates.length > 1) if (dates.length - 1 > 1) throw Error('@@Date should only be used once.'); - else for (let i = 0; i < substrings.length; i++) { - if (substrings[i].test(start)) throw Error(`@@Date should only be used by itself. Found: ${substringsErr[i]}`); - } - - // check the correct usage of @@Day, @@Month, @@Year - else { - startcopy = start.slice(); - for (let i = 0; i < substrings.length; i++) { - if (start.split(substrings[i]).length - 1 > 1) throw Error(`${substringsErr[i]} may only be used 0 or 1 time. Input: ${start}`); - startcopy = startcopy.replace(substrings[i], ''); - } - // if (startcopy.length !== 0) throw Error(`Unkown tokens in the start section: ${startcopy}`); - } - - // check if the calculation part is written correctly - if (!mid_regex.test(mid)) throw Error('Error parsing the calculation section. Example: +@@23,Day-@@Month,1'); - - // check if the format part is written correctly - if (!end_regex.test(end)) throw Error('Error parsing the format section. Example: @@format:XXXXXX€€. Where XXXXX is the Format String. Example: @@format:DD-MM-YY'); - - // Get the format e.g @@format:XXXXX€€ - let format = value.match(/(@@format:.*€€)/g); - - // Start Date - const currDate = new Date(); - let day = value.match(/(@@Day,\d{1,2})/g); - if (day) day = parseInt(day[0].match(/@@Day,(\d+)/)[1]); - let month = value.match(/(@@Month,\d{1,2})/g); - if (month) month = parseInt(month[0].match(/@@Month,(\d+)/)[1] - 1); - let year = value.match(/(@@Year,\d\d\d\d)/g); - if (year) year = parseInt(year[0].match(/@@Year,(\d+)/)[1]); - - currDate.setFullYear( - year == null ? currDate.getFullYear() : year, - month == null ? currDate.getMonth() : month, - day == null ? currDate.getDate() : day - ); - - // If no format was found, check the given format e.g. @@Date, @@Day@@Month, @@Day ... - if (format == null) { - // Get the Substring until the first add,sub or format e.g @@Day@@Month+@@ ... -> @@Day@@Month - format = value.split(/[\+\-]/)[0]; - // Replace the @@Day, @@Month, @@Year - format = format.replace(/@@Day(,(\d\d){1,2}){0,1}/, 'DD.').replace(/@@Month(,(\d\d){1,2}){0,1}/, 'MM.') - .replace(/@@Year(,(\d\d\d\d)){0,1}/, 'YYYY.') - .replace('@@Date', 'DD.MM.YYYY.') - .slice(0, -1); - } else - // Get @@format: tag and €€ at the end - format = format[0].slice(9, -2); - - // console.log(`Day: ${day}\nMonth: ${month}\nYear: ${year}\nFormat: ${format}\nDate: ${currDate.toDateString()}`); - - // Get all adds e.g +@@2,Month - let adds = value.match(/\+@@(\d+),(\w+)/g); - // Read values e.g. of +@@5,Day -> {number: 5, kind: "Day"}; or set to empty array if null (no match) - adds = adds ? adds.map((element) => { - const match = element.match(/\+@@(\d+),(\w+)/); - return { number: parseInt(match[1]), kind: match[2] }; - }) : []; - // Get all subs e.g -@@10,Year - let subs = value.match(/\-@@(\d+),(\w+)/g); - // Read values e.g. of -@@2,Month -> {number: 2, kind: "Month"}; or set to empty array if null (no match) - subs = subs ? subs.map((element) => { - const match = element.match(/\-@@(\d+),(\w+)/); - return { number: parseInt(match[1]), kind: match[2] }; - }) : []; - - // Add every add in the adds array - adds.forEach((add) => { - switch (add.kind) { - case 'Day': - currDate.setDate(currDate.getDate() + add.number); - break; - case 'Month': - currDate.setMonth(currDate.getMonth() + add.number); - break; - case 'Year': - currDate.setFullYear(currDate.getFullYear() + add.number); - break; - default: - new Error(`Unknown type to add to the date: ${add.kind}`); - } - }); - - // Substract every sub in the subs array - subs.forEach((sub) => { - switch (sub.kind) { - case 'Day': - currDate.setDate(currDate.getDate() - sub.number); - break; - case 'Month': - currDate.setMonth(currDate.getMonth() - sub.number); - break; - case 'Year': - currDate.setFullYear(currDate.getFullYear() - sub.number); - break; - default: - new Error(`Unknown type to substract of the date: ${sub.kind}`); - } - }); - - // Format the date - const result = moment(currDate).format(format); - return result; -} - // "Radio" When('I select {string} from the selection {string}', async function clickRadioButton(radioname, label) { const world = this; @@ -465,7 +317,7 @@ When('I select {string} from the selection {string}', async function clickRadioB When('I select the option {string} from the drop-down-menue {string}', async (value, dropd) => { let world; const identifiers = [`//*[@*='${dropd}']/option[text()='${value}']`, `//label[contains(text(),'${dropd}')]/following::button[text()='${value}']`, - `//label[contains(text(),'${dropd}')]/following::span[text()='${value}']`, `//*[contains(text(),'${dropd}')]/following::*[contains(text(),'${value}']`, `${dropd}`]; + `//label[contains(text(),'${dropd}')]/following::span[text()='${value}']`, `//*[contains(text(),'${dropd}')]/following::*[contains(text(),'${value}']`, `${dropd}`]; const promises = []; for (const idString of identifiers) promises.push(driver.wait(until.elementLocated(By.xpath(idString)), searchTimeout, `Timed out after ${searchTimeout} ms`, 100)); @@ -540,7 +392,7 @@ When('I hover over the element {string} and select the option {string}', async f }); // TODO: -When('I select from the {string} multiple selection, the values {string}{string}{string}', async () => {}); +When('I select from the {string} multiple selection, the values {string}{string}{string}', async () => { }); // Check the Checkbox with a specific name or id When('I check the box {string}', async function checkBox(name) { @@ -668,7 +520,7 @@ Then('So I can see the text {string} in the textbox: {string}', async function c const world = this; const identifiers = [`//*[@id='${label}']`, `//*[@*='${label}']`, `//*[contains(@*, '${label}')]`, - `//label[contains(text(),'${label}')]/following::input[@type='text']`, `${label}`]; + `//label[contains(text(),'${label}')]/following::input[@type='text']`, `${label}`]; const promises = []; for (const idString of identifiers) promises.push(driver.wait(until.elementLocated(By.xpath(idString)), searchTimeout, `Timed out after ${searchTimeout} ms`, 100)); @@ -761,7 +613,8 @@ Then('So the picture {string} has the name {string}', async function checkPictur const identifiers = [`//picture[source[contains(@srcset, '${picture}')] or img[contains(@src, '${picture}') or contains(@alt, '${picture}') or @id='${picture}' or contains(@title, '${picture}')]]`, `//img[contains(@src, '${picture}') or contains(@alt, '${picture}') or @id='${picture}' or contains(@title, '${picture}')]`, `${picture}`]; const promises = []; for (const idString of identifiers) promises.push(driver.wait(until.elementLocated(By.xpath(idString)), searchTimeout, `Timed out after ${searchTimeout} ms`, 100)); - const domain = (await driver.getCurrentUrl()).split('/').slice(0, 3).join('/'); + const domain = (await driver.getCurrentUrl()).split('/').slice(0, 3) + .join('/'); let finSrc = ''; await Promise.any(promises) .then(async (elem) => { @@ -785,7 +638,7 @@ Then('So the picture {string} has the name {string}', async function checkPictur }); throw Error(e); }); - await fetch(domain + finSrc, { method: 'HEAD' }) + await fetch(domain + finSrc, { method: 'HEAD' }) .then((response) => { if (!response.ok) throw Error(`Image ${finSrc} not Found`); }) @@ -850,13 +703,15 @@ Then('So on element {string} the css property {string} is {string}', async funct await Promise.any(promises) .then(async (elem) => { const actual = await elem.getCssValue(property); - if (actual.startsWith('rgba')) {// in selenium colors are always rgba. support.Color is not implemented in javascript - const colorNumbers = actual.replace('rgba(', '').replace(')', '').split(','); + if (actual.startsWith('rgba')) { // in selenium colors are always rgba. support.Color is not implemented in javascript + const colorNumbers = actual.replace('rgba(', '').replace(')', '') + .split(','); const [r, g, b] = colorNumbers.map((v) => Number(v).toString(16)); const hex = `#${r}${g}${b}`; expect(value.toString()).to.equal(hex.toString(), `actual ${hex} does not match ${value}`); } else expect(value.toString()).to.equal(actual.toString(), `actual ${actual} does not match ${value}`); - }).catch(async (e) => { + }) + .catch(async (e) => { await driver.takeScreenshot().then(async (buffer) => { world.attach(buffer, 'image/png'); }); @@ -874,14 +729,15 @@ Then('So the element {string} has the tool-tip {string}', async function toolTip .then(async (elem) => { const actual = await elem.getAttribute('title'); expect(actual).to.equal(value); - }).catch(async (e) => { + }) + .catch(async (e) => { await driver.takeScreenshot().then(async (buffer) => { world.attach(buffer, 'image/png'); }); if (Object.keys(e).length === 0) throw NotFoundError(`The Element ${element} could not be found (check tool-tip).`); throw Error(e); }); - await driver.sleep(100 + currentParameters.waitTime); + await driver.sleep(100 + currentParameters.waitTime); }); // Closes the webdriver (Browser) diff --git a/backend/package-lock.json b/backend/package-lock.json index 0b9af320c..12bf807eb 100644 --- a/backend/package-lock.json +++ b/backend/package-lock.json @@ -1,12 +1,12 @@ { "name": "seed-test-backend", - "version": "1.6.2", + "version": "1.7.0", "lockfileVersion": 2, "requires": true, "packages": { "": { "name": "seed-test-backend", - "version": "1.6.2", + "version": "1.7.0", "dependencies": { "@cucumber/cucumber": "^7.3.2", "@types/promise-fs": "^2.1.2", @@ -24,7 +24,7 @@ "express-session": "^1.17.2", "geckodriver": "^3.0.1", "moment": "^2.29.4", - "mongodb": "^3.7.3", + "mongodb": "^6.2.0", "node-fetch": "^2.6.7", "nodemailer": "^6.7.5", "passport": "^0.6.0", @@ -56,7 +56,7 @@ "typescript": "^4.9.4" }, "engines": { - "node": "18.13.0" + "node": "^18.13.0" } }, "node_modules/@aashutoshrathi/word-wrap": { @@ -1763,6 +1763,14 @@ "node-pre-gyp": "bin/node-pre-gyp" } }, + "node_modules/@mongodb-js/saslprep": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@mongodb-js/saslprep/-/saslprep-1.1.1.tgz", + "integrity": "sha512-t7c5K033joZZMspnHg/gWPE4kandgc2OxE74aYOtGKfgB9VPuVJPix0H6fhmm2erj5PBJ21mqcx34lpIGtUCsQ==", + "dependencies": { + "sparse-bitfield": "^3.0.3" + } + }, "node_modules/@nodelib/fs.scandir": { "version": "2.1.5", "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", @@ -2083,6 +2091,20 @@ "resolved": "https://registry.npmjs.org/@types/uuid/-/uuid-8.3.0.tgz", "integrity": "sha512-eQ9qFW/fhfGJF8WKHGEHZEyVWfZxrT+6CLIJGBcZPfxUh/+BnEj+UCGYMlr9qZuX/2AltsvwrGqp0LhEW8D0zQ==" }, + "node_modules/@types/webidl-conversions": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/@types/webidl-conversions/-/webidl-conversions-7.0.3.tgz", + "integrity": "sha512-CiJJvcRtIgzadHCYXw7dqEnMNRjhGZlYK05Mj9OyktqV8uVT8fD2BFOB7S1uwBE3Kj2Z+4UyPmFw/Ixgw/LAlA==" + }, + "node_modules/@types/whatwg-url": { + "version": "8.2.2", + "resolved": "https://registry.npmjs.org/@types/whatwg-url/-/whatwg-url-8.2.2.tgz", + "integrity": "sha512-FtQu10RWgn3D9U4aazdwIE2yzphmTJREDqNdODHrbrZmmMqI0vMheC/6NE/J1Yveaj8H+ela+YwWTjq5PGmuhA==", + "dependencies": { + "@types/node": "*", + "@types/webidl-conversions": "*" + } + }, "node_modules/@types/yargs": { "version": "17.0.11", "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-17.0.11.tgz", @@ -2555,9 +2577,9 @@ } }, "node_modules/bl/node_modules/readable-stream": { - "version": "2.3.7", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", - "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.8.tgz", + "integrity": "sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==", "dependencies": { "core-util-is": "~1.0.0", "inherits": "~2.0.3", @@ -2568,7 +2590,7 @@ "util-deprecate": "~1.0.1" } }, - "node_modules/bl/node_modules/readable-stream/node_modules/safe-buffer": { + "node_modules/bl/node_modules/safe-buffer": { "version": "5.1.2", "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" @@ -2581,11 +2603,6 @@ "safe-buffer": "~5.1.0" } }, - "node_modules/bl/node_modules/string_decoder/node_modules/safe-buffer": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", - "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" - }, "node_modules/bluebird": { "version": "3.7.2", "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.7.2.tgz", @@ -2673,11 +2690,11 @@ } }, "node_modules/bson": { - "version": "1.1.6", - "resolved": "https://registry.npmjs.org/bson/-/bson-1.1.6.tgz", - "integrity": "sha512-EvVNVeGo4tHxwi8L6bPj3y3itEvStdwvvlojVxxbyYfoaxJ6keLgrTuKdyfEAszFK+H3olzBuafE0yoh0D1gdg==", + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/bson/-/bson-6.2.0.tgz", + "integrity": "sha512-ID1cI+7bazPDyL9wYy9GaQ8gEEohWvcUl/Yf0dIdutJxnmInEEyCsb4awy/OiBfall7zBA179Pahi3vCdFze3Q==", "engines": { - "node": ">=0.6.19" + "node": ">=16.20.1" } }, "node_modules/buffer-from": { @@ -3019,6 +3036,52 @@ "mongodb": "^3.1.0" } }, + "node_modules/connect-mongo/node_modules/bson": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/bson/-/bson-1.1.6.tgz", + "integrity": "sha512-EvVNVeGo4tHxwi8L6bPj3y3itEvStdwvvlojVxxbyYfoaxJ6keLgrTuKdyfEAszFK+H3olzBuafE0yoh0D1gdg==", + "engines": { + "node": ">=0.6.19" + } + }, + "node_modules/connect-mongo/node_modules/mongodb": { + "version": "3.7.4", + "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-3.7.4.tgz", + "integrity": "sha512-K5q8aBqEXMwWdVNh94UQTwZ6BejVbFhh1uB6c5FKtPE9eUMZPUO3sRZdgIEcHSrAWmxzpG/FeODDKL388sqRmw==", + "dependencies": { + "bl": "^2.2.1", + "bson": "^1.1.4", + "denque": "^1.4.1", + "optional-require": "^1.1.8", + "safe-buffer": "^5.1.2" + }, + "engines": { + "node": ">=4" + }, + "optionalDependencies": { + "saslprep": "^1.0.0" + }, + "peerDependenciesMeta": { + "aws4": { + "optional": true + }, + "bson-ext": { + "optional": true + }, + "kerberos": { + "optional": true + }, + "mongodb-client-encryption": { + "optional": true + }, + "mongodb-extjson": { + "optional": true + }, + "snappy": { + "optional": true + } + } + }, "node_modules/console-control-strings": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/console-control-strings/-/console-control-strings-1.1.0.tgz", @@ -7380,8 +7443,7 @@ "node_modules/memory-pager": { "version": "1.5.0", "resolved": "https://registry.npmjs.org/memory-pager/-/memory-pager-1.5.0.tgz", - "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==", - "optional": true + "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==" }, "node_modules/merge-descriptors": { "version": "1.0.1", @@ -7510,27 +7572,34 @@ } }, "node_modules/mongodb": { - "version": "3.7.3", - "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-3.7.3.tgz", - "integrity": "sha512-Psm+g3/wHXhjBEktkxXsFMZvd3nemI0r3IPsE0bU+4//PnvNWKkzhZcEsbPcYiWqe8XqXJJEg4Tgtr7Raw67Yw==", + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-6.2.0.tgz", + "integrity": "sha512-d7OSuGjGWDZ5usZPqfvb36laQ9CPhnWkAGHT61x5P95p/8nMVeH8asloMwW6GcYFeB0Vj4CB/1wOTDG2RA9BFA==", "dependencies": { - "bl": "^2.2.1", - "bson": "^1.1.4", - "denque": "^1.4.1", - "optional-require": "^1.1.8", - "safe-buffer": "^5.1.2" + "@mongodb-js/saslprep": "^1.1.0", + "bson": "^6.2.0", + "mongodb-connection-string-url": "^2.6.0" }, "engines": { - "node": ">=4" + "node": ">=16.20.1" }, - "optionalDependencies": { - "saslprep": "^1.0.0" + "peerDependencies": { + "@aws-sdk/credential-providers": "^3.188.0", + "@mongodb-js/zstd": "^1.1.0", + "gcp-metadata": "^5.2.0", + "kerberos": "^2.0.1", + "mongodb-client-encryption": ">=6.0.0 <7", + "snappy": "^7.2.2", + "socks": "^2.7.1" }, "peerDependenciesMeta": { - "aws4": { + "@aws-sdk/credential-providers": { "optional": true }, - "bson-ext": { + "@mongodb-js/zstd": { + "optional": true + }, + "gcp-metadata": { "optional": true }, "kerberos": { @@ -7539,14 +7608,54 @@ "mongodb-client-encryption": { "optional": true }, - "mongodb-extjson": { + "snappy": { "optional": true }, - "snappy": { + "socks": { "optional": true } } }, + "node_modules/mongodb-connection-string-url": { + "version": "2.6.0", + "resolved": "https://registry.npmjs.org/mongodb-connection-string-url/-/mongodb-connection-string-url-2.6.0.tgz", + "integrity": "sha512-WvTZlI9ab0QYtTYnuMLgobULWhokRjtC7db9LtcVfJ+Hsnyr5eo6ZtNAt3Ly24XZScGMelOcGtm7lSn0332tPQ==", + "dependencies": { + "@types/whatwg-url": "^8.2.1", + "whatwg-url": "^11.0.0" + } + }, + "node_modules/mongodb-connection-string-url/node_modules/tr46": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-3.0.0.tgz", + "integrity": "sha512-l7FvfAHlcmulp8kr+flpQZmVwtu7nfRV7NZujtN0OqES8EL4O4e0qqzL0DC5gAvx/ZC/9lk6rhcUwYvkBnBnYA==", + "dependencies": { + "punycode": "^2.1.1" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/mongodb-connection-string-url/node_modules/webidl-conversions": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-7.0.0.tgz", + "integrity": "sha512-VwddBukDzu71offAQR975unBIGqfKZpM+8ZX6ySk8nYhVoo5CYaZyzt3YBvYtRtO+aoGlqxPg/B87NGVZ/fu6g==", + "engines": { + "node": ">=12" + } + }, + "node_modules/mongodb-connection-string-url/node_modules/whatwg-url": { + "version": "11.0.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-11.0.0.tgz", + "integrity": "sha512-RKT8HExMpoYx4igMiVMY83lN6UeITKJlBQ+vR/8ZJ8OCdSiN3RwCq+9gH0+Xzj0+5IrM6i4j/6LuvzbZIQgEcQ==", + "dependencies": { + "tr46": "^3.0.0", + "webidl-conversions": "^7.0.0" + }, + "engines": { + "node": ">=12" + } + }, "node_modules/ms": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", @@ -8246,8 +8355,6 @@ "version": "2.1.1", "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==", - "dev": true, - "peer": true, "engines": { "node": ">=6" } @@ -8805,8 +8912,7 @@ "node_modules/sparse-bitfield": { "version": "3.0.3", "resolved": "https://registry.npmjs.org/sparse-bitfield/-/sparse-bitfield-3.0.3.tgz", - "integrity": "sha1-/0rm5oZWBWuks+eSqzM004JzyhE=", - "optional": true, + "integrity": "sha512-kvzhi7vqKTfkh0PZU+2D2PIllw2ymqJKujUcyPMd9Y75Nv4nPbGJZXNhxsgdQab2BmlDct1YnfQCguEvHr7VsQ==", "dependencies": { "memory-pager": "^1.0.2" } @@ -11051,6 +11157,14 @@ "tar": "^6.1.11" } }, + "@mongodb-js/saslprep": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@mongodb-js/saslprep/-/saslprep-1.1.1.tgz", + "integrity": "sha512-t7c5K033joZZMspnHg/gWPE4kandgc2OxE74aYOtGKfgB9VPuVJPix0H6fhmm2erj5PBJ21mqcx34lpIGtUCsQ==", + "requires": { + "sparse-bitfield": "^3.0.3" + } + }, "@nodelib/fs.scandir": { "version": "2.1.5", "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", @@ -11352,6 +11466,20 @@ "resolved": "https://registry.npmjs.org/@types/uuid/-/uuid-8.3.0.tgz", "integrity": "sha512-eQ9qFW/fhfGJF8WKHGEHZEyVWfZxrT+6CLIJGBcZPfxUh/+BnEj+UCGYMlr9qZuX/2AltsvwrGqp0LhEW8D0zQ==" }, + "@types/webidl-conversions": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/@types/webidl-conversions/-/webidl-conversions-7.0.3.tgz", + "integrity": "sha512-CiJJvcRtIgzadHCYXw7dqEnMNRjhGZlYK05Mj9OyktqV8uVT8fD2BFOB7S1uwBE3Kj2Z+4UyPmFw/Ixgw/LAlA==" + }, + "@types/whatwg-url": { + "version": "8.2.2", + "resolved": "https://registry.npmjs.org/@types/whatwg-url/-/whatwg-url-8.2.2.tgz", + "integrity": "sha512-FtQu10RWgn3D9U4aazdwIE2yzphmTJREDqNdODHrbrZmmMqI0vMheC/6NE/J1Yveaj8H+ela+YwWTjq5PGmuhA==", + "requires": { + "@types/node": "*", + "@types/webidl-conversions": "*" + } + }, "@types/yargs": { "version": "17.0.11", "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-17.0.11.tgz", @@ -11711,9 +11839,9 @@ }, "dependencies": { "readable-stream": { - "version": "2.3.7", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", - "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.8.tgz", + "integrity": "sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==", "requires": { "core-util-is": "~1.0.0", "inherits": "~2.0.3", @@ -11722,28 +11850,19 @@ "safe-buffer": "~5.1.1", "string_decoder": "~1.1.1", "util-deprecate": "~1.0.1" - }, - "dependencies": { - "safe-buffer": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", - "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" - } } }, + "safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" + }, "string_decoder": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", "requires": { "safe-buffer": "~5.1.0" - }, - "dependencies": { - "safe-buffer": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", - "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" - } } } } @@ -11812,9 +11931,9 @@ } }, "bson": { - "version": "1.1.6", - "resolved": "https://registry.npmjs.org/bson/-/bson-1.1.6.tgz", - "integrity": "sha512-EvVNVeGo4tHxwi8L6bPj3y3itEvStdwvvlojVxxbyYfoaxJ6keLgrTuKdyfEAszFK+H3olzBuafE0yoh0D1gdg==" + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/bson/-/bson-6.2.0.tgz", + "integrity": "sha512-ID1cI+7bazPDyL9wYy9GaQ8gEEohWvcUl/Yf0dIdutJxnmInEEyCsb4awy/OiBfall7zBA179Pahi3vCdFze3Q==" }, "buffer-from": { "version": "1.1.2", @@ -12074,6 +12193,26 @@ "integrity": "sha512-0Mx88079Z20CG909wCFlR3UxhMYGg6Ibn1hkIje1hwsqOLWtL9HJV+XD0DAjUvQScK6WqY/FA8tSVQM9rR64Rw==", "requires": { "mongodb": "^3.1.0" + }, + "dependencies": { + "bson": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/bson/-/bson-1.1.6.tgz", + "integrity": "sha512-EvVNVeGo4tHxwi8L6bPj3y3itEvStdwvvlojVxxbyYfoaxJ6keLgrTuKdyfEAszFK+H3olzBuafE0yoh0D1gdg==" + }, + "mongodb": { + "version": "3.7.4", + "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-3.7.4.tgz", + "integrity": "sha512-K5q8aBqEXMwWdVNh94UQTwZ6BejVbFhh1uB6c5FKtPE9eUMZPUO3sRZdgIEcHSrAWmxzpG/FeODDKL388sqRmw==", + "requires": { + "bl": "^2.2.1", + "bson": "^1.1.4", + "denque": "^1.4.1", + "optional-require": "^1.1.8", + "safe-buffer": "^5.1.2", + "saslprep": "^1.0.0" + } + } } }, "console-control-strings": { @@ -15408,8 +15547,7 @@ "memory-pager": { "version": "1.5.0", "resolved": "https://registry.npmjs.org/memory-pager/-/memory-pager-1.5.0.tgz", - "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==", - "optional": true + "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==" }, "merge-descriptors": { "version": "1.0.1", @@ -15505,16 +15643,46 @@ "integrity": "sha512-5LC9SOxjSc2HF6vO2CyuTDNivEdoz2IvyJJGj6X8DJ0eFyfszE0QiEd+iXmBvUP3WHxSjFH/vIsA0EN00cgr8w==" }, "mongodb": { - "version": "3.7.3", - "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-3.7.3.tgz", - "integrity": "sha512-Psm+g3/wHXhjBEktkxXsFMZvd3nemI0r3IPsE0bU+4//PnvNWKkzhZcEsbPcYiWqe8XqXJJEg4Tgtr7Raw67Yw==", + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-6.2.0.tgz", + "integrity": "sha512-d7OSuGjGWDZ5usZPqfvb36laQ9CPhnWkAGHT61x5P95p/8nMVeH8asloMwW6GcYFeB0Vj4CB/1wOTDG2RA9BFA==", "requires": { - "bl": "^2.2.1", - "bson": "^1.1.4", - "denque": "^1.4.1", - "optional-require": "^1.1.8", - "safe-buffer": "^5.1.2", - "saslprep": "^1.0.0" + "@mongodb-js/saslprep": "^1.1.0", + "bson": "^6.2.0", + "mongodb-connection-string-url": "^2.6.0" + } + }, + "mongodb-connection-string-url": { + "version": "2.6.0", + "resolved": "https://registry.npmjs.org/mongodb-connection-string-url/-/mongodb-connection-string-url-2.6.0.tgz", + "integrity": "sha512-WvTZlI9ab0QYtTYnuMLgobULWhokRjtC7db9LtcVfJ+Hsnyr5eo6ZtNAt3Ly24XZScGMelOcGtm7lSn0332tPQ==", + "requires": { + "@types/whatwg-url": "^8.2.1", + "whatwg-url": "^11.0.0" + }, + "dependencies": { + "tr46": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-3.0.0.tgz", + "integrity": "sha512-l7FvfAHlcmulp8kr+flpQZmVwtu7nfRV7NZujtN0OqES8EL4O4e0qqzL0DC5gAvx/ZC/9lk6rhcUwYvkBnBnYA==", + "requires": { + "punycode": "^2.1.1" + } + }, + "webidl-conversions": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-7.0.0.tgz", + "integrity": "sha512-VwddBukDzu71offAQR975unBIGqfKZpM+8ZX6ySk8nYhVoo5CYaZyzt3YBvYtRtO+aoGlqxPg/B87NGVZ/fu6g==" + }, + "whatwg-url": { + "version": "11.0.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-11.0.0.tgz", + "integrity": "sha512-RKT8HExMpoYx4igMiVMY83lN6UeITKJlBQ+vR/8ZJ8OCdSiN3RwCq+9gH0+Xzj0+5IrM6i4j/6LuvzbZIQgEcQ==", + "requires": { + "tr46": "^3.0.0", + "webidl-conversions": "^7.0.0" + } + } } }, "ms": { @@ -16033,9 +16201,7 @@ "punycode": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", - "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==", - "dev": true, - "peer": true + "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==" }, "qs": { "version": "6.10.3", @@ -16425,8 +16591,7 @@ "sparse-bitfield": { "version": "3.0.3", "resolved": "https://registry.npmjs.org/sparse-bitfield/-/sparse-bitfield-3.0.3.tgz", - "integrity": "sha1-/0rm5oZWBWuks+eSqzM004JzyhE=", - "optional": true, + "integrity": "sha512-kvzhi7vqKTfkh0PZU+2D2PIllw2ymqJKujUcyPMd9Y75Nv4nPbGJZXNhxsgdQab2BmlDct1YnfQCguEvHr7VsQ==", "requires": { "memory-pager": "^1.0.2" } diff --git a/backend/package.json b/backend/package.json index f185e42a5..0e00d6f1d 100644 --- a/backend/package.json +++ b/backend/package.json @@ -1,8 +1,8 @@ { "name": "seed-test-backend", - "version": "1.6.2", + "version": "1.7.0", "engines": { - "node": "18.13.0" + "node": "^18.13.0" }, "scripts": { "start": "tsc && node src/server.js", @@ -32,7 +32,7 @@ "express-session": "^1.17.2", "geckodriver": "^3.0.1", "moment": "^2.29.4", - "mongodb": "^3.7.3", + "mongodb": "^6.2.0", "node-fetch": "^2.6.7", "nodemailer": "^6.7.5", "passport": "^0.6.0", @@ -69,6 +69,8 @@ "/spec/serverHelper.spec.js", "/node_modules/" ], - "setupFiles": ["./spec/setupTest.js"] + "setupFiles": [ + "./spec/setupTest.js" + ] } } diff --git a/backend/src/database/DbConnector.js b/backend/src/database/DbConnector.js index eab5ffb36..1c2b6fb4e 100644 --- a/backend/src/database/DbConnector.js +++ b/backend/src/database/DbConnector.js @@ -1,21 +1,23 @@ const { MongoClient } = require('mongodb'); -const { exit } = require('process'); +const { setTimeout } = require('timers/promises'); const uri = process.env.DATABASE_URI || 'mongodb://SeedAdmin:SeedTest@seedmongodb:27017'; let connection = []; // Create the database connection -async function establishConnection(callback) { - // eslint-disable-next-line max-len - MongoClient.connect(uri, { poolSize: 20, useNewUrlParser: true, useUnifiedTopology: true }, async (err, db) => { - if (err) { - console.log('DB_ERROR: Can`t connect to DB. The Project may not be set up correctly. For more information read the README'); - exit(-1); - } - connection = db.db('Seed'); - if (typeof callback === 'function' && callback()) callback(connection); - console.log('Established Database Connection!'); - }); +async function establishConnection(attempt) { + const attempts = attempt || 1; + if (attempt > 3) throw new Error('\x1b[31mFailed to connect to the database after multiple retries.\x1b[0m'); + + try { + const client = await MongoClient.connect(uri, { maxPoolSize: 20 }); + connection = client.db('Seed'); + return client; + } catch (err) { + console.log(`\x1b[38;5;208mConnection failed! Retrying... ${attempts}\x1b[0m`); + await setTimeout(3000); + return establishConnection(attempts + 1); + } } function getConnection() { diff --git a/backend/src/database/DbServices.js b/backend/src/database/DbServices.js index 7271dd8db..96ad726c0 100644 --- a/backend/src/database/DbServices.js +++ b/backend/src/database/DbServices.js @@ -13,11 +13,11 @@ const toString = require('stream-to-string'); const assert = require('assert'); const mongodb = require('mongodb'); const fs = require('fs'); +const Collection = require('mongodb/lib/collection'); const dbConnection = require('./DbConnector'); const emptyStory = require('../models/emptyStory'); const emptyScenario = require('../models/emptyScenario'); const emptyBackground = require('../models/emptyBackground'); -const Collection = require('mongodb/lib/collection'); if (process.env.NODE_ENV !== 'production') { require('dotenv').config(); @@ -37,7 +37,7 @@ const ReportsCollection = 'Reports'; // Opening a pooling Database Connection via DbConnector dbConnection.establishConnection() - .then(() => console.log('db ', dbConnection.getConnection())); + .then(() => console.log('\x1b[32m%s\x1b[0m \x1b[33m%s\x1b[0m', 'Connected to database @', dbConnection.getConnection().client.s.options.srvHost)); /** * Writes a PasswordResetRequest in the DB @@ -87,7 +87,7 @@ async function getResetRequestByEmail(mail) { } /** - * deletes Password reset request + * deletes Password reset request * @param {*} mail * @returns deletion Report */ @@ -110,7 +110,7 @@ async function deleteRequest(mail) { async function getUserById(userId) { try { const db = dbConnection.getConnection(); - return await db.collection(userCollection).findOne({ _id: ObjectId(userId) }); + return await db.collection(userCollection).findOne({ _id: new ObjectId(userId) }); } catch (e) { console.log(`ERROR in getUserById: ${e}`); throw e; @@ -150,7 +150,7 @@ async function getUserByEmail(email) { /** * - * @param {object} user {email:string, userId: String, password: string} + * @param {object} user {email:string, userId: String, password: string} * @returns */ async function registerUser(user) { @@ -161,7 +161,7 @@ async function registerUser(user) { let result; if (dbUser !== null) throw Error('User already exists'); else if (user.userId) { // update in register? attacker with userId could re-set anything - result = await collection.update({ _id: ObjectId(user.userId) }, { $set: { email: user.email, password: user.password } }); + result = await collection.updateOne({ _id: new ObjectId(user.userId) }, { $set: { email: user.email, password: user.password } }); } else { delete user.userId; const query = { email: user.email.toString(), password: user.password }; @@ -197,14 +197,14 @@ async function registerGithubUser(user) { * @returns */ function replaceUser(newUser, collection) { - const myObjt = { _id: ObjectId(newUser._id) }; - return collection.findOneAndReplace(myObjt, newUser).then((res) => res.value); + const myObjt = { _id: new ObjectId(newUser._id) }; + return collection.findOneAndReplace(myObjt, newUser); } async function updateGithubToken(objId, updatedToken) { try { const db = dbConnection.getConnection(); - return await db.collection(userCollection).updateOne({ _id: ObjectId(objId) }, { $set: { 'github.githubToken': updatedToken } }); + return await db.collection(userCollection).updateOne({ _id: new ObjectId(objId) }, { $set: { 'github.githubToken': updatedToken } }); } catch (e) { console.log(`ERROR in updateGithubToken: ${e}`); throw e; @@ -242,18 +242,11 @@ async function mergeGithub(userId, login, id) { * * @returns updated UserObject * @param storyId - * @param storySource * @param collection */ -// TODO: storySource wont be needed anymore function findStory(storyId, collection) { - const id = ObjectId(storyId); - return new Promise((resolve, reject) => { - collection.findOne({ _id: id }, (err, result) => { - if (err) reject(err); - else resolve(result); - }); - }); + const id = new ObjectId(storyId); + return collection.findOne({ _id: id }); } /** @@ -264,15 +257,10 @@ function findStory(storyId, collection) { */ function replace(story, collection) { const filter = { - _id: ObjectId(story._id.toString()), + _id: new ObjectId(story._id.toString()) }; - story._id = ObjectId(story._id); - return new Promise((resolve, reject) => { - collection.findOneAndReplace(filter, story, (err, result) => { - if (err) reject(err); - else resolve(result.value); - }); - }); + story._id = new ObjectId(story._id); + return collection.findOneAndReplace(filter, story); } async function disconnectGithub(user) { @@ -290,8 +278,8 @@ async function updateStory(updatedStory) { try { const db = dbConnection.getConnection(); const collection = await db.collection(storiesCollection); - updatedStory._id = ObjectId(updatedStory._id.toString()) - return await collection.findOneAndReplace({_id: ObjectId(updatedStory._id.toString())}, updatedStory, {returnDocument: "after"}) + updatedStory._id = new ObjectId(updatedStory._id.toString()); + return await collection.findOneAndReplace({ _id: new ObjectId(updatedStory._id.toString()) }, updatedStory, { returnDocument: 'after' }); } catch (e) { console.log(`ERROR updateStory: ${e}`); throw e; @@ -299,7 +287,7 @@ async function updateStory(updatedStory) { } // get One Story -// searches the story either by mongoDB _id:ObjectId() or by story_id (from GitHub or Jira) +// searches the story either by mongoDB _id:new ObjectId() or by story_id (from GitHub or Jira) async function getOneStory(storyId) { let query; try { @@ -312,7 +300,7 @@ async function getOneStory(storyId) { }; } else { query = { - _id: ObjectId(storyId.toString()) + _id: new ObjectId(storyId.toString()) }; } return await collection.findOne(query); @@ -329,17 +317,17 @@ async function createStoryGroup(repoID, name, members, sequence) { try { const db = dbConnection.getConnection(); const groups = await db.collection(repositoriesCollection).findOneAndUpdate( - { _id: ObjectId(repoID) }, + { _id: new ObjectId(repoID) }, { $push: { groups: { - _id: ObjectId(), name, member_stories: members, isSequential: sequence + _id: new ObjectId(), name, member_stories: members, isSequential: sequence } } }, { upsert: true, projection: { groups: 1 } } ); - return groups.value.groups.slice(-1)._id; + return groups.groups.slice(-1)._id; } catch (e) { console.log(`ERROR in createStoryGroup: ${e}`); } @@ -348,13 +336,13 @@ async function createStoryGroup(repoID, name, members, sequence) { async function updateStoryGroup(repoId, groupId, updatedGroup) { try { const db = dbConnection.getConnection(); - updatedGroup._id = ObjectId(updatedGroup._id); + updatedGroup._id = new ObjectId(updatedGroup._id); const collection = await db.collection(repositoriesCollection); - const repo = await collection.findOne({ _id: ObjectId(repoId) }); + const repo = await collection.findOne({ _id: new ObjectId(repoId) }); // leave with double equal: const index = repo.groups.findIndex((o) => o._id == groupId); repo.groups[index] = updatedGroup; - await collection.updateOne({ _id: ObjectId(repoId) }, { $set: repo }); + await collection.updateOne({ _id: new ObjectId(repoId) }, { $set: repo }); return updatedGroup; } catch (e) { console.log(`ERROR in updateStoryGroup: ${e}`); @@ -365,11 +353,11 @@ async function deleteStoryGroup(repoId, groupId) { try { const db = dbConnection.getConnection(); const collection = await db.collection(repositoriesCollection); - const repo = await collection.findOne({ _id: ObjectId(repoId) }); + const repo = await collection.findOne({ _id: new ObjectId(repoId) }); // leave with double equal: const index = repo.groups.findIndex((o) => o._id == groupId); repo.groups.splice(index, 1); - await collection.updateOne({ _id: ObjectId(repoId) }, { $set: repo }); + await collection.updateOne({ _id: new ObjectId(repoId) }, { $set: repo }); return null; } catch (e) { console.log(`ERROR in deleteStoryGroup: ${e}`); @@ -379,7 +367,7 @@ async function deleteStoryGroup(repoId, groupId) { async function getAllStoryGroups(repoId) { try { const db = dbConnection.getConnection(); - return await db.collection(repositoriesCollection).findOne({ _id: ObjectId(repoId) }, { projection: { groups: 1 } }); + return await db.collection(repositoriesCollection).findOne({ _id: new ObjectId(repoId) }, { projection: { groups: 1 } }); } catch (e) { console.log(`ERROR in getAllStoryGroups: ${e}`); } @@ -419,7 +407,7 @@ async function removeFromStoryGroup(repoId, groupId, storyId) { async function updateStoryGroupsArray(repoId, groupsArray) { try { const db = dbConnection.getConnection(); - return await db.collection(repositoriesCollection).findOneAndUpdate({ _id: ObjectId(repoId) }, { $set: { groups: groupsArray } }, { projection: { groups: 1 } }); + return await db.collection(repositoriesCollection).findOneAndUpdate({ _id: new ObjectId(repoId) }, { $set: { groups: groupsArray } }, { projection: { groups: 1 } }); } catch (e) { console.log(`ERROR in updateStoryGroupsArray: ${e}`); } @@ -445,7 +433,7 @@ async function updateBackground(storyId, updatedBackground) { try { const db = dbConnection.getConnection(); const collection = await db.collection(storiesCollection); - return collection.findOneAndUpdate({ _id: ObjectId(storyId) },{$set:{"background": updatedBackground}},{returnDocument: "after", upsert: true}).then((res)=>res.value); + return collection.findOneAndUpdate({ _id: new ObjectId(storyId) }, { $set: { background: updatedBackground } }, { returnDocument: 'after', upsert: true }); } catch (e) { console.log(`ERROR in updateBackground: ${e}`); throw e; @@ -462,10 +450,10 @@ async function createStory(storyTitle, storyDescription, repoId) { let finalIssueNumber = 1; try { const db = dbConnection.getConnection(); - const repo = await db.collection(repositoriesCollection).findOne({ _id: ObjectId(repoId) }); + const repo = await db.collection(repositoriesCollection).findOne({ _id: new ObjectId(repoId) }); if (repo && repo.stories.length > 0) { for (const storyId of repo.stories) { - const story = await db.collection(storiesCollection).findOne({ _id: ObjectId(storyId) }); + const story = await db.collection(storiesCollection).findOne({ _id: new ObjectId(storyId) }); iNumberArray.push(story.issue_number); } finalIssueNumber = iNumberArray.findIndex((_, i) => !iNumberArray.includes(i + 1)) + 1; @@ -487,18 +475,18 @@ async function createStory(storyTitle, storyDescription, repoId) { * @param {*} storyId * @returns deleteReport */ -async function deleteStory(repoId, storyId) { //TODO refactor use promise all +async function deleteStory(repoId, storyId) { // TODO refactor use promise all try { const db = dbConnection.getConnection(); const repo = await db.collection(repositoriesCollection); try { - const groups = await repo.findOne({ _id: ObjectId(repoId) }, { projection: { groups: 1 } }); + const groups = await repo.findOne({ _id: new ObjectId(repoId) }, { projection: { groups: 1 } }); for (const index in groups.groups) groups.groups[index].member_stories = groups.groups[index].member_stories.filter((story) => story !== storyId); - await repo.findOneAndUpdate({ _id: ObjectId(repoId) }, { $set: { groups: groups.groups } }); + await repo.findOneAndUpdate({ _id: new ObjectId(repoId) }, { $set: { groups: groups.groups } }); try { - await repo.findOneAndUpdate({ _id: ObjectId(repoId) }, { $pull: { stories: ObjectId(storyId) } }); + await repo.findOneAndUpdate({ _id: new ObjectId(repoId) }, { $pull: { stories: new ObjectId(storyId) } }); try { - return await db.collection(storiesCollection).findOneAndDelete({ _id: ObjectId(storyId) }); + return await db.collection(storiesCollection).findOneAndDelete({ _id: new ObjectId(storyId) }); } catch (e) { console.log(`ERROR in deleteStory, couldn't delete the Story. Trying to recreate the Repo- and GroupsEntry: ${e}`); // TODO: recreate both Entrys @@ -520,7 +508,7 @@ async function deleteStory(repoId, storyId) { //TODO refactor use promise all async function insertStoryIdIntoRepo(storyId, repoId) { try { const db = dbConnection.getConnection(); - return await db.collection(repositoriesCollection).findOneAndUpdate({ _id: ObjectId(repoId) }, { $push: { stories: ObjectId(storyId) } }); + return await db.collection(repositoriesCollection).findOneAndUpdate({ _id: new ObjectId(repoId) }, { $push: { stories: new ObjectId(storyId) } }); } catch (e) { console.log(`ERROR in insertStoryIdIntoRepo: ${e}`); throw e; @@ -530,20 +518,20 @@ async function insertStoryIdIntoRepo(storyId, repoId) { async function updateScenarioList(storyId, scenarioList) { try { const db = dbConnection.getConnection(); - return await db.collection(storiesCollection).findOneAndUpdate({ _id: ObjectId(storyId) }, { $set: { scenarios: scenarioList } }); + return await db.collection(storiesCollection).findOneAndUpdate({ _id: new ObjectId(storyId) }, { $set: { scenarios: scenarioList } }); } catch (e) { console.log(`ERROR in insertStoryIdIntoRepo: ${e}`); throw e; } } -async function getAllStoriesOfRepo( repoId) { +async function getAllStoriesOfRepo(repoId) { const storiesArray = []; try { const db = dbConnection.getConnection(); - const repo = await db.collection(repositoriesCollection).findOne({ _id: ObjectId(repoId) }); + const repo = await db.collection(repositoriesCollection).findOne({ _id: new ObjectId(repoId) }); if (repo) for (const entry of repo.stories) { - const story = await db.collection(storiesCollection).findOne({ _id: ObjectId(entry) }); + const story = await db.collection(storiesCollection).findOne({ _id: new ObjectId(entry) }); storiesArray.push(story); } return storiesArray; @@ -557,7 +545,7 @@ async function getAllStoriesOfRepo( repoId) { async function getOneScenario(storyId, scenarioId) { try { const db = dbConnection.getConnection(); - const scenarios = await db.collection(storiesCollection).findOne({ _id: ObjectId(storyId), 'scenarios.scenario_id': scenarioId }, { projection: { scenarios: 1 } }); + const scenarios = await db.collection(storiesCollection).findOne({ _id: new ObjectId(storyId), 'scenarios.scenario_id': scenarioId }, { projection: { scenarios: 1 } }); return scenarios.scenarios.find((o) => o.scenario_id === scenarioId); } catch (e) { console.log(`ERROR in getOneScenario: ${e}`); @@ -605,10 +593,14 @@ async function updateScenario(storyId, updatedScenario) { try { const db = dbConnection.getConnection(); const collection = await db.collection(storiesCollection); - return collection.findOneAndUpdate({ _id: ObjectId(storyId) },{$set:{"scenarios.$[it]": updatedScenario}}, - {arrayFilters:[{"it.scenario_id": updatedScenario.scenario_id}], returnDocument: "after", upsert: true, projection:{scenarios:true}})//Options - .then((res)=>{return res.value}) - .then((result)=> result.scenarios.find((scen)=>scen.scenario_id==updatedScenario.scenario_id)) + return collection.findOneAndUpdate( + { _id: new ObjectId(storyId) }, + { $set: { 'scenarios.$[it]': updatedScenario } }, + { + arrayFilters: [{ 'it.scenario_id': updatedScenario.scenario_id }], returnDocument: 'after', upsert: true, projection: { scenarios: true } + } + )// Options + .then((result) => result.scenarios.find((scen) => scen.scenario_id == updatedScenario.scenario_id)); } catch (e) { console.log(`ERROR in updateScenario: ${e}`); throw e; @@ -620,7 +612,7 @@ async function deleteScenario(storyId, scenarioId) { try { const db = dbConnection.getConnection(); const collection = await db.collection(storiesCollection); - return collection.findOneAndUpdate({ _id: ObjectId(storyId) },{$pull:{"scenarios":{"scenario_id": scenarioId}}},{returnDocument: "after"}).then((res)=> res.value); + return collection.findOneAndUpdate({ _id: new ObjectId(storyId) }, { $pull: { scenarios: { scenario_id: scenarioId } } }, { returnDocument: 'after' }).then((res) => res); } catch (e) { console.log(`ERROR in deleteScenario: ${e}`); throw e; @@ -630,19 +622,19 @@ async function deleteScenario(storyId, scenarioId) { // gets all Repositories of one user async function getRepository(userID) { try { - const myObjt = { owner: ObjectId(userID) }; + const myObjt = { owner: new ObjectId(userID) }; const db = dbConnection.getConnection(); const wGCollection = await db.collection(WorkgroupsCollection); const repoCollection = await db.collection(repositoriesCollection); - const user = await db.collection(userCollection).findOne({ _id: ObjectId(userID) }); + const user = await db.collection(userCollection).findOne({ _id: new ObjectId(userID) }); const positiveWorkgroups = await wGCollection.find({ Members: { $elemMatch: { email: user.email, canEdit: true } } }).toArray(); - const PWgArray = positiveWorkgroups.map((entry) => ObjectId(entry.Repo)); + const PWgArray = positiveWorkgroups.map((entry) => new ObjectId(entry.Repo)); const PWgRepos = await repoCollection.find({ _id: { $in: PWgArray } }).toArray(); PWgRepos.forEach((element) => { element.canEdit = true; }); const negativeWorkgroups = await wGCollection.find({ Members: { $elemMatch: { email: user.email, canEdit: false } } }).toArray(); - const NWgArray = negativeWorkgroups.map((entry) => ObjectId(entry.Repo)); + const NWgArray = negativeWorkgroups.map((entry) => new ObjectId(entry.Repo)); const NWgRepos = await repoCollection.find({ _id: { $in: NWgArray } }).toArray(); NWgRepos.forEach((element) => { element.canEdit = false; @@ -661,7 +653,7 @@ async function getRepository(userID) { // deletes all Repositories of own User async function deleteRepositorys(ownerID) { // TODO: Dringend! Die eingetragenen Storys und die Einträge in Stories und Groups müssen gelöscht werden try { - const query = { owner: ObjectId(ownerID) }; + const query = { owner: new ObjectId(ownerID) }; const db = dbConnection.getConnection(); const collection = await db.collection(repositoriesCollection); return await collection.deleteMany(query); @@ -672,14 +664,14 @@ async function deleteRepositorys(ownerID) { // TODO: Dringend! Die eingetragenen } async function deleteRepository(repoId, ownerId) { // TODO: Dringend! Die eingetragenen Storys und die Einträge in Stories und Groups müssen gelöscht werden - try {// todo delete Workgroup, delete story Reports + try { // todo delete Workgroup, delete story Reports const db = dbConnection.getConnection(); - const collectionRepo = await db.collection(repositoriesCollection) + const collectionRepo = await db.collection(repositoriesCollection); // const collectionStory = await db.collection(storiesCollection) - // const repo = await collectionRepo.findOne({ owner: ObjectId(ownerId), _id: ObjectId(repoId)}) - // const storIds = repo.stories.map((val)=>ObjectId(val)) + // const repo = await collectionRepo.findOne({ owner: new ObjectId(ownerId), _id: new ObjectId(repoId)}) + // const storIds = repo.stories.map((val)=>new ObjectId(val)) // const storiesRes = await collectionStory.deleteMany({_id:{$in: storIds}}) - return collectionRepo.deleteOne({ owner: ObjectId(ownerId), _id: ObjectId(repoId)}) + return collectionRepo.deleteOne({ owner: new ObjectId(ownerId), _id: new ObjectId(repoId) }); } catch (e) { console.log(`ERROR in deleteRepository${e}`); throw e; @@ -688,7 +680,7 @@ async function deleteRepository(repoId, ownerId) { // TODO: Dringend! Die einget async function getOneRepository(ownerId, name) { try { - const repo = { owner: ObjectId(ownerId), repoName: name }; + const repo = { owner: new ObjectId(ownerId), repoName: name }; const db = dbConnection.getConnection(); return db.collection(repositoriesCollection).findOne(repo); } catch (e) { @@ -698,7 +690,7 @@ async function getOneRepository(ownerId, name) { async function getOneRepositoryById(repoId) { try { - const repo = { _id: ObjectId(repoId) }; + const repo = { _id: new ObjectId(repoId) }; const db = dbConnection.getConnection(); return db.collection(repositoriesCollection).findOne(repo); } catch (e) { @@ -758,14 +750,14 @@ async function getAllSourceReposFromDb(source) { async function createRepo(ownerId, name) { try { const emptyRepo = { - owner: ObjectId(ownerId), repoName: name.toString(), stories: [], repoType: 'db', customBlocks: [], groups: [] + owner: new ObjectId(ownerId), repoName: name.toString(), stories: [], repoType: 'db', customBlocks: [], groups: [] }; const db = dbConnection.getConnection(); const collection = await db.collection(repositoriesCollection); - const query = { owner: ObjectId(ownerId), repoName: name.toString() }; + const query = { owner: new ObjectId(ownerId), repoName: name.toString() }; const existingRepo = await collection.findOne(query); if (existingRepo !== null || !name) return 'Sie besitzen bereits ein Repository mit diesem Namen!';// existing or empty name - return collection.insertOne(emptyRepo).then((ret)=>ret.insertedId) + return collection.insertOne(emptyRepo).then((ret) => ret.insertedId); } catch (e) { console.log(`ERROR in createRepo${e}`); } @@ -780,7 +772,7 @@ async function createRepo(ownerId, name) { */ async function updateRepository(repoID, newName, user) { try { - const repoFilter = { owner: ObjectId(user), _id: ObjectId(repoID) }; + const repoFilter = { owner: new ObjectId(user), _id: new ObjectId(repoID) }; const db = dbConnection.getConnection(); const collection = await db.collection(repositoriesCollection); return collection.findOneAndUpdate(repoFilter, { $set: { repoName: newName } }, { returnNewDocument: true }); @@ -810,7 +802,7 @@ async function createGitRepo(gitOwnerId, repoName, userGithubId, userId) { newRepo = { owner: '', gitOwner: gitOwnerId, repoName, stories: [], repoType: 'github', customBlocks: [] }; - if (userGithubId === gitOwnerId) newRepo.owner = ObjectId(userId); + if (userGithubId === gitOwnerId) newRepo.owner = new ObjectId(userId); return await db.collection(repositoriesCollection).insertOne(newRepo); } catch (e) { console.log(`ERROR in createGitRepo${e}`); @@ -822,11 +814,11 @@ async function removeFromWorkgroup(repoId, user) { try { const db = dbConnection.getConnection(); const wGcollection = await db.collection(WorkgroupsCollection); - const repo = await db.collection(repositoriesCollection).findOne({ _id: ObjectId(repoId) }); + const repo = await db.collection(repositoriesCollection).findOne({ _id: new ObjectId(repoId) }); const owner = await db.collection(userCollection).findOne({ _id: repo.owner }); - const workGroup = await wGcollection.findOneAndUpdate({ Repo: ObjectId(repoId) }, { $pull: { Members: { email: user.email } } }); - if (workGroup.value) { - const wG = await wGcollection.findOne({ Repo: ObjectId(repoId) }); + const workGroup = await wGcollection.findOneAndUpdate({ Repo: new ObjectId(repoId) }, { $pull: { Members: { email: user.email } } }); + if (workGroup) { + const wG = await wGcollection.findOne({ Repo: new ObjectId(repoId) }); const result = { owner: {}, member: [] }; result.owner = { email: owner.email, canEdit: true }; result.member = wG.Members; @@ -834,7 +826,7 @@ async function removeFromWorkgroup(repoId, user) { } return; } catch (e) { - console.log(`ERROR in removeFromWorkgroup: ${e }`); + console.log(`ERROR in removeFromWorkgroup: ${e}`); throw e; } } @@ -845,14 +837,14 @@ async function updateOwnerInRepo(repoId, newOwnerId, oldOwnerId) { const oldOwner = await getUserById(oldOwnerId); // set new Owner for the given Repo const newOwner = await getUserById(newOwnerId); - await db.collection(repositoriesCollection).findOne({ _id: ObjectId(repoId) }); - await db.collection(repositoriesCollection).findOneAndUpdate({ _id: ObjectId(repoId) }, { $set: { owner: newOwnerId } }); + await db.collection(repositoriesCollection).findOne({ _id: new ObjectId(repoId) }); + await db.collection(repositoriesCollection).findOneAndUpdate({ _id: new ObjectId(repoId) }, { $set: { owner: newOwnerId } }); // remove the new Owner from Workgroup await removeFromWorkgroup(repoId, newOwner); // add old Owner as Member and update Email in Workgroup const wgMember = { email: oldOwner.email, canEdit: Boolean(true) }; - await db.collection(WorkgroupsCollection).findOneAndUpdate({ Repo: ObjectId(repoId) }, { $set: { owner: newOwner.email }, $push: { Members: wgMember } }); + await db.collection(WorkgroupsCollection).findOneAndUpdate({ Repo: new ObjectId(repoId) }, { $set: { owner: newOwner.email }, $push: { Members: wgMember } }); return 'Success'; } catch (e) { console.log(`ERROR in updateOwnerInRepo ${e}`); @@ -862,9 +854,9 @@ async function updateOwnerInRepo(repoId, newOwnerId, oldOwnerId) { async function updateStoriesArrayInRepo(repoId, storiesArray) { // TODO: vllt in updateStory reinnehmen dann spare ich den DBAufruf try { - const sortedStoriesArray = storiesArray.map((s) => ObjectId(s)); + const sortedStoriesArray = storiesArray.map((s) => new ObjectId(s)); const db = dbConnection.getConnection(); - return await db.collection(repositoriesCollection).findOneAndUpdate({ _id: ObjectId(repoId) }, { $set: { stories: sortedStoriesArray } }, { returnNewDocument: true }); + return await db.collection(repositoriesCollection).findOneAndUpdate({ _id: new ObjectId(repoId) }, { $set: { stories: sortedStoriesArray } }, { returnNewDocument: true }); } catch (e) { console.log(`ERROR in updateStoriesArrayInRepo${e}`); throw e; @@ -882,13 +874,13 @@ async function upsertEntry(storyId, updatedContent) { upsert: false }); // TODO remove later when all used stories have the tag storySource - if (!result.value) { + if (!result) { myObjt.storySource = undefined; result = await collection.findOneAndUpdate(myObjt, { $set: updatedContent }, { upsert: true }); } - return result.value; + return result; } catch (e) { console.log(`ERROR in upsertEntry: ${e}`); throw e; @@ -901,7 +893,7 @@ async function getTestReports(storyId) { const db = dbConnection.getConnection(); console.log('Getting Report for storyId :', storyId); result = await db.collection(ReportDataCollection).find( - { storyId: ObjectId(storyId) }, + { storyId: new ObjectId(storyId) }, { projection: { jsonReport: 0, reportOptions: 0, json: 0 } } ) .toArray(); @@ -917,7 +909,7 @@ async function getGroupTestReports(storyId) { const db = dbConnection.getConnection(); console.log('Getting Groups Reports for storyId :', storyId); // projection value 0 excludes from returning - const query = { storyStatuses: { $elemMatch: { storyId: ObjectId(storyId) } } }; + const query = { storyStatuses: { $elemMatch: { storyId: new ObjectId(storyId) } } }; const result = await db.collection(ReportDataCollection).find( query, { projection: { jsonReport: 0, reportOptions: 0, json: 0 } } @@ -937,18 +929,18 @@ async function deleteReport(reportId) { try { const db = dbConnection.getConnection(); const collection = await db.collection(ReportDataCollection); - const reportData = await collection.findOne({ _id: ObjectId(reportId) }); + const reportData = await collection.findOne({ _id: new ObjectId(reportId) }); if (reportData.smallReport) { idToDelete = reportData.smallReport; console.log('Trying to delete smallReport', idToDelete, ' in DB for Report', reportId); - await db.collection(ReportsCollection).deleteOne({ _id: ObjectId(idToDelete) }); - result = await collection.deleteOne({ _id: ObjectId(reportId) }); + await db.collection(ReportsCollection).deleteOne({ _id: new ObjectId(idToDelete) }); + result = await collection.deleteOne({ _id: new ObjectId(reportId) }); } else { idToDelete = reportData.bigReport; console.log('trying to delete bigReport', idToDelete, ' in DB for Report', reportId); const bucket = await new mongodb.GridFSBucket(db, { bucketName: 'GridFS' }); - bucket.delete(ObjectId(idToDelete)); - result = await collection.deleteOne({ _id: ObjectId(reportId) }); + bucket.delete(new ObjectId(idToDelete)); + result = await collection.deleteOne({ _id: new ObjectId(reportId) }); } } catch (e) { console.log('ERROR in deleteReport', e); @@ -959,7 +951,7 @@ async function deleteReport(reportId) { async function setIsSavedTestReport(testReportId, isSaved) { try { const db = dbConnection.getConnection(); - db.collection(ReportDataCollection).updateOne({ _id: ObjectId(testReportId) }, { + db.collection(ReportDataCollection).updateOne({ _id: new ObjectId(testReportId) }, { $set: { isSaved } }); } catch (e) { @@ -971,7 +963,7 @@ async function setIsSavedTestReport(testReportId, isSaved) { async function updateStoryStatus(storyId, storyLastTestStatus) { try { const db = dbConnection.getConnection(); - db.collection(storiesCollection).updateOne({ _id: ObjectId(storyId) }, { + db.collection(storiesCollection).updateOne({ _id: new ObjectId(storyId) }, { $set: { lastTestPassed: storyLastTestStatus } }); return 'done'; @@ -984,16 +976,15 @@ async function updateStoryStatus(storyId, storyLastTestStatus) { async function updateScenarioStatus(storyId, scenarioId, scenarioLastTestStatus) { // TODO: testen try { const db = dbConnection.getConnection(); - return await db.collection(storiesCollection).updateOne( - { - _id: ObjectId(storyId), - scenarios: { - $elemMatch: - { scenario_id: scenarioId } - } - }, { - $set: { 'scenarios.$.lastTestPassed': scenarioLastTestStatus } - }); + return await db.collection(storiesCollection).updateOne({ + _id: new ObjectId(storyId), + scenarios: { + $elemMatch: + { scenario_id: scenarioId } + } + }, { + $set: { 'scenarios.$.lastTestPassed': scenarioLastTestStatus } + }); } catch (e) { console.log('Error in updateScenarioStatus. Could not set scenario LastTestPassed: ', e); } @@ -1002,7 +993,7 @@ async function updateScenarioStatus(storyId, scenarioId, scenarioLastTestStatus) async function uploadBigJsonData(data, fileName) { const db = dbConnection.getConnection(); const bucket = await new mongodb.GridFSBucket(db, { bucketName: 'GridFS' }); - const id = ObjectId(); + const id = new ObjectId(); str(JSON.stringify(data)) .pipe(bucket.openUploadStreamWithId(id, fileName)) .on('error', async (error) => { @@ -1022,7 +1013,7 @@ async function uploadReport(reportResults) { const collection = await db.collection(ReportDataCollection); fs.readFile(reportResults.reportOptions.jsonFile, 'utf8', async (err, data) => { if (err) console.log(err); - const jReport = { jsonReport: data, created: new Date()}; + const jReport = { jsonReport: data, created: new Date() }; const len = Buffer.byteLength(JSON.stringify(data)); if (len >= 16000000) { try { @@ -1058,7 +1049,7 @@ async function getReportFromDB(report) { }; } else { const bucket = await new mongodb.GridFSBucket(db, { bucketName: 'GridFS' }); - const reportString = await toString(bucket.openDownloadStream(ObjectId(report.bigReport.toString()))); + const reportString = await toString(bucket.openDownloadStream(new ObjectId(report.bigReport.toString()))); const reportJson = JSON.parse(reportString); result = { _id: report._id, @@ -1086,7 +1077,7 @@ async function getReportByName(reportName) { async function getReportById(reportId) { try { const db = dbConnection.getConnection(); - const report = await db.collection(ReportDataCollection).findOne({ _id: ObjectId(reportId.toString()) }); + const report = await db.collection(ReportDataCollection).findOne({ _id: new ObjectId(reportId.toString()) }); return await getReportFromDB(report); } catch (e) { console.log('ERROR in getReportById (DBServices)', e); @@ -1098,7 +1089,7 @@ async function getReportDataById(reportId) { try { const db = dbConnection.getConnection(); return await db.collection(ReportDataCollection) - .findOne({ _id: ObjectId(reportId.toString()) }); + .findOne({ _id: new ObjectId(reportId.toString()) }); } catch (e) { console.log('ERROR in getReportDataById (DBServices)', e); return {}; @@ -1107,17 +1098,18 @@ async function getReportDataById(reportId) { // delete User in DB needs ID async function deleteUser(userID) { - try {// delete user from Workgroup - const oId = ObjectId(userID); + try { // delete user from Workgroup + const oId = new ObjectId(userID); const myObjt = { _id: oId }; const db = dbConnection.getConnection(); - const repos = await db.collection(repositoriesCollection).find({ owner: oId }).toArray(); + const repos = await db.collection(repositoriesCollection).find({ owner: oId }) + .toArray(); if (repos) { - for (const repo of repos) for (const storyID of repo.stories) await db.collection(storiesCollection).deleteOne({ _id: ObjectId(storyID) }); // use delete repo? + for (const repo of repos) for (const storyID of repo.stories) await db.collection(storiesCollection).deleteOne({ _id: new ObjectId(storyID) }); // use delete repo? const resultRepo = await db.collection(repositoriesCollection).deleteMany({ owner: oId }); const resultUser = await db.collection(userCollection).deleteOne(myObjt); - return {resultUser, resultRepo}; + return { resultUser, resultRepo }; } return null; } catch (e) { @@ -1129,11 +1121,11 @@ async function deleteUser(userID) { // update a User in DB needs ID and JsonObject User returns altered JsonObject User async function updateUser(userID, updatedUser) { try { - const oId = ObjectId(userID); + const oId = new ObjectId(userID); const myObjt = { _id: oId }; const db = dbConnection.getConnection(); const result = await db.collection(userCollection).findOneAndReplace(myObjt, updatedUser); - return result.value; + return result; } catch (e) { console.log(`ERROR in updateUser: ${e}`); throw e; @@ -1143,7 +1135,7 @@ async function updateUser(userID, updatedUser) { // get UserData needs ID returns JsonObject User async function getUserData(userID) { try { - const oId = ObjectId(userID); + const oId = new ObjectId(userID); const myObjt = { _id: oId }; const db = dbConnection.getConnection(); return await db.collection(userCollection).findOne(myObjt); @@ -1156,8 +1148,8 @@ async function getUserData(userID) { async function saveBlock(block) { try { block = mongoSanitize(block); - block.repositoryId = ObjectId(block.repositoryId) - block.owner = ObjectId(block.owner.toString()) + block.repositoryId = new ObjectId(block.repositoryId); + block.owner = new ObjectId(block.owner.toString()); const db = dbConnection.getConnection(); return await db.collection(CustomBlocksCollection).insertOne(block); } catch (e) { @@ -1168,14 +1160,14 @@ async function saveBlock(block) { async function updateBlock(blockId, updatedBlock) { try { - updatedBlock._id = ObjectId(updatedBlock._id); - updatedBlock.repositoryId = ObjectId(updatedBlock.repositoryId); - updatedBlock.owner = ObjectId(updatedBlock.owner); + updatedBlock._id = new ObjectId(updatedBlock._id); + updatedBlock.repositoryId = new ObjectId(updatedBlock.repositoryId); + updatedBlock.owner = new ObjectId(updatedBlock.owner); const db = dbConnection.getConnection(); - updatedBlock._id = ObjectId(updatedBlock._id) - updatedBlock.repositoryId = ObjectId(updatedBlock.repositoryId); - updatedBlock.owner = ObjectId(updatedBlock.owner); - await db.collection(CustomBlocksCollection).findOneAndReplace({_id: ObjectId(blockId)}, updatedBlock); + updatedBlock._id = new ObjectId(updatedBlock._id); + updatedBlock.repositoryId = new ObjectId(updatedBlock.repositoryId); + updatedBlock.owner = new ObjectId(updatedBlock.owner); + await db.collection(CustomBlocksCollection).findOneAndReplace({ _id: new ObjectId(blockId) }, updatedBlock); } catch (e) { console.log(`ERROR in updateBlock: ${e}`); throw e; @@ -1187,7 +1179,7 @@ async function getBlock(blockId) { try { const db = dbConnection.getConnection(); return await db.collection(CustomBlocksCollection) - .findOne({ _id: ObjectId(blockId) }); + .findOne({ _id: new ObjectId(blockId) }); } catch (e) { console.log(`ERROR in getBlock: ${e}`); throw e; @@ -1198,8 +1190,8 @@ async function getBlock(blockId) { async function getBlocks(repoId) { try { const db = dbConnection.getConnection(); - return await db.collection(CustomBlocksCollection).find({ repositoryId: ObjectId(repoId) }) - .toArray(); + return await db.collection(CustomBlocksCollection).find({ repositoryId: new ObjectId(repoId) }) + .toArray(); } catch (e) { console.log(`ERROR in getBlocks: ${e}`); throw e; @@ -1209,8 +1201,8 @@ async function getBlocks(repoId) { async function deleteBlock(blockId, userId) { try { const myObjt = { - _id: ObjectId(blockId), - owner: ObjectId(userId) + _id: new ObjectId(blockId), + owner: new ObjectId(userId) }; const db = dbConnection.getConnection(); await db.collection(CustomBlocksCollection).deleteOne(myObjt); @@ -1224,7 +1216,7 @@ async function deleteBlock(blockId, userId) { async function getWorkgroup(id) { try { const db = dbConnection.getConnection(); - return await db.collection(WorkgroupsCollection).findOne({ Repo: ObjectId(id) }); + return await db.collection(WorkgroupsCollection).findOne({ Repo: new ObjectId(id) }); } catch (e) { console.log(`ERROR in getWorkgroup: ${e}`); throw e; @@ -1232,29 +1224,29 @@ async function getWorkgroup(id) { } /** - * + * * @param {*} repoId Repository Id * @param {object} user User object {email:string, canEdit:boolean} - * @returns + * @returns */ async function addMember(repoId, user) { try { const db = dbConnection.getConnection(); const wGCollection = await db.collection(WorkgroupsCollection); - const check = await wGCollection.findOne({ Repo: ObjectId(repoId), Members: { $elemMatch: { email: user.email } } }); + const check = await wGCollection.findOne({ Repo: new ObjectId(repoId), Members: { $elemMatch: { email: user.email } } }); if (check) return 'Dieser User ist bereits in der Workgroup'; - const repo = await db.collection(repositoriesCollection).findOne({ _id: ObjectId(repoId) }); + const repo = await db.collection(repositoriesCollection).findOne({ _id: new ObjectId(repoId) }); const owner = await db.collection(userCollection).findOne({ _id: repo.owner }); - const workGroup = await wGCollection.findOne({ Repo: ObjectId(repoId) }); - if (!workGroup) { + const workGroup = await wGCollection.findOne({ Repo: new ObjectId(repoId) }); + if (!workGroup) { await wGCollection.insertOne({ - name: repo.repoName, owner: owner.email, Repo: ObjectId(repoId), Members: [{ email: user.email, canEdit: Boolean(user.canEdit) }] + name: repo.repoName, owner: owner.email, Repo: new ObjectId(repoId), Members: [{ email: user.email, canEdit: Boolean(user.canEdit) }] }); } else { - await wGCollection.findOneAndUpdate({ Repo: ObjectId(repoId) }, { $push: { Members: user } }); + await wGCollection.findOneAndUpdate({ Repo: new ObjectId(repoId) }, { $push: { Members: user } }); } const result = { owner: {}, member: [] }; - const wG = await wGCollection.findOne({ Repo: ObjectId(repoId) }); + const wG = await wGCollection.findOne({ Repo: new ObjectId(repoId) }); result.owner = { email: owner.email, canEdit: true }; result.member = wG.Members; return result; @@ -1268,12 +1260,12 @@ async function updateMemberStatus(repoId, user) { try { const db = dbConnection.getConnection(); const wGCollection = await db.collection(WorkgroupsCollection); - const repo = await db.collection(repositoriesCollection).findOne({ _id: ObjectId(repoId) }); + const repo = await db.collection(repositoriesCollection).findOne({ _id: new ObjectId(repoId) }); const usersCollection = await db.collection(userCollection); const owner = await usersCollection.findOne({ _id: repo.owner }); - const updatedWG = await wGCollection.findOneAndUpdate({ Repo: ObjectId(repoId) }, { $set: { 'Members.$[elem].canEdit': Boolean(user.canEdit) } }, { arrayFilters: [{ 'elem.email': user.email }] }); + const updatedWG = await wGCollection.findOneAndUpdate({ Repo: new ObjectId(repoId) }, { $set: { 'Members.$[elem].canEdit': Boolean(user.canEdit) } }, { arrayFilters: [{ 'elem.email': user.email }] }); if (updatedWG) { - const wG = await wGCollection.findOne({ Repo: ObjectId(repoId) }); + const wG = await wGCollection.findOne({ Repo: new ObjectId(repoId) }); const result = { owner: {}, member: [] }; result.owner = { email: owner.email, canEdit: true }; result.member = wG.Members; @@ -1288,9 +1280,9 @@ async function updateMemberStatus(repoId, user) { async function getMembers(id) { try { const db = dbConnection.getConnection(); - const repo = await db.collection(repositoriesCollection).findOne({ _id: ObjectId(id) }); + const repo = await db.collection(repositoriesCollection).findOne({ _id: new ObjectId(id) }); const owner = await db.collection(userCollection).findOne({ _id: repo.owner }); - const wG = await db.collection(WorkgroupsCollection).findOne({ Repo: ObjectId(id) }); + const wG = await db.collection(WorkgroupsCollection).findOne({ Repo: new ObjectId(id) }); if (!wG) return { owner: { email: owner.email, canEdit: true }, member: [] }; const result = { owner: {}, member: [] }; result.owner = { email: owner.email, canEdit: true }; @@ -1307,10 +1299,10 @@ async function updateOneDriver(id, driver) { const oneDriver = !driver.oneDriver; const db = dbConnection.getConnection(); const result = await db.collection(storiesCollection).findOneAndUpdate( - { _id: ObjectId(id) }, + { _id: new ObjectId(id) }, { $set: { oneDriver } } ); - return result.value; + return result; } catch (e) { console.log('ERROR in updateOneDriver: ', e); } @@ -1318,7 +1310,7 @@ async function updateOneDriver(id, driver) { function mongoSanitize(v) { // from https://github.com/vkarpov15/mongo-sanitize if (v instanceof Object) { - for (var key in v) { + for (const key in v) { if (/^\$/.test(key)) { delete v[key]; } else { @@ -1327,7 +1319,7 @@ function mongoSanitize(v) { // from https://github.com/vkarpov15/mongo-sanitize } } return v; -}; +} module.exports = { setIsSavedTestReport, @@ -1402,4 +1394,4 @@ module.exports = { updateOwnerInRepo, updateRepository, getOneRepositoryById -}; \ No newline at end of file +}; diff --git a/backend/src/database/installDatabase.js b/backend/src/database/installDatabase.js index 26072b963..378eb316b 100644 --- a/backend/src/database/installDatabase.js +++ b/backend/src/database/installDatabase.js @@ -5,17 +5,17 @@ require('dotenv').config(); const uri = process.env.DATABASE_URI || 'mongodb://SeedAdmin:SeedTest@localhost:27017'; -async function checkConnection() { - let fails = 1; - while (fails <= 3) try { - const client = await MongoClient.connect(uri, { poolSize: 20, useNewUrlParser: true, useUnifiedTopology: true }); +async function getConnection(attempt) { + const attempts = attempt || 1; + if (attempt > 3) throw new Error('\x1b[31mFailed to connect to the database after multiple retries.\x1b[0m'); + + try { + const client = await MongoClient.connect(uri, { maxPoolSize: 20 }); return client; } catch (err) { - console.log(`Connection failed! Retrying... ${fails}`); - fails++; + console.log(`\x1b[38;5;208mConnection failed! Retrying... ${attempts}\x1b[0m`); + return getConnection(attempts + 1); } - - throw new Error('Failed to connect to the database after multiple retries.'); } async function makeCollection(dbo, name) { @@ -41,7 +41,7 @@ async function makeCollection(dbo, name) { async function installDatabase() { console.log(`\x1b[33m Setting Up DB in: ${uri}\n\x1b[0m`); - const client = await checkConnection(); + const client = await getConnection(); const dbo = client.db('Seed'); console.log('Starting: steps'); diff --git a/backend/src/database/mongoDB_admin.js b/backend/src/database/mongoDB_admin.js index 33fdef021..3244a84ee 100644 --- a/backend/src/database/mongoDB_admin.js +++ b/backend/src/database/mongoDB_admin.js @@ -7,7 +7,7 @@ const mongo = require('./DbServices'); const stepTypes = require('./stepTypes'); require('dotenv').config(); -const uri = process.env.DATABASE_URI || "mongodb://SeedAdmin:SeedTest@seedmongodb:27017"; +const uri = process.env.DATABASE_URI || 'mongodb://SeedAdmin:SeedTest@seedmongodb:27017'; const dbName = 'Seed'; // /////////////////////////////////////////// ADMIN METHODS //////////////////////////////////////////// @@ -83,7 +83,7 @@ async function createContent() { // show all Collections function getCollections() { - MongoClient.connect(uri, { useNewUrlParser: true }, (err, db) => { + MongoClient.connect(uri, (err, db) => { if (err) throw err; const dbo = db.db(dbName); dbo.listCollections().toArray((error, result) => { @@ -94,18 +94,10 @@ function getCollections() { }); } -async function installDatabase() { - console.log (`Setting Up DB in: ${uri}`); - await makeCollection('stepTypes'); - await makeCollection('Stories'); - await makeCollection('User'); - await insertMore('stepTypes', stepTypes()); -} - // create Collection async function makeCollection(name) { let connection = []; - await MongoClient.connect(uri, { poolSize: 20, useNewUrlParser: true, useUnifiedTopology: true }, async (err, dbo) => { + await MongoClient.connect(uri, { maxPoolSize: 20 }, async (err, dbo) => { if (err) throw err; connection = dbo.db('Seed'); }); @@ -120,7 +112,7 @@ async function makeCollection(name) { // insert One document (collectionname, {document}) function insertOne(collection, content) { - MongoClient.connect(uri, { useNewUrlParser: true }, (err, db) => { + MongoClient.connect(uri, (err, db) => { if (err) throw err; const dbo = db.db(dbName); dbo.collection(collection).insertOne(content, (error) => { @@ -156,7 +148,7 @@ async function backupScenarios() { // insert Many documents ("collectionname", [{documents},{...}] ) function insertMore(name, content) { - MongoClient.connect(uri, { useNewUrlParser: true }, (err, db) => { + MongoClient.connect(uri, (err, db) => { if (err) throw err; const dbo = db.db(dbName); dbo.collection(name).insertMany(content, (error, res) => { @@ -168,7 +160,7 @@ function insertMore(name, content) { } function update(story_id, updatedStuff) { - MongoClient.connect(uri, { useNewUrlParser: true }, (err, db) => { + MongoClient.connect(uri, (err, db) => { if (err) throw err; const dbo = db.db(dbName); dbo.collection(collection).updateOne({ story_id }, { $set: updatedStuff }, (error, res) => { @@ -180,7 +172,7 @@ function update(story_id, updatedStuff) { // doesnt work yet function eraseAllStories() { - MongoClient.connect(uri, { useNewUrlParser: true }, (err, db) => { + MongoClient.connect(uri, (err, db) => { if (err) throw err; const dbo = db.db(dbName); dbo.collection(collection).deleteOne({}, (error) => { @@ -192,7 +184,7 @@ function eraseAllStories() { // shows single story function showStory(story_id) { - MongoClient.connect(uri, { useNewUrlParser: true }, (err, db) => { + MongoClient.connect(uri, (err, db) => { if (err) throw err; const dbo = db.db(dbName); const myObjt = { story_id }; @@ -206,7 +198,7 @@ function showStory(story_id) { // delete collection function dropCollection() { - MongoClient.connect(uri, { useNewUrlParser: true }, (err, db) => { + MongoClient.connect(uri, (err, db) => { if (err) throw err; const dbo = db.db(dbName); dbo.collection(collection).drop((error, delOK) => { @@ -217,9 +209,8 @@ function dropCollection() { }); } - function deleteOldReports() { - MongoClient.connect(uri, { useNewUrlParser: true }, (err, db) => { + MongoClient.connect(uri, (err, db) => { if (err) throw err; const dbo = db.db(dbName); dbo.collection('TestReport').deleteMany({ reportTime: { $lt: 1622505600000 } }); @@ -229,15 +220,15 @@ function deleteOldReports() { } function fixOldReports() { - MongoClient.connect(uri, { useNewUrlParser: true }, (err, db) => { + MongoClient.connect(uri, (err, db) => { if (err) throw err; const dbo = db.db(dbName); dbo.collection('TestReport') // use updateMany for all reports .updateOne({}, { $rename: { - 'testStatus': 'overallTestStatus', - 'jsonReport': 'json' + testStatus: 'overallTestStatus', + jsonReport: 'json' } }); console.log('Updated Something'); @@ -245,6 +236,14 @@ function fixOldReports() { }); } +async function installDatabase() { + console.log(`Setting Up DB in: ${uri}`); + await makeCollection('stepTypes'); + await makeCollection('Stories'); + await makeCollection('User'); + await insertMore('stepTypes', stepTypes()); +} + module.exports = { installDatabase }; diff --git a/backend/src/database/mongodatabase.js b/backend/src/database/mongodatabase.js index 0f0d4e5f5..95e2ee87c 100644 --- a/backend/src/database/mongodatabase.js +++ b/backend/src/database/mongodatabase.js @@ -16,7 +16,7 @@ if (!process.env.NODE_ENV) { const dotenv = require('dotenv').config(); } -const uri = process.env.DATABASE_URI || "mongodb://SeedAdmin:SeedTest@seedmongodb:27017"; +const uri = process.env.DATABASE_URI || 'mongodb://SeedAdmin:SeedTest@seedmongodb:27017'; const dbName = 'Seed'; const userCollection = 'User'; const storiesCollection = 'Stories'; @@ -40,7 +40,7 @@ const ReportsCollection = 'Reports'; function connectDb() { return new Promise((resolve, reject) => { mongodb.MongoClient - .connect(uri, { useNewUrlParser: true, useUnifiedTopology: true }, (err, db) => { + .connect(uri, (err, db) => { if (err) reject(err); else resolve(db); }); @@ -118,11 +118,11 @@ async function registerUser(user) { let result; if (dbUser !== null) throw Error('User already exists'); else - if (user.userId) result = await collection.update({ _id: ObjectId(user.userId) }, { $set: { email: user.email, password: user.password } }); - else { - delete user.userId; - result = await collection.insertOne(user); - } + if (user.userId) result = await collection.updateOne({ _id: new ObjectId(user.userId) }, { $set: { email: user.email, password: user.password } }); + else { + delete user.userId; + result = await collection.insertOne(user); + } if (db) db.close(); console.log('I am closing the DB here - registerUser'); return result; @@ -199,7 +199,7 @@ async function getUserById(id) { const db = await connectDb(); const dbo = await db.db(dbName); const collection = await dbo.collection(userCollection); - const result = await collection.findOne({ _id: ObjectId(id) }); + const result = await collection.findOne({ _id: new ObjectId(id) }); db.close(); console.log('I am closing the DB here - getUserById'); return result; @@ -216,7 +216,7 @@ async function updateGithubToken(objId, updatedToken) { const db = await connectDb(); const dbo = await db.db(dbName); const collection = await dbo.collection(userCollection); - const user = await collection.updateOne({ _id: ObjectId(objId) }, { $set: { 'github.githubToken': updatedToken } }); + const user = await collection.updateOne({ _id: new ObjectId(objId) }, { $set: { 'github.githubToken': updatedToken } }); db.close(); console.log('I am closing the DB here - updateGithubToken'); return user; @@ -251,7 +251,7 @@ function selectUsersCollection(db) { } function findStory(storyId, collection) { - const id = ObjectId(storyId); + const id = new ObjectId(storyId); return new Promise((resolve, reject) => { collection.findOne({ _id: id }, (err, result) => { if (err) reject(err); @@ -262,16 +262,11 @@ function findStory(storyId, collection) { function replace(story, collection) { const filter = { - _id: ObjectId(story._id), + _id: new ObjectId(story._id), storySource: story.storySource }; - story._id = ObjectId(story._id); - return new Promise((resolve, reject) => { - collection.findOneAndReplace(filter, story, (err, result) => { - if (err) reject(err); - else resolve(result.value); - }); - }); + story._id = new ObjectId(story._id); + return collection.findOneAndReplace(filter, story); } async function disconnectGithub(user) { @@ -282,13 +277,8 @@ async function disconnectGithub(user) { } function replaceUser(newUser, collection) { - const myObjt = { _id: ObjectId(newUser._id) }; - return new Promise((resolve, reject) => { - collection.findOneAndReplace(myObjt, newUser, (err, result) => { - if (err) reject(err); - else resolve(result.value); - }); - }); + const myObjt = { _id: new ObjectId(newUser._id) }; + return collection.findOneAndReplace(myObjt, newUser); } async function updateStory(updatedStuff) { @@ -312,9 +302,9 @@ async function getOneStory(storyId) { try { db = await connectDb(); const collection = await selectStoriesCollection(db); - let story = await collection.findOne({ _id: ObjectId(storyId) }); + let story = await collection.findOne({ _id: new ObjectId(storyId) }); // TODO remove later when all used stories have the tag storySource - if (!story) story = await collection.findOne({ _id: ObjectId(storyId) }); + if (!story) story = await collection.findOne({ _id: new ObjectId(storyId) }); return story; } catch (e) { console.log(`UPS!!!! FEHLER in getOneStory: ${e}`); @@ -349,17 +339,17 @@ async function createStoryGroup(repoID, name, members, sequence) { const collection = await selectRepositoryCollection(db); const groups = await collection.findOneAndUpdate( - { _id: ObjectId(repoID) }, + { _id: new ObjectId(repoID) }, { $push: { groups: { - _id: ObjectId(), name, member_stories: members, isSequential: sequence + _id: new ObjectId(), name, member_stories: members, isSequential: sequence } } }, { upsert: true, projection: { groups: 1 } } ); - return groups.value.groups.slice(-1)._id; + return groups.groups.slice(-1)._id; } catch (e) { console.log(`UPS!!!! FEHLER in createStoryGroup: ${e}`); } finally { @@ -372,13 +362,13 @@ async function updateStoryGroup(repoId, groupId, updatedGroup) { let db; try { db = await connectDb(); - updatedGroup._id = ObjectId(updatedGroup._id); + updatedGroup._id = new ObjectId(updatedGroup._id); const collection = await selectRepositoryCollection(db); - const repo = await collection.findOne({ _id: ObjectId(repoId) }); + const repo = await collection.findOne({ _id: new ObjectId(repoId) }); // leave with double equal: const index = repo.groups.findIndex((o) => o._id == groupId); repo.groups[index] = updatedGroup; - await collection.updateOne({ _id: ObjectId(repoId) }, { $set: repo }); + await collection.updateOne({ _id: new ObjectId(repoId) }, { $set: repo }); return updatedGroup; } catch (e) { console.log(`UPS!!!! FEHLER in updateStoryGroup: ${e}`); @@ -393,11 +383,11 @@ async function deleteStoryGroup(repoId, groupId) { try { db = await connectDb(); const collection = await selectRepositoryCollection(db); - const repo = await collection.findOne({ _id: ObjectId(repoId) }); + const repo = await collection.findOne({ _id: new ObjectId(repoId) }); // leave with double equal: const index = repo.groups.findIndex((o) => o._id == groupId); repo.groups.splice(index, 1); - await collection.updateOne({ _id: ObjectId(repoId) }, { $set: repo }); + await collection.updateOne({ _id: new ObjectId(repoId) }, { $set: repo }); return null; } catch (e) { console.log(`UPS!!!! FEHLER in deleteStoryGroup: ${e}`); @@ -434,7 +424,7 @@ async function getAllStoryGroups(repoId) { try { db = await connectDb(); const collection = await selectRepositoryCollection(db); - return await collection.findOne({ _id: ObjectId(repoId) }, { projection: { groups: 1 } }); + return await collection.findOne({ _id: new ObjectId(repoId) }, { projection: { groups: 1 } }); } catch (e) { console.log(`UPS!!!! FEHLER in getAllStoryGroups: ${e}`); } finally { @@ -448,7 +438,7 @@ async function updateStoryGroupsArray(repoId, groupsArray) { try { db = await connectDb(); const collection = await selectRepositoryCollection(db); - return await collection.findOneAndUpdate({ _id: ObjectId(repoId) }, { $set: { groups: groupsArray } }, { projection: { groups: 1 } }); + return await collection.findOneAndUpdate({ _id: new ObjectId(repoId) }, { $set: { groups: groupsArray } }, { projection: { groups: 1 } }); } catch (e) { console.log(`UPS!!!! FEHLER in updateStoryGroupsArray: ${e}`); } finally { @@ -527,10 +517,10 @@ async function createStory(storyTitel, storyDescription, repoId) { db = await connectDb(); const repoCollection = await selectRepositoryCollection(db); const collection = await selectStoriesCollection(db); - const repo = await repoCollection.findOne({ _id: ObjectId(repoId) }); + const repo = await repoCollection.findOne({ _id: new ObjectId(repoId) }); if (repo) if (repo.stories.length > 0) { for (const storyId of repo.stories) { - const story = await collection.findOne({ _id: ObjectId(storyId) }); + const story = await collection.findOne({ _id: new ObjectId(storyId) }); iNumberArray.push(story.issue_number); } for (let i = 0; i <= iNumberArray.length; i++) { @@ -563,12 +553,12 @@ async function deleteStory(repoId, storyId) { db = await connectDb(); const collection = await selectStoriesCollection(db); const repo = await selectRepositoryCollection(db); - const delStory = await collection.findOneAndDelete({ _id: ObjectId(storyId) }); - await repo.findOneAndUpdate({ _id: ObjectId(repoId) }, { $pull: { stories: ObjectId(storyId) } }); + const delStory = await collection.findOneAndDelete({ _id: new ObjectId(storyId) }); + await repo.findOneAndUpdate({ _id: new ObjectId(repoId) }, { $pull: { stories: new ObjectId(storyId) } }); - const groups = await repo.findOne({ _id: ObjectId(repoId) }, { projection: { groups: 1 } }); + const groups = await repo.findOne({ _id: new ObjectId(repoId) }, { projection: { groups: 1 } }); for (const index in groups.groups) groups.groups[index].member_stories = groups.groups[index].member_stories.filter((story) => story !== storyId); - await repo.findOneAndUpdate({ _id: ObjectId(repoId) }, { $set: { groups: groups.groups } }); + await repo.findOneAndUpdate({ _id: new ObjectId(repoId) }, { $set: { groups: groups.groups } }); return delStory; } catch (e) { console.log(`UPS!!!! FEHLER in deleteStory: ${e}`); @@ -584,7 +574,7 @@ async function insertStoryIdIntoRepo(storyId, repoId) { try { db = await connectDb(); const collectionRepo = await selectRepositoryCollection(db); - return await collectionRepo.findOneAndUpdate({ _id: ObjectId(repoId) }, { $push: { stories: ObjectId(storyId) } }); + return await collectionRepo.findOneAndUpdate({ _id: new ObjectId(repoId) }, { $push: { stories: new ObjectId(storyId) } }); } catch (e) { console.log(`UPS!!!! FEHLER in insertStoryIdIntoRepo: ${e}`); throw e; @@ -599,7 +589,7 @@ async function updateScenarioList(storyId, scenarioList) { try { db = await connectDb(); const collection = await selectStoriesCollection(db); - return await collection.findOneAndUpdate({ _id: ObjectId(storyId) }, { $set: { scenarios: scenarioList } }); + return await collection.findOneAndUpdate({ _id: new ObjectId(storyId) }, { $set: { scenarios: scenarioList } }); } catch (e) { console.log(`UPS!!!! FEHLER in insertStoryIdIntoRepo: ${e}`); throw e; @@ -616,9 +606,9 @@ async function getAllStoriesOfRepo(ownerId, repoName, repoId) { db = await connectDb(); const collectionRepo = await selectRepositoryCollection(db); const collectionStories = await selectStoriesCollection(db); - const repo = await collectionRepo.findOne({ _id: ObjectId(repoId) }); + const repo = await collectionRepo.findOne({ _id: new ObjectId(repoId) }); if (repo) for (const entry of repo.stories) { - const story = await collectionStories.findOne({ _id: ObjectId(entry) }); + const story = await collectionStories.findOne({ _id: new ObjectId(entry) }); storiesArray.push(story); } return storiesArray; @@ -637,7 +627,7 @@ async function getOneScenario(storyId, storySource, scenarioId) { try { db = await connectDb(); const collection = await selectStoriesCollection(db); - const scenarios = await collection.findOne({ _id: ObjectId(storyId), storySource, 'scenarios.scenario_id': scenarioId }, { projection: { scenarios: 1 } }); + const scenarios = await collection.findOne({ _id: new ObjectId(storyId), storySource, 'scenarios.scenario_id': scenarioId }, { projection: { scenarios: 1 } }); return scenarios.scenarios.find((o) => o.scenario_id === scenarioId); } catch (e) { console.log(`UPS!!!! FEHLER in getOneScenario: ${e}`); @@ -731,21 +721,21 @@ async function deleteScenario(storyId, scenarioId) { async function getRepository(userID) { let db; try { - const myObjt = { owner: ObjectId(userID) }; + const myObjt = { owner: new ObjectId(userID) }; db = await connectDb(); const dbo = db.db(dbName); const wGCollection = await dbo.collection(WorkgroupsCollection); const repoCollection = await selectRepositoryCollection(db); const usersCollection = await selectUsersCollection(db); - const user = await usersCollection.findOne({ _id: ObjectId(userID) }); + const user = await usersCollection.findOne({ _id: new ObjectId(userID) }); const positiveWorkgroups = await wGCollection.find({ Members: { $elemMatch: { email: user.email, canEdit: true } } }).toArray(); - const PWgArray = positiveWorkgroups.map((entry) => ObjectId(entry.Repo)); + const PWgArray = positiveWorkgroups.map((entry) => new ObjectId(entry.Repo)); const PWgRepos = await repoCollection.find({ _id: { $in: PWgArray } }).toArray(); PWgRepos.forEach((element) => { element.canEdit = true; }); const negativeWorkgroups = await wGCollection.find({ Members: { $elemMatch: { email: user.email, canEdit: false } } }).toArray(); - const NWgArray = negativeWorkgroups.map((entry) => ObjectId(entry.Repo)); + const NWgArray = negativeWorkgroups.map((entry) => new ObjectId(entry.Repo)); const NWgRepos = await repoCollection.find({ _id: { $in: NWgArray } }).toArray(); NWgRepos.forEach((element) => { element.canEdit = false; @@ -769,7 +759,7 @@ async function getRepository(userID) { async function deleteRepositorys(ownerID) { let db; try { - const myObjt = { owner: ObjectId(ownerID) }; + const myObjt = { owner: new ObjectId(ownerID) }; db = await connectDb(); const collection = await selectRepositoryCollection(db); return await collection.deleteMany(myObjt); @@ -787,7 +777,7 @@ async function deleteRepository(repoId, ownerId) { try { db = await connectDb(); const collectionRepo = await selectRepositoryCollection(db); - const repo = await collectionRepo.findOne({ owner: ObjectId(ownerId), _id: ObjectId(repoId) }); + const repo = await collectionRepo.findOne({ owner: new ObjectId(ownerId), _id: new ObjectId(repoId) }); return await collectionRepo.deleteOne(repo); } catch (e) { console.log(`UPS!!!! FEHLER in deleteRepository${e}`); @@ -800,7 +790,7 @@ async function deleteRepository(repoId, ownerId) { async function getOneRepository(ownerId, name) { try { - const repo = { owner: ObjectId(ownerId), repoName: name }; + const repo = { owner: new ObjectId(ownerId), repoName: name }; const db = await connectDb(); const collection = await selectRepositoryCollection(db); const result = await collection.findOne(repo); @@ -832,7 +822,7 @@ async function createRepo(ownerId, name) { }; const db = await connectDb(); const collection = await selectRepositoryCollection(db); - const result = await collection.findOne({ owner: ObjectId(ownerId), repoName: name }); + const result = await collection.findOne({ owner: new ObjectId(ownerId), repoName: name }); if (result !== null) return 'Sie besitzen bereits ein Repository mit diesem Namen!'; collection.insertOne(emptyRepo); } @@ -866,7 +856,7 @@ async function createGitOwnerRepoIfNoneExists(ownerId, githubId, gitOwnerId, rep try { db = await connectDb(); const collection = await selectRepositoryCollection(db); - const result = await collection.findOne({ owner: ObjectId(ownerId), repoName }); + const result = await collection.findOne({ owner: new ObjectId(ownerId), repoName }); if (result === null) { let repo = await collection.findOne({ gitOwner: gitOwnerId, repoName }); // create repo / project if there is none @@ -877,7 +867,7 @@ async function createGitOwnerRepoIfNoneExists(ownerId, githubId, gitOwnerId, rep repo = await collection.insertOne(newRepo); return repo; } - if (repo.gitOwner === githubId) repo.owner = ObjectId(ownerId); + if (repo.gitOwner === githubId) repo.owner = new ObjectId(ownerId); return repo; } return result._id; @@ -893,10 +883,10 @@ async function createGitOwnerRepoIfNoneExists(ownerId, githubId, gitOwnerId, rep async function updateStoriesArrayInRepo(repoId, storiesArray) { let db; try { - const sortedStoriesArray = storiesArray.map((s) => ObjectId(s)); + const sortedStoriesArray = storiesArray.map((s) => new ObjectId(s)); db = await connectDb(); const collection = await selectRepositoryCollection(db); - return await collection.findOneAndUpdate({ _id: ObjectId(repoId) }, { $set: { stories: sortedStoriesArray } }, { returnNewDocument: true }); + return await collection.findOneAndUpdate({ _id: new ObjectId(repoId) }, { $set: { stories: sortedStoriesArray } }, { returnNewDocument: true }); } catch (e) { console.log(`UPS!!!! FEHLER in updateStoriesArrayInRepo${e}`); throw e; @@ -919,13 +909,13 @@ async function upsertEntry(storyId, updatedContent, storySource) { upsert: false }); // TODO remove later when all used stories have the tag storySource - if (!result.value) { + if (!result) { myObjt.storySource = undefined; result = await collection.findOneAndUpdate(myObjt, { $set: updatedContent }, { upsert: true }); } - return result.value; + return result; } catch (e) { console.log(`UPS!!!! FEHLER in upsertEntry: ${e}`); throw e; @@ -943,8 +933,10 @@ async function getTestReports(storyId) { const dbo = db.db(dbName); const collection = await dbo.collection(ReportDataCollection); console.log('Getting Report for storyId :', storyId); - result = await collection.find({ storyId: ObjectId(storyId) }, - { projection: { jsonReport: 0, reportOptions: 0, json: 0 } }).toArray(); + result = await collection.find( + { storyId: new ObjectId(storyId) }, + { projection: { jsonReport: 0, reportOptions: 0, json: 0 } } + ).toArray(); console.log('Got ', result.length, ' reports for :', storyId); } catch (e) { console.log('UPS!!!! FEHLER in getTestReports', e); @@ -963,9 +955,11 @@ async function getGroupTestReports(storyId) { const collection = await dbo.collection(ReportDataCollection); console.log('Getting Groups Reports for storyId :', storyId); // projection value 0 excludes from returning - const query = { storyStatuses: { $elemMatch: { storyId: ObjectId(storyId) } } }; - const result = await collection.find(query, - { projection: { jsonReport: 0, reportOptions: 0, json: 0 } }).toArray(); + const query = { storyStatuses: { $elemMatch: { storyId: new ObjectId(storyId) } } }; + const result = await collection.find( + query, + { projection: { jsonReport: 0, reportOptions: 0, json: 0 } } + ).toArray(); db.close(); console.log('I am closing the DB here - getGroupTestReports'); console.log('Got ', result.length, ' Group Reports for :', storyId); @@ -984,19 +978,19 @@ async function deleteReport(reportId) { db = await connectDb(); const dbo = db.db(dbName); const collection = await dbo.collection(ReportDataCollection); - const reportData = await collection.findOne({ _id: ObjectId(reportId) }); + const reportData = await collection.findOne({ _id: new ObjectId(reportId) }); if (reportData.smallReport) { idToDelete = reportData.smallReport; console.log('Trying to delete smallReport', idToDelete, ' in DB for Report', reportId); const reportsCollection = await dbo.collection(ReportsCollection); - await reportsCollection.deleteOne({ _id: ObjectId(idToDelete) }); - result = await collection.deleteOne({ _id: ObjectId(reportId) }); + await reportsCollection.deleteOne({ _id: new ObjectId(idToDelete) }); + result = await collection.deleteOne({ _id: new ObjectId(reportId) }); } else { idToDelete = reportData.bigReport; console.log('trying to delete bigReport', idToDelete, ' in DB for Report', reportId); const bucket = await new mongodb.GridFSBucket(dbo, { bucketName: 'GridFS' }); - bucket.delete(ObjectId(idToDelete)); - result = await collection.deleteOne({ _id: ObjectId(reportId) }); + bucket.delete(new ObjectId(idToDelete)); + result = await collection.deleteOne({ _id: new ObjectId(reportId) }); } } catch (e) { console.log('UPS!!!! FEHLER in deleteReport', e); @@ -1014,10 +1008,12 @@ async function setIsSavedTestReport(testReportId, isSaved) { db = await connectDb(); const dbo = db.db(dbName); const collection = await dbo.collection(ReportDataCollection); - const updatedReport = await collection.findOne({ _id: ObjectId(testReportId) }); + const updatedReport = await collection.findOne({ _id: new ObjectId(testReportId) }); updatedReport.isSaved = isSaved; - result = await collection.findOneAndReplace({ _id: ObjectId(testReportId) }, - updatedReport); + result = await collection.findOneAndReplace( + { _id: new ObjectId(testReportId) }, + updatedReport + ); } catch (e) { console.log('UPS!!!! FEHLER in setIsSavedTestReport', e); } finally { @@ -1032,7 +1028,7 @@ async function updateStoryStatus(storyId, storyLastTestStatus) { try { db = await connectDb(); const dbo = db.db(dbName); - dbo.collection(storiesCollection).updateOne({ _id: ObjectId(storyId) }, { + dbo.collection(storiesCollection).updateOne({ _id: new ObjectId(storyId) }, { $set: { lastTestPassed: storyLastTestStatus } }); // db.close(); @@ -1049,7 +1045,7 @@ async function updateScenarioStatus(storyId, scenarioId, scenarioLastTestStatus) db = await connectDb(); const dbo = db.db(dbName); const collection = await dbo.collection(storiesCollection); - const story = await collection.findOne({ _id: ObjectId(storyId) }); + const story = await collection.findOne({ _id: new ObjectId(storyId) }); const scenarioList = story.scenarios; const scenario = scenarioList.find((scen) => scen.scenario_id === parseInt(scenarioId, 10)); @@ -1058,7 +1054,7 @@ async function updateScenarioStatus(storyId, scenarioId, scenarioLastTestStatus) scenario.lastTestPassed = scenarioLastTestStatus; story.scenarios[index] = scenario; } - return await collection.findOneAndReplace({ _id: ObjectId(storyId) }, story); + return await collection.findOneAndReplace({ _id: new ObjectId(storyId) }, story); } catch (e) { console.log('Error in updateScenarioStatus. Could not set scenario LastTestPassed: ', e); } finally { @@ -1071,7 +1067,7 @@ async function uploadBigJsonData(data, fileName) { const db = await connectDb(); const dbo = db.db(dbName); const bucket = await new mongodb.GridFSBucket(dbo, { bucketName: 'GridFS' }); - const id = ObjectId(); + const id = new ObjectId(); str(JSON.stringify(data)) .pipe(bucket.openUploadStreamWithId(id, fileName)) .on('error', async (error) => { @@ -1079,7 +1075,7 @@ async function uploadBigJsonData(data, fileName) { }) .on('finish', async () => { console.log('Done! Uplaoded BigReport'); - console.log('ObjectID: of Big Report: ' + id); + console.log(`ObjectID: of Big Report: ${id}`); return id; }); return id; @@ -1096,7 +1092,7 @@ async function uploadReport(reportResults) { if (len >= 16000000) { try { reportData.bigReport = await uploadBigJsonData(jReport, reportResults.storyId); - console.log('ObjectID: of Big Report in UplaodReport: ' + reportData.bigReport); + console.log(`ObjectID: of Big Report in UplaodReport: ${reportData.bigReport}`); collection.insertOne(reportData); } catch (e) { console.log('UPS!!!! FEHLER in uploadReport', e); @@ -1130,11 +1126,11 @@ async function getReport(reportName) { const report = await collection.findOne({ reportName }); if (report.smallReport) { const reportCollection = await dbo.collection(ReportsCollection); - const reportJson = await reportCollection.findOne({ _id: ObjectId(report.smallReport) }); + const reportJson = await reportCollection.findOne({ _id: new ObjectId(report.smallReport) }); result = { _id: report._id, jsonReport: reportJson.jsonReport }; } else { const bucket = await new mongodb.GridFSBucket(dbo, { bucketName: 'GridFS' }); - const reportString = await toString(bucket.openDownloadStream(ObjectId(report.bigReport.toString()))); + const reportString = await toString(bucket.openDownloadStream(new ObjectId(report.bigReport.toString()))); const reportJson = JSON.parse(reportString); result = { _id: report._id, jsonReport: reportJson.jsonReport }; } @@ -1170,7 +1166,7 @@ async function createUser(user) { async function deleteUser(userID) { let db; try { - const oId = ObjectId(userID); + const oId = new ObjectId(userID); const myObjt = { _id: oId }; db = await connectDb(); const collection = await selectUsersCollection(db); @@ -1178,7 +1174,7 @@ async function deleteUser(userID) { const collectionStories = await selectStoriesCollection(db); const repos = await collectionRepo.find({ owner: oId }).toArray(); if (repos) { - for (const repo of repos) for (const storyID of repo.stories) await collectionStories.deleteOne({ _id: ObjectId(storyID) }); + for (const repo of repos) for (const storyID of repo.stories) await collectionStories.deleteOne({ _id: new ObjectId(storyID) }); const resultRepo = await collectionRepo.deleteMany({ owner: oId }); const resultUser = await collection.deleteOne(myObjt); @@ -1198,12 +1194,12 @@ async function deleteUser(userID) { async function updateUser(userID, updatedUser) { let db; try { - const oId = ObjectId(userID); + const oId = new ObjectId(userID); const myObjt = { _id: oId }; db = await connectDb(); const collection = await selectUsersCollection(db); const result = await collection.findOneAndReplace(myObjt, updatedUser); - return result.value; + return result; } catch (e) { console.log(`UPS!!!! FEHLER in updateUser: ${e}`); throw e; @@ -1217,7 +1213,7 @@ async function updateUser(userID, updatedUser) { async function getUserData(userID) { let db; try { - const oId = ObjectId(userID); + const oId = new ObjectId(userID); const myObjt = { _id: oId }; db = await connectDb(); const collection = await selectUsersCollection(db); @@ -1234,7 +1230,7 @@ async function getUserData(userID) { async function saveBlock(block) { let db; try { - block.repositoryId = ObjectId(block.repositoryId); + block.repositoryId = new ObjectId(block.repositoryId); db = await connectDb(); const dbo = db.db(dbName); const collection = await dbo.collection(CustomBlocksCollection); @@ -1272,7 +1268,7 @@ async function getBlocks(userId, repoId) { db = await connectDb(); const dbo = db.db(dbName); const collection = await dbo.collection(CustomBlocksCollection); - return await collection.find({ repositoryId: ObjectId(repoId) }).toArray(); + return await collection.find({ repositoryId: new ObjectId(repoId) }).toArray(); } catch (e) { console.log(`UPS!!!! FEHLER in getBlocks: ${e}`); throw e; @@ -1286,8 +1282,8 @@ async function deleteBlock(blockId, userId) { let db; try { const myObjt = { - _id: ObjectId(blockId), - owner: ObjectId(userId) + _id: new ObjectId(blockId), + owner: new ObjectId(userId) }; db = await connectDb(); const dbo = db.db(dbName); @@ -1309,7 +1305,7 @@ async function getWorkgroup(id) { db = await connectDb(); const dbo = db.db(dbName); const collection = await dbo.collection(WorkgroupsCollection); - return await collection.findOne({ Repo: ObjectId(id) }); + return await collection.findOne({ Repo: new ObjectId(id) }); } catch (e) { console.log(`UPS!!!! FEHLER in getWorkgroup: ${e}`); throw e; @@ -1325,27 +1321,27 @@ async function addMember(id, user) { db = await connectDb(); const dbo = db.db(dbName); const wGCollection = await dbo.collection(WorkgroupsCollection); - const check = await wGCollection.findOne({ Repo: ObjectId(id), Members: { $elemMatch: { email: user.email } } }); + const check = await wGCollection.findOne({ Repo: new ObjectId(id), Members: { $elemMatch: { email: user.email } } }); if (check) return 'Dieser User ist bereits in der Workgroup'; const rCollection = await dbo.collection(repositoriesCollection); - const repo = await rCollection.findOne({ _id: ObjectId(id) }); + const repo = await rCollection.findOne({ _id: new ObjectId(id) }); const usersCollection = await selectUsersCollection(db); const owner = await usersCollection.findOne({ _id: repo.owner }); - const workGroup = await wGCollection.findOne({ Repo: ObjectId(id) }); + const workGroup = await wGCollection.findOne({ Repo: new ObjectId(id) }); if (!workGroup) { await wGCollection.insertOne({ - name: repo.repoName, owner: owner.email, Repo: ObjectId(id), Members: [{ email: user.email, canEdit: user.canEdit }] + name: repo.repoName, owner: owner.email, Repo: new ObjectId(id), Members: [{ email: user.email, canEdit: user.canEdit }] }); const result = { owner: {}, member: [] }; - const wG = await wGCollection.findOne({ Repo: ObjectId(id) }); + const wG = await wGCollection.findOne({ Repo: new ObjectId(id) }); result.owner = { email: owner.email, canEdit: true }; result.member = wG.Members; return result; } // if there is a workGroup already: - await wGCollection.findOneAndUpdate({ Repo: ObjectId(id) }, { $push: { Members: user } }); + await wGCollection.findOneAndUpdate({ Repo: new ObjectId(id) }, { $push: { Members: user } }); const result = { owner: {}, member: [] }; - const wG = await wGCollection.findOne({ Repo: ObjectId(id) }); + const wG = await wGCollection.findOne({ Repo: new ObjectId(id) }); result.owner = { email: owner.email, canEdit: true }; result.member = wG.Members; return result; @@ -1365,12 +1361,12 @@ async function updateMemberStatus(repoId, user) { const dbo = db.db(dbName); const wGCollection = await dbo.collection(WorkgroupsCollection); const rCollection = await dbo.collection(repositoriesCollection); - const repo = await rCollection.findOne({ _id: ObjectId(repoId) }); + const repo = await rCollection.findOne({ _id: new ObjectId(repoId) }); const usersCollection = await selectUsersCollection(db); const owner = await usersCollection.findOne({ _id: repo.owner }); - const updatedWG = await wGCollection.findOneAndUpdate({ Repo: ObjectId(repoId) }, { $set: { 'Members.$[elem].canEdit': user.canEdit } }, { arrayFilters: [{ 'elem.email': user.email }] }); + const updatedWG = await wGCollection.findOneAndUpdate({ Repo: new ObjectId(repoId) }, { $set: { 'Members.$[elem].canEdit': user.canEdit } }, { arrayFilters: [{ 'elem.email': user.email }] }); if (updatedWG) { - const wG = await wGCollection.findOne({ Repo: ObjectId(repoId) }); + const wG = await wGCollection.findOne({ Repo: new ObjectId(repoId) }); const result = { owner: {}, member: [] }; result.owner = { email: owner.email, canEdit: true }; result.member = wG.Members; @@ -1392,10 +1388,10 @@ async function getMembers(id) { const dbo = db.db(dbName); const wGCollection = await dbo.collection(WorkgroupsCollection); const rCollection = await dbo.collection(repositoriesCollection); - const repo = await rCollection.findOne({ _id: ObjectId(id) }); + const repo = await rCollection.findOne({ _id: new ObjectId(id) }); const usersCollection = await selectUsersCollection(db); const owner = await usersCollection.findOne({ _id: repo.owner }); - const wG = await wGCollection.findOne({ Repo: ObjectId(id) }); + const wG = await wGCollection.findOne({ Repo: new ObjectId(id) }); if (!wG) return { owner: { email: owner.email, canEdit: true }, member: [] }; const result = { owner: {}, member: [] }; result.owner = { email: owner.email, canEdit: true }; @@ -1417,12 +1413,12 @@ async function removeFromWorkgroup(id, user) { const dbo = db.db(dbName); const wGcollection = await dbo.collection(WorkgroupsCollection); const rCollection = await dbo.collection(repositoriesCollection); - const repo = await rCollection.findOne({ _id: ObjectId(id) }); + const repo = await rCollection.findOne({ _id: new ObjectId(id) }); const usersCollection = await selectUsersCollection(db); const owner = await usersCollection.findOne({ _id: repo.owner }); - const workGroup = await wGcollection.findOneAndUpdate({ Repo: ObjectId(id) }, { $pull: { Members: { email: user.email } } }); + const workGroup = await wGcollection.findOneAndUpdate({ Repo: new ObjectId(id) }, { $pull: { Members: { email: user.email } } }); if (workGroup) { - const wG = await wGcollection.findOne({ Repo: ObjectId(id) }); + const wG = await wGcollection.findOne({ Repo: new ObjectId(id) }); const result = { owner: {}, member: [] }; result.owner = { email: owner.email, canEdit: true }; result.member = wG.Members; @@ -1444,10 +1440,10 @@ async function updateOneDriver(id, driver) { db = await connectDb(); const collection = await selectStoriesCollection(db); const result = await collection.findOneAndUpdate( - { _id: ObjectId(id) }, + { _id: new ObjectId(id) }, { $set: { oneDriver } } ); - return result.value; + return result; } catch (e) { console.log('UPS!!!! FEHLER in updateOneDriver: ', e); } finally { diff --git a/backend/src/server.js b/backend/src/server.js index 14aa358c5..8bc489531 100644 --- a/backend/src/server.js +++ b/backend/src/server.js @@ -40,11 +40,11 @@ if (process.env.NODE_ENV) app .use(flash()) .use(session({ store: new MongoStore({ - url: process.env.DATABASE_URI || "mongodb://SeedAdmin:SeedTest@seedmongodb:27017", + url: process.env.DATABASE_URI || 'mongodb://SeedAdmin:SeedTest@seedmongodb:27017', dbName: 'Seed', collection: 'Sessions' }), - secret: process.env.SESSION_SECRET || "unsaveSecret", + secret: process.env.SESSION_SECRET || 'unsaveSecret', resave: false, saveUninitialized: false, proxy: true, @@ -56,7 +56,7 @@ if (process.env.NODE_ENV) app else app .use(flash()) .use(session({ - secret: process.env.SESSION_SECRET || "unsaveSecret", + secret: process.env.SESSION_SECRET || 'unsaveSecret', resave: false, saveUninitialized: false, proxy: true @@ -74,7 +74,7 @@ app .use(passport.initialize()) .use(passport.session()) .use((_, __, next) => { - logging.httpLog(_, __, next) + logging.httpLog(_, __, next); }) .use('/api/script', scriptRouter) .use('/api/run', runReportRouter) diff --git a/backend/src/serverHelper.js b/backend/src/serverHelper.js index 9fee04b51..e79911168 100644 --- a/backend/src/serverHelper.js +++ b/backend/src/serverHelper.js @@ -8,7 +8,7 @@ const path = require('path'); const AdmZip = require('adm-zip'); const os = require('os'); const mongo = require('./database/DbServices'); -const { log } = require('console'); +const moment = require('../node_modules/moment'); // adds content of each values to output function getValues(values) { @@ -145,7 +145,7 @@ async function updateFeatureFile(issueID) { if (story != null) { story.scenarios = await replaceRefBlocks(story.scenarios); writeFile(story); - }; + } } async function deleteFeatureFile(storyTitle, storyId) { @@ -164,7 +164,7 @@ async function executeTest(req, mode, story) { let parameters = {}; if (mode === 'scenario') { - const scenario = story.scenarios.find(elem => elem.scenario_id === parseInt(req.params.scenarioId, 10)); + const scenario = story.scenarios.find((elem) => elem.scenario_id === parseInt(req.params.scenarioId, 10)); const scenarioCount = Math.max(scenario.stepDefinitions.example.length, 1); parameters = { @@ -201,10 +201,10 @@ async function executeTest(req, mode, story) { const jsParam = JSON.stringify(parameters); const cucumberArgs = [ path.normalize(featurePath), - ...(mode === 'scenario' ? [`--tags`, `@${req.params.issueID}_${req.params.scenarioId}`] : []), - `--format`, `json:${path.normalize(jsonPath)}`, - `--world-parameters`, jsParam, - `--exit` + ...(mode === 'scenario' ? ['--tags', `@${req.params.issueID}_${req.params.scenarioId}`] : []), + '--format', `json:${path.normalize(jsonPath)}`, + '--world-parameters', jsParam, + '--exit' ]; const cmd = os.platform().includes('win') ? '.cmd' : ''; @@ -226,12 +226,16 @@ async function executeTest(req, mode, story) { return new Promise((resolve) => { runner.on('error', (error) => { console.error(`exec error: ${error}`); - resolve({ reportTime, story, scenarioId: req.params.scenarioId, reportName }); + resolve({ + reportTime, story, scenarioId: req.params.scenarioId, reportName + }); }); runner.on('exit', () => { console.log('test finished'); - resolve({ reportTime, story, scenarioId: req.params.scenarioId, reportName }); + resolve({ + reportTime, story, scenarioId: req.params.scenarioId, reportName + }); }); }); } @@ -355,14 +359,14 @@ async function replaceRefBlocks(scenarios) { if (!scenarios.some((scen) => scen.hasRefBlock)) return scenarios; const retScenarios = []; for (const scen of scenarios) { - let stepdef = {}; + const stepdef = {}; // eslint-disable-next-line guard-for-in for (const steps in scen.stepDefinitions) { // iterate over given, when, then const promised = await scen.stepDefinitions[steps].map(async (elem) => { if (!elem._blockReferenceId) return [elem]; return mongo.getBlock(elem._blockReferenceId).then((block) => { // Get an array of the values of the given, when, then and example properties - let steps = Object.values(block.stepDefinitions); + const steps = Object.values(block.stepDefinitions); // Flatten array return steps.flat(1); }); @@ -402,6 +406,189 @@ async function exportProjectFeatureFiles(repoId, versionId) { }); } +function applyDateCommand(str) { + console.log(str); + let indices = [ + str.indexOf('@@Day'), + str.indexOf('@@Month'), + str.indexOf('@@Year'), + str.indexOf('@@Date') + ]; + + // Filtere alle Indizes heraus, die -1 sind (nicht gefunden) + indices = indices.filter((index) => index !== -1); + + // Wenn das Array leer ist, wurden keine Substrings gefunden + if (indices.length === 0) { + return -1; + } + + // Finde den niedrigsten Index (die erste Vorkommen) + const start = Math.min(...indices); + + const endString = str.substring(start); + const end = start + (endString.indexOf(' ') === -1 ? endString.length : endString.indexOf(' ')); + + const date = calcDate(str.substring(start, end)); + + const startToDate = str.substring(0, start); + const dateToEnd = str.substring(end); + + return startToDate + date + dateToEnd; +} + +function calcDate(value) { + // Regex that matches the middle: e.g. +@@Day,2-@@Month,4 .... + const mid_regex = /(^((\+|\-)@@(\d+),(Day|Month|Year))*)|(^\s*$)/; + // Regex that matches the format end: e.g @@format:DDMMYY€€ + const end_regex = /(^(@@format:\w*€€)*)|(^\s*$)/; + + function getStart(str) { + let endIndex = str.length; + const symbols = ['+', '-', '@@format']; + + symbols.forEach((symbol) => { + const symbolIndex = str.indexOf(symbol); + if (symbolIndex !== -1 && symbolIndex < endIndex) endIndex = symbolIndex; + }); + return str.substring(0, endIndex); + } + + function getMid(str) { + let endIndex = str.length; + const symbols = ['@@format']; + + symbols.forEach((symbol) => { + const symbolIndex = str.indexOf(symbol); + if (symbolIndex !== -1 && symbolIndex < endIndex) endIndex = symbolIndex; + }); + return str.substring(0, endIndex); + } + const start = getStart(value).replace(' ', ''); + const mid = getMid(value.replace(start, '')).replace(' ', ''); + const end = mid.replace(mid, '').trim(); + + // check if the start part is written correctly + const dates = start.split(/@@Date/); + const substrings = [/@@Day,\d{1,2}|@@Day/, /@@Month,\d{1,2}|@@Month/, /@@Year,\d{4}|@@Year/]; + const substringsErr = ['@@Day', '@@Month', '@@Year']; + // check if @@Date has been used + if (dates.length > 1) if (dates.length - 1 > 1) throw Error('@@Date should only be used once.'); + else for (let i = 0; i < substrings.length; i++) { + if (substrings[i].test(start)) throw Error(`@@Date should only be used by itself. Found: ${substringsErr[i]}`); + } + + // check the correct usage of @@Day, @@Month, @@Year + else { + startcopy = start.slice(); + for (let i = 0; i < substrings.length; i++) { + if (start.split(substrings[i]).length - 1 > 1) throw Error(`${substringsErr[i]} may only be used 0 or 1 time. Input: ${start}`); + startcopy = startcopy.replace(substrings[i], ''); + } + // if (startcopy.length !== 0) throw Error(`Unkown tokens in the start section: ${startcopy}`); + } + + // check if the calculation part is written correctly + if (!mid_regex.test(mid)) throw Error('Error parsing the calculation section. Example: +@@23,Day-@@Month,1'); + + // check if the format part is written correctly + if (!end_regex.test(end)) throw Error('Error parsing the format section. Example: @@format:XXXXXX€€. Where XXXXX is the Format String. Example: @@format:DD-MM-YY'); + + // Get the format e.g @@format:XXXXX€€ + let format = value.match(/(@@format:.*€€)/g); + + // Start Date + const currDate = new Date(); + let day = value.match(/(@@Day,\d{1,2})/g); + if (day) day = parseInt(day[0].match(/@@Day,(\d+)/)[1]); + let month = value.match(/(@@Month,\d{1,2})/g); + if (month) month = parseInt(month[0].match(/@@Month,(\d+)/)[1] - 1); + let year = value.match(/(@@Year,\d\d\d\d)/g); + if (year) year = parseInt(year[0].match(/@@Year,(\d+)/)[1]); + + currDate.setFullYear( + year == null ? currDate.getFullYear() : year, + month == null ? currDate.getMonth() : month, + day == null ? currDate.getDate() : day + ); + + // If no format was found, check the given format e.g. @@Date, @@Day@@Month, @@Day ... + if (format == null) { + // Get the Substring until the first add,sub or format e.g @@Day@@Month+@@ ... -> @@Day@@Month + format = value.split(/[\+\-]/)[0]; + // Replace the @@Day, @@Month, @@Year + format = format.replace(/@@Day(,(\d\d){1,2}){0,1}/, 'DD.').replace(/@@Month(,(\d\d){1,2}){0,1}/, 'MM.') + .replace(/@@Year(,(\d\d\d\d)){0,1}/, 'YYYY.') + .replace('@@Date', 'DD.MM.YYYY.') + .slice(0, -1); + } else + // Get @@format: tag and €€ at the end + format = format[0].slice(9, -2); + + // console.log(`Day: ${day}\nMonth: ${month}\nYear: ${year}\nFormat: ${format}\nDate: ${currDate.toDateString()}`); + + // Get all adds e.g +@@2,Month + let adds = value.match(/\+@@(\d+),(\w+)/g); + // Read values e.g. of +@@5,Day -> {number: 5, kind: "Day"}; or set to empty array if null (no match) + adds = adds ? adds.map((element) => { + const match = element.match(/\+@@(\d+),(\w+)/); + return { number: parseInt(match[1]), kind: match[2] }; + }) : []; + // Get all subs e.g -@@10,Year + let subs = value.match(/\-@@(\d+),(\w+)/g); + // Read values e.g. of -@@2,Month -> {number: 2, kind: "Month"}; or set to empty array if null (no match) + subs = subs ? subs.map((element) => { + const match = element.match(/\-@@(\d+),(\w+)/); + return { number: parseInt(match[1]), kind: match[2] }; + }) : []; + + // Add every add in the adds array + adds.forEach((add) => { + switch (add.kind) { + case 'Day': + currDate.setDate(currDate.getDate() + add.number); + break; + case 'Month': + currDate.setMonth(currDate.getMonth() + add.number); + break; + case 'Year': + currDate.setFullYear(currDate.getFullYear() + add.number); + break; + default: + Error(`Unknown type to add to the date: ${add.kind}`); + } + }); + + // Substract every sub in the subs array + subs.forEach((sub) => { + switch (sub.kind) { + case 'Day': + currDate.setDate(currDate.getDate() - sub.number); + break; + case 'Month': + currDate.setMonth(currDate.getMonth() - sub.number); + break; + case 'Year': + currDate.setFullYear(currDate.getFullYear() - sub.number); + break; + default: + Error(`Unknown type to substract of the date: ${sub.kind}`); + } + }); + + // Format the date + const result = moment(currDate).format(format); + return result; +} + +function applySpecialCommands(str) { + let appliedCommandsString = ''; + if (str.includes('@@Day') || str.includes('@@Month') || str.includes('@@Year') || str.includes('@@Date')) { + appliedCommandsString = applyDateCommand(str); + } + return appliedCommandsString; +} + module.exports = { executeTest, updateLatestTestStatus, @@ -418,5 +605,6 @@ module.exports = { updateFeatureFile, deleteFeatureFile, exportSingleFeatureFile, - exportProjectFeatureFiles + exportProjectFeatureFiles, + applySpecialCommands }; diff --git a/backend/src/serverRouter/blockRouter.js b/backend/src/serverRouter/blockRouter.js index b453f72cf..6a84ffa33 100644 --- a/backend/src/serverRouter/blockRouter.js +++ b/backend/src/serverRouter/blockRouter.js @@ -37,7 +37,7 @@ router.post('/', async (req, res) => { try { const { body } = req; if (!req.user) { res.sendStatus(401); return; } - body.owner = ObjectID(req.user._id); + body.owner = new ObjectID(req.user._id); const result = await mongo.saveBlock(body); res.status(200).json(result); } catch (error) { @@ -52,7 +52,7 @@ router.put('/block', async (req, res) => { if (!req.user) res.sendStatus(401); else { const result = await mongo.updateBlock(body); - res.status(200).json(result.value); + res.status(200).json(result); } } catch (error) { handleError(res, error, error, 500); diff --git a/backend/src/serverRouter/storyRouter.js b/backend/src/serverRouter/storyRouter.js index 05ad5ffed..45c763f8c 100644 --- a/backend/src/serverRouter/storyRouter.js +++ b/backend/src/serverRouter/storyRouter.js @@ -181,4 +181,13 @@ router.post('/oneDriver/:storyID', async (req, res) => { } }); +router.post('/specialCommands/resolve', async (req, res) => { + try { + const result = helper.applySpecialCommands(req.body.command); + res.status(200).json(result); + } catch (error) { + handleError(res, error, error, 500); + } +}); + module.exports = router; diff --git a/backend/src/serverRouter/userRouter.js b/backend/src/serverRouter/userRouter.js index 75b27587f..769df09d5 100644 --- a/backend/src/serverRouter/userRouter.js +++ b/backend/src/serverRouter/userRouter.js @@ -63,8 +63,8 @@ router.post('/resetpassword', async (req, res) => { try { await nodeMail.sendResetLink(thisUser.email, id); res.status(200) - .json(); - } catch(err) { + .json(); + } catch (err) { res.status(500).send(err.message); } } catch (error) { @@ -72,7 +72,7 @@ router.post('/resetpassword', async (req, res) => { .json(error); } else { console.log('UserRouter/ResetPassword: der Benutzer konnte nicht in der Datenbank gefunden werden!'); - res.status(404).send("No user found with the given email adress!"); + res.status(404).send('No user found with the given email adress!'); } }); @@ -167,7 +167,7 @@ router.post('/register', async (req, res) => { // logout for user router.get('/logout', async (req, res) => { - req.logout({}, () => {}); + req.logout({}, () => { }); res.clearCookie('connect.sid', { path: '/' }); res.status(200).send({ status: 'success' }); }); @@ -257,7 +257,7 @@ router.get('/stories', async (req, res) => { // put into ticketManagement.ts const { source } = req.query; // get GitHub Repo / Projects if (source === 'github' || !source) try { - if (!userMng.checkValidGithub(req.query.githubName, req.query.repository))console.log('Username or Reponame not valid'); + if (!userMng.checkValidGithub(req.query.githubName, req.query.repository)) console.log('Username or Reponame not valid'); const githubName = (req.user) ? req.query.githubName : process.env.TESTACCOUNT_NAME; const githubRepo = (req.user) ? req.query.repository : process.env.TESTACCOUNT_REPO; @@ -310,7 +310,7 @@ router.get('/stories', async (req, res) => { // put into ticketManagement.ts } catch (err) { res.status(503).send(err.message); - // get Jira Repo / Projects + // get Jira Repo / Projects } else if (source === 'jira' && typeof req.user !== 'undefined' && typeof req.user.jira !== 'undefined' && req.query.projectKey !== 'null') { // prepare request const { projectKey } = req.query; @@ -318,7 +318,7 @@ router.get('/stories', async (req, res) => { // put into ticketManagement.ts const clearPass = jiraTracker.decryptPassword(req.user.jira); const { AccountName, AuthMethod, Host } = req.user.jira; let authString = `Bearer ${clearPass}`; - if (AuthMethod === 'basic') { + if (AuthMethod === 'basic') { const auth = Buffer.from(`${AccountName}:${clearPass}`).toString('base64'); authString = `Basic ${auth}`; } @@ -438,9 +438,9 @@ router.get('/callback', (req, res) => { const TOKEN_URL = 'https://github.com/login/oauth/access_token'; const params = new URLSearchParams(); if (!process.env.GITHUB_CLIENT_ID || !process.env.GITHUB_CLIENT_SECRET) { - console.log("To use github authentication please provide your GITHUB_CLIENT_ID and your GITHUB_CLIENT_SECRET. You can see how to in the README."); - res.status(501).send("No GITHUB_CLIENT_ID or GITHUB_CLIENT_SECRET provided.") - return + console.log('To use github authentication please provide your GITHUB_CLIENT_ID and your GITHUB_CLIENT_SECRET. You can see how to in the README.'); + res.status(501).send('No GITHUB_CLIENT_ID or GITHUB_CLIENT_SECRET provided.'); + return; } params.append('client_id', process.env.GITHUB_CLIENT_ID); @@ -467,7 +467,7 @@ router.get('/callback', (req, res) => { router.post('/log', (req, res) => { const stream = fs.createWriteStream('./logs/front.log', { flags: 'a' }); - stream.write(req.body.message + JSON.stringify(req.body.additional) + '\n'); + stream.write(`${req.body.message + JSON.stringify(req.body.additional)}\n`); stream.close(); res.status(200).json('logged'); }); diff --git a/frontend/package-lock.json b/frontend/package-lock.json index 6ee2dd25b..92295d688 100644 --- a/frontend/package-lock.json +++ b/frontend/package-lock.json @@ -1,12 +1,12 @@ { "name": "seed-test-frontend", - "version": "1.6.2", + "version": "1.7.0", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "seed-test-frontend", - "version": "1.6.2", + "version": "1.7.0", "dependencies": { "@angular-devkit/build-angular": "^15.2.9", "@angular/animations": "^15.0.4", diff --git a/frontend/package.json b/frontend/package.json index 8085ec660..98ec34b9b 100644 --- a/frontend/package.json +++ b/frontend/package.json @@ -1,6 +1,6 @@ { "name": "seed-test-frontend", - "version": "1.6.2", + "version": "1.7.0", "engines": { "node": "18.13.0" }, diff --git a/frontend/src/app/Services/story.service.ts b/frontend/src/app/Services/story.service.ts index 07876b430..334637421 100644 --- a/frontend/src/app/Services/story.service.ts +++ b/frontend/src/app/Services/story.service.ts @@ -175,7 +175,7 @@ export class StoryService { */ public changeOneDriver(oneDriver: boolean, storyID: any) { return this.http - .post(this.apiService.apiServer + '/story/oneDriver/' + storyID, { oneDriver }); + .post(this.apiService.apiServer + '/story/oneDriver/' + storyID, { oneDriver }, ApiService.getOptions()); } /** * Checking the same name of the story diff --git a/frontend/src/app/app.component.html b/frontend/src/app/app.component.html index ce9ec947c..cda109466 100644 --- a/frontend/src/app/app.component.html +++ b/frontend/src/app/app.component.html @@ -3,42 +3,47 @@
- Version 1.6.3 {{version === '' ? " (demo)" : ""}} + Version 1.7.0 {{version === '' ? " (demo)" : ""}}
- This is a demo version of Seed-Test. - Please note that to use it in production, you need to make some configurations to ensure it's fully functional. - Certain features may not work until you complete the installation process. + This is a demo version of Seed-Test. + Please note that to use it in production, you need to make some configurations to ensure it's fully + functional. + Certain features may not work until you complete the installation process. We strongly advise against using this unconfigured version in a production environment. For more information, please refer to the README. @@ -59,20 +64,19 @@

brightness_5 - + bedtime
+
\ No newline at end of file