From bafcb5e7dd22d8a99b32d59a12cbdd07f470db81 Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 16:38:36 -0600 Subject: [PATCH 01/24] Added: files repository --- src/db/repositories/fileRepository.js | 185 ++++++++++++++++++++++++++ 1 file changed, 185 insertions(+) create mode 100644 src/db/repositories/fileRepository.js diff --git a/src/db/repositories/fileRepository.js b/src/db/repositories/fileRepository.js new file mode 100644 index 0000000..15f1fe7 --- /dev/null +++ b/src/db/repositories/fileRepository.js @@ -0,0 +1,185 @@ +/** + * Represents a repository for managing files in the database + */ +class FileRepository { + constructor (database) { + this.db = database + this.createTable() + } + + /** + * Creates the "files" table in the database if it does not exist + */ + createTable () { + this.db.prepare(` + CREATE TABLE IF NOT EXISTS files ( + id INTEGER PRIMARY KEY AUTOINCREMENT, + description TEXT NOT NULL, + filename TEXT NOT NULL, + mimetype TEXT NOT NULL, + src TEXT NOT NULL, + username TEXT NOT NULL + ) + `).run() + // this.db.prepare('ADD COLUMN IF NOT EXISTS') + } + + /** + * Inserts a single file record into the database + * @param {Object} file - An object containing description, filename, mimetype, src, and username + * @returns {Object} The inserted file record + */ + insertFile ({ description, filename, mimetype, src, username }) { + const result = this.db + .prepare(` + INSERT INTO files (description, filename, mimetype, src, username) + VALUES (@description, @filename, @mimetype, @src, @username) + RETURNING * + `) + .run({ description, filename, mimetype, src, username }) + const dbFile = this.getFileById(result.lastInsertRowid) + // console.log(`[INFO] FileRepository.insertFile(): "${filename}"`, dbFile) + return dbFile + } + + /** + * Inserts multiple file records into the database in a single transaction. + * @param {Array} data - An array of objects (filename, descriptionm mimetype, src, and username) to be inserted into the database. + */ + bulkInsertFiles (data) { + const insert = this.db.prepare(` + INSERT INTO files + (description, filename, mimetype, src, username) + VALUES + (@description, @filename, @mimetype, @src, @username) + `) + + const insertMany = this.db.transaction((cats) => { + for (const cat of cats) { + insert.run({ + description: cat.description, + filename: cat.filename, + mimetype: cat.mimetype, + src: cat.src, + username: cat.username, + }) + } + }) + insertMany(data) + } + + /** + * Retrieves a file record from the database by its ID. + * @param {number} id - The ID of the file record. + * @returns {Object} The file record matching the provided ID. + */ + getFileById (id) { + return this.db.prepare(` + SELECT * FROM files + WHERE id = :id + `).get({ id }) + } + + /** + * Retrieves the count of files associated with a specific user. + * @param {string} username - The username of the user. + * @returns {number} The count of files uploaded by the user. + */ + getUserFilesCount (username) { + return this.db.prepare(` + SELECT COUNT(*) AS count FROM files + WHERE username = :username + `).get({ username }).count + } + + /** + * Retrieves paginated files associated with a specific user. + * @param {string} username - The username of the user. + * @returns {object} An object of file records uploaded by the user and total. + */ + getUserFiles ({ username }) { + return this.db.prepare(` + SELECT * FROM files + WHERE username = :username + `).all({ username }) + } + + /** + * Retrieves the count of all files. + * @returns {number} The count of all files. + */ + getTotal () { + return this.db.prepare(` + SELECT COUNT(*) AS count FROM files + `).get().count + } + + /** + * Retrieves all files stored in the database. + * @returns {Array} An array of all file records stored. + * @returns {Array} An array of all file records. + */ + getAllFiles () { + return this.db.prepare(`SELECT * FROM files`).all() + } + + /** + * Deletes a file record from the database by its ID. + * @param {number} id - The ID of the file record to be deleted. + * @returns {boolean} A boolean, return true if deletion was successful. + */ + deleteFileById (id) { + const deleteStmt = this.db.prepare(` + DELETE FROM files + WHERE id = :id + `) + return deleteStmt.run({ id }).changes > 0 + } + + /** + * Retrieves the original file record matching the provided criteria. + * @param {Object} criteria - An object match the containing (username, filename, and base64) for finding the original file. + * @returns {Object|null} The original file record matching the provided criteria or null if not found. + */ + getOriginalFile ({ username, filename, base64 }) { + return this.db.prepare(` + SELECT * + FROM files + WHERE + filename == :filename + AND src == :src + AND username == :username + `) + .get({ + filename, + username, + src: base64, + }) + } + + // getCountMatchingCriteria + /** + * Retrieves all files similar to the provided criteria. Used to get the list of duplicate document + * @param {Object} criteria - An object containing the criteria (username, filename, fileExt, and base64) for finding similar files. + * @returns {Array} An object with records similar to the provided criteria and total matching in DB. + */ + findAllLike ({ username, filename, fileExt, base64 }) { + return this.db.prepare(` + SELECT * + FROM files + WHERE + filename == :filename OR filename LIKE :similarFilename + AND src == :src + AND username == :username + ORDER BY filename ASC + `) + .all({ + filename: `${filename}.${fileExt}`, + similarFilename: `${filename}(%.${fileExt}`, + src: base64, + username, + }) + } +} + +module.exports = FileRepository \ No newline at end of file From 9d37ae7bbc7800d075a3ea2912f9f308d5654cf1 Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 16:39:40 -0600 Subject: [PATCH 02/24] Updated: seed data, added username --- scripts/seed.json | 13 +++++++++---- 1 file changed, 9 insertions(+), 4 deletions(-) diff --git a/scripts/seed.json b/scripts/seed.json index 6ffc481..aec2181 100644 --- a/scripts/seed.json +++ b/scripts/seed.json @@ -4,30 +4,35 @@ "description": "A kitten that is 200x300", "filename": "kitten.jpg", "mimetype": "image/jpg", - "src": "http://placekitten.com/200/300" + "src": "http://placekitten.com/200/300", + "username": "testuser" }, { "id": "6eb00541-1fd5-4779-a155-ba0e53e0fdef", "description": "Another kitten, 500x300", "filename": "kitten(1).jpg", "mimetype": "image/jpg", - "src": "http://placekitten.com/500/300" + "src": "http://placekitten.com/500/300", + "username": "testuser" }, { "id": "6eb00541-1fd5-4779-a155-ba0e53e0f123", "description": "The last kitten. Black & White because 🎨 (800x700)", "filename": "kitten(2).jpg", "mimetype": "image/jpg", - "src": "http://placekitten.com/g/800/700" + "src": "http://placekitten.com/g/800/700", + "username": "testuser" }, { "id": "6eb00541-1fd5-4779-a155-ba0e53e0f12a", "description": "A fluffy dog. Way too much fur for that beach", "filename": "dog(2).jpg", "mimetype": "image/jpg", - "src": "https://placedog.net/1200/550" + "src": "https://placedog.net/1200/550", + "username": "testuser" }, { "id": "6eb00541-1fd5-4779-a155-ba0e53e0f0c0", "description": "The last known picture of a Elvis (probably)", "filename": "elvis.jpg", "mimetype": "image/jpeg", + "username": "testuser", "src": "" }] } From f0947617e17304ebda7a62dbf0b8e9f6be5185ef Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 16:40:30 -0600 Subject: [PATCH 03/24] Updated: seed script to use FileRepository methods --- scripts/create-seed-db.js | 32 +++++++------------------------- 1 file changed, 7 insertions(+), 25 deletions(-) diff --git a/scripts/create-seed-db.js b/scripts/create-seed-db.js index 8256e1d..c6e001a 100644 --- a/scripts/create-seed-db.js +++ b/scripts/create-seed-db.js @@ -2,6 +2,7 @@ const fs = require('fs') const path = require('path') const Database = require('better-sqlite3') const seedData = require('./seed.json') +const FileRepository = require('../src/db/repositories/fileRepository') const SEED_PATH = path.join(__dirname, '..', 'src', 'db', 'seed.db') @@ -12,34 +13,15 @@ function main () { } const db = new Database(SEED_PATH) - createTables(db) - addSeedData(db) + const fileRepository = new FileRepository(db) + fileRepository.createTable() + addSeedData(fileRepository) } -function createTables (db) { - console.log('Creating files table') - db.prepare(` - CREATE TABLE files ( - id INTEGER PRIMARY KEY AUTOINCREMENT, - description TEXT not null, - filename TEXT not null, - mimetype TEXT not null, - src TEXT not null - ) - `).run() -} - -function addSeedData (db) { - console.log('Inserting seed data') - const insertFiles = db.prepare(` - INSERT INTO files - (description, filename, mimetype, src) - VALUES - (@description, @filename, @mimetype, @src) - `) - +function addSeedData (fileRepository) { + console.log('Inserting seed data'); for (const file of seedData.files) { - insertFiles.run(file) + fileRepository.insertFile(file.description, file.filename, file.mimetype, file.src, file.username); } } From e4fa0748146cb0cf145d644d857339ccd3947803 Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 16:42:31 -0600 Subject: [PATCH 04/24] Added: file upload helper --- src/utils/fileUploader.js | 115 ++++++++++++++++++++++++++++++++++++++ 1 file changed, 115 insertions(+) create mode 100644 src/utils/fileUploader.js diff --git a/src/utils/fileUploader.js b/src/utils/fileUploader.js new file mode 100644 index 0000000..12e5d37 --- /dev/null +++ b/src/utils/fileUploader.js @@ -0,0 +1,115 @@ +/** + * A utility class for handling file uploads. + */ +class FileUploadHelper { + /** + * Initializes the FileUploadHelper with file data. + * @param {Object} file - The file data object. + */ + constructor (file) { + /** + * The file data object. + * @type {Object} + */ + this.fileData = file + /** + * The original filename of the file. + * @type {string} + */ + this.originalFilename = file.name + + this.fileExt = null + this.fileBaseName = null + this.setFilenameParts() + } + + /** + * Parses the filename string to return the base filename and the file extension. + * @param {string} filename - The filename to parse. + * @returns {Object} An object containing the base filename and file extension. + */ + extractFilenameParts (filename) { + const fileNameParts = filename.split('.') + let baseFileName + + const fileExt = fileNameParts[fileNameParts.length - 1] + if (fileNameParts.length > 2) { + baseFileName = fileNameParts.slice(0, -1).join('.') + } else { + baseFileName = fileNameParts[0] + } + return { + baseFileName, + fileExt, + } + } + + /** + * Sets values for the class's base filename and the file extension + */ + setFilenameParts () { + const { fileExt, baseFileName } = this.extractFilenameParts(this.fileData.name) + this.fileExt = fileExt + this.fileBaseName = baseFileName + } + + /** + * Checks if the provided database file has the same name and source data as the current file. + * @param {Object} dbFile - The file data from the database. + * @returns {boolean} Returns true if the files are duplicate, otherwise false. + */ + isDuplicate (dbFile) { + return this.fileData.name === dbFile.filename && this.fileData.base64 === dbFile.src + } + + /** + * Generates a new filename for a duplicate file. + * If the filename already exists in the database, it appends a duplicate number to the filename. + * If the filename already contains a duplicate number, it replaces it with the new duplicate number. + * @param {string} dbFilename - The filename from the database. + * @param {number} dupNum - The duplicate number to append to the filename. + * @returns {string} The new filename. + */ + generateDuplicateFilename (dbFilename, dupNum) { + // If duplicate number is not provided, default to 1 + dupNum = dupNum || 1 + + const { baseFileName: dbBaseName, fileExt: dbFileExt } = this.extractFilenameParts(dbFilename) + // if the file to be uploaded has the same file name as entity in database, adding duplicate name syntax + if (dbFilename === this.originalFilename) { + return `${dbBaseName}(${dupNum}).${dbFileExt}` + } + + // For uploading files with names like 'test(1).jpg' or 'test(1)(1).jpg', handle duplicate name syntax + const filenamePattern = /^(?[\w\-_\d]+(\(\d+\))*?)(?=(?((\(\d+\))?\.[a-z]+$)))/ + let newFilename = dbFilename + + const matchResult = newFilename.match(filenamePattern) + if (matchResult) { + /** + * Extract base filename and extension from the matched groups + * Files like: + * - test.png will have a groups.name "test" and groups.end ".png" + * - test(1).png have a group.name "test" and groups.end "(1).png", + */ + newFilename = matchResult.groups.name + const extension = matchResult.groups.end + + // Extract the duplicate number from the ending using regex pattern + const dupNumPattern = /\d+/ + const dupNumMatch = extension.match(dupNumPattern) + + // If the duplicate number is found in the ending, replace it with the new duplicate number + if (dupNumMatch) { + extension.replace(dupNumMatch[0], dupNum.toString()) + newFilename += extension.replace(dupNumMatch[0], dupNum.toString()) + } else { + // If no duplicate number found, append the new duplicate number to the filename + newFilename += `(${dupNum.toString()})${extension}` + } + } + return newFilename + } +} + +module.exports = FileUploadHelper \ No newline at end of file From be5ca6d873bec5c5b8bb9ae81a0bda11c8dd92f4 Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 16:43:42 -0600 Subject: [PATCH 05/24] Added: middleware to send username in request --- src/server/index.js | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/src/server/index.js b/src/server/index.js index 084533e..5ee876d 100644 --- a/src/server/index.js +++ b/src/server/index.js @@ -1,4 +1,5 @@ const path = require('path') +const os = require('os') const express = require('express') const appModulePath = require('app-module-path') @@ -7,6 +8,12 @@ appModulePath.addPath(path.join(__dirname, '..', '..', 'src')) const routes = require('./routes') const app = express() +// Middleware to attach username to request object +app.use((req, res, next) => { + req.username = os.userInfo().username + next() +}) + app.use(express.static('dist')) app.use(express.json({ inflate: true, From 42c0a471c9698b0ee2f1aebd663e1ac25179d6a1 Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 16:47:50 -0600 Subject: [PATCH 06/24] Updated: get username endpoint to get username from request --- src/server/routes/index.js | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/server/routes/index.js b/src/server/routes/index.js index 9da68c8..8b9c991 100644 --- a/src/server/routes/index.js +++ b/src/server/routes/index.js @@ -1,4 +1,3 @@ -const os = require('os') const db = require('db') // SQLite usage @@ -7,7 +6,7 @@ const db = require('db') function buildRoutes (router) { router.get('/api/username', async (req, res) => { - return res.send({ username: os.userInfo().username }) + return res.send({ username: req.username }) }) router.get('/api/files', async (req, res) => { From e3d075c4c96de10a804a894e856354e352708fde Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 17:23:37 -0600 Subject: [PATCH 07/24] Updated: impleted using FileRepository on endpoints --- src/server/routes/index.js | 32 +++++++++++++------------------- 1 file changed, 13 insertions(+), 19 deletions(-) diff --git a/src/server/routes/index.js b/src/server/routes/index.js index 8b9c991..fdc6bdb 100644 --- a/src/server/routes/index.js +++ b/src/server/routes/index.js @@ -1,39 +1,33 @@ const db = require('db') +const FileRepository = require('db/repositories/fileRepository') // SQLite usage // https://github.com/WiseLibs/better-sqlite3 // https://github.com/WiseLibs/better-sqlite3/blob/master/docs/api.md function buildRoutes (router) { + const fileRepository = new FileRepository(db.instance) + router.get('/api/username', async (req, res) => { return res.send({ username: req.username }) }) router.get('/api/files', async (req, res) => { - const files = db.instance - .prepare(` - SELECT * FROM files - `) - .all() // use .get() to fetch a single row + const files = fileRepository.getUserFiles({ username: req.username }) return res.send(files) }) router.post('/api/files', async (req, res) => { + const username = req.username const { description, file } = req.body - const newFile = db.instance - .prepare(` - INSERT INTO files - (description, filename, mimetype, src) - VALUES - (@description, @filename, @mimetype, @src) - RETURNING * - `) - .get({ - description, - filename: file.name, - mimetype: file.mimetype, - src: file.base64, - }) + + const newFile = fileRepository.insertFile({ + description, + filename: file.name, + mimetype: file.mimetype, + src: file.base64, + username, + }) return res.send(newFile) }) From 9891d5df7418d97adcffb5455f0dab5bb2c2a977 Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 17:47:32 -0600 Subject: [PATCH 08/24] Updated: put file endpoint, use FileUploadHelper to generate duplicate filename --- src/server/routes/index.js | 37 ++++++++++++++++++++++++++++++++++++- 1 file changed, 36 insertions(+), 1 deletion(-) diff --git a/src/server/routes/index.js b/src/server/routes/index.js index fdc6bdb..f5c4f33 100644 --- a/src/server/routes/index.js +++ b/src/server/routes/index.js @@ -1,5 +1,6 @@ const db = require('db') const FileRepository = require('db/repositories/fileRepository') +const FileUploadHelper = require('utils/fileUploader') // SQLite usage // https://github.com/WiseLibs/better-sqlite3 @@ -21,9 +22,43 @@ function buildRoutes (router) { const username = req.username const { description, file } = req.body + const dbOriginalFile = fileRepository.getOriginalFile({ + username, + filename: file.name, + base64: file.base64, + }) + + let filename = file.name + + if (dbOriginalFile) { + const fileUploader = new FileUploadHelper(file) + const existingFiles = fileRepository.findAllLike({ + username, + filename: fileUploader.fileBaseName, + base64: file.base64, + fileExt: fileUploader.fileExt, + }) + + if (!existingFiles.length) { + filename = fileUploader.generateDuplicateFilename(dbOriginalFile.filename) + } else { + // If there are multiple duplicate db files, finding the next duplicate number (fill the gaps) + let i = 0 + while (i < existingFiles.length - 1) { + iters += 1 + // if current file name identifier does not match the current iteration, breakout out of loop. Missing a duplicate, gap should be filled + if (existingFiles[i].filename !== `${fileUploader.fileBaseName}(${i+1}).${fileUploader.fileExt}`) { + break + } + i += 1 + } + filename = fileUploader.generateDuplicateFilename(existingFiles[i].filename, i+1) + } + } + const newFile = fileRepository.insertFile({ description, - filename: file.name, + filename, mimetype: file.mimetype, src: file.base64, username, From a992bcdeadfd31c489528053819cf6407f961253 Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 18:08:24 -0600 Subject: [PATCH 09/24] Added: test data generation helper function --- test/server/routes/index.test.js | 20 ++++++++++++++++++++ 1 file changed, 20 insertions(+) diff --git a/test/server/routes/index.test.js b/test/server/routes/index.test.js index 90e3a00..d8c8ab9 100644 --- a/test/server/routes/index.test.js +++ b/test/server/routes/index.test.js @@ -5,6 +5,26 @@ const buildMockRouter = require('../buildMockRouter') describe('routes', function () { let router, route, res, req + + const testBaseFileName = 'bobby-tables' + const testFileExtension = 'jpg' + + function buildUploadData ({ + baseFileName=null, + fileExt=null, + base64=null, + mimetype=null, + }) { + return { + description: 'A portait of an artist', + file: { + name: `${baseFileName || testBaseFileName}.${fileExt || testFileExtension}`, + mimetype: mimetype || 'image/jpg', + base64: base64 || '', + }, + } + } + beforeEach(async function () { req = {} res = { From fae6d52131250dca25679d1d4184c2a3c608257d Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 18:09:07 -0600 Subject: [PATCH 10/24] Added: test validation function --- test/server/routes/index.test.js | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/test/server/routes/index.test.js b/test/server/routes/index.test.js index d8c8ab9..5e49388 100644 --- a/test/server/routes/index.test.js +++ b/test/server/routes/index.test.js @@ -25,6 +25,14 @@ describe('routes', function () { } } + function validateTest ({ inputFile, expectedFilename, responseBody }) { + expect(responseBody.id).to.be.ok + expect(responseBody.description).to.equal(inputFile.description) + expect(responseBody.filename).to.equal(expectedFilename) + expect(responseBody.mimetype).to.equal(inputFile.file.mimetype) + expect(responseBody.src).to.equal(inputFile.file.base64) + } + beforeEach(async function () { req = {} res = { From 2956f0d076b6a70687f6e385c73c5099c6ba2ecc Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 18:14:04 -0600 Subject: [PATCH 11/24] Updated: tests to use helper functions --- test/server/routes/index.test.js | 29 +++++++++++++---------------- 1 file changed, 13 insertions(+), 16 deletions(-) diff --git a/test/server/routes/index.test.js b/test/server/routes/index.test.js index 5e49388..bb719bc 100644 --- a/test/server/routes/index.test.js +++ b/test/server/routes/index.test.js @@ -4,10 +4,11 @@ const buildRoutes = require('server/routes') const buildMockRouter = require('../buildMockRouter') describe('routes', function () { - let router, route, res, req + let router, route, res, req, inputFile const testBaseFileName = 'bobby-tables' const testFileExtension = 'jpg' + const username = 'testuser' function buildUploadData ({ baseFileName=null, @@ -34,7 +35,9 @@ describe('routes', function () { } beforeEach(async function () { - req = {} + req = { + username, + } res = { send: (value) => { res.body = value }, } @@ -56,24 +59,18 @@ describe('routes', function () { describe('POST /api/files', function () { beforeEach(async function () { route = '/api/files' + inputFile = buildUploadData({}) }) it('uploads a file and returns its metadata', async function () { - const input = { - description: 'A portait of an artist', - file: { - name: 'bobby-tables.jpg', - mimetype: 'image/jpg', - base64: '', - }, - } - req.body = input + req.body = inputFile await router.postRoutes[route](req, res) - expect(res.body.id).to.be.ok - expect(res.body.description).to.equal(input.description) - expect(res.body.filename).to.equal(input.file.name) - expect(res.body.mimetype).to.equal(input.file.mimetype) - expect(res.body.src).to.equal(input.file.base64) + + validateTest({ + inputFile, + expectedFilename: inputFile.file.name, + responseBody: res.body, + }) }) }) }) From 452bbfbc53ac19ad77ba4ed903f88e93e8c29f67 Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 18:16:06 -0600 Subject: [PATCH 12/24] Added: test for getting different user files --- test/server/routes/index.test.js | 13 +++++++++++-- 1 file changed, 11 insertions(+), 2 deletions(-) diff --git a/test/server/routes/index.test.js b/test/server/routes/index.test.js index bb719bc..b8d92fe 100644 --- a/test/server/routes/index.test.js +++ b/test/server/routes/index.test.js @@ -2,13 +2,15 @@ const { expect } = require('chai') const db = require('db') const buildRoutes = require('server/routes') const buildMockRouter = require('../buildMockRouter') +const FileRepository = require('db/repositories/fileRepository') describe('routes', function () { - let router, route, res, req, inputFile + let router, route, res, req, inputFile, totalSeedTestFiles const testBaseFileName = 'bobby-tables' const testFileExtension = 'jpg' const username = 'testuser' + const fileRepository = new FileRepository(db.instance) function buildUploadData ({ baseFileName=null, @@ -48,11 +50,18 @@ describe('routes', function () { describe('GET /api/files', function () { beforeEach(async function () { route = '/api/files' + totalSeedTestFiles = fileRepository.getTotal() }) it('returns all the files', async function () { await router.getRoutes[route](req, res) - expect(res.body).to.have.length(5) + expect(res.body).to.have.length(totalSeedTestFiles) + }) + + it('returns 0 files for user with no uploads', async function () { + req.username = 'test1' + await router.getRoutes[route](req, res) + expect(res.body).to.have.length(0) }) }) From 2ce6c7feac661aa465e2bc10659638e78f2ddd7f Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 18:19:44 -0600 Subject: [PATCH 13/24] Added: tests for dup upload --- test/server/routes/index.test.js | 63 ++++++++++++++++++++++++++++++++ 1 file changed, 63 insertions(+) diff --git a/test/server/routes/index.test.js b/test/server/routes/index.test.js index b8d92fe..b451854 100644 --- a/test/server/routes/index.test.js +++ b/test/server/routes/index.test.js @@ -81,5 +81,68 @@ describe('routes', function () { responseBody: res.body, }) }) + + it('users uploads a file, same filename as file uploaded by different user', async function () { + const username2 = 'testuser2' + fileRepository.insertFile({ + description: inputFile.description, + filename: inputFile.file.name, + mimetype: inputFile.file.mimetype, + src: inputFile.file.base64, + username: username2, + }) + const totalRecords = fileRepository.getTotal() + expect(totalRecords).to.equal(totalSeedTestFiles + 1) + + const input = buildUploadData({}) + req.body = input + await router.postRoutes[route](req, res) + + validateTest({ + inputFile, + expectedFilename:input.file.name, + responseBody: res.body, + }) + }) + + it('uploads 1st duplicate, original exists in db', async function () { + fileRepository.insertFile({ + description: inputFile.description, + filename: inputFile.file.name, + mimetype: inputFile.file.mimetype, + src: inputFile.file.base64, + username, + }) + + // uploading a duplicate + req.body = inputFile + await router.postRoutes[route](req, res) + + validateTest({ + inputFile, + expectedFilename: `${testBaseFileName}(1).${testFileExtension}`, + responseBody: res.body, + }) + }) + + it('uploads 1st duplicate, original exists in db for user with 100+ duplicate files', async function () { + fileRepository.insertFile({ + description: inputFile.description, + filename: inputFile.file.name, + mimetype: inputFile.file.mimetype, + src: inputFile.file.base64, + username, + }) + + // uploading a duplicate + req.body = inputFile + await router.postRoutes[route](req, res) + + validateTest({ + inputFile, + expectedFilename: `${testBaseFileName}(1).${testFileExtension}`, + responseBody: res.body, + }) + }) }) }) From 16c225d6e72df0aa2f51e5188eeb8e35cae9e5df Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 18:22:05 -0600 Subject: [PATCH 14/24] Added: tests for dup upload to fill gaps --- test/server/routes/index.test.js | 60 +++++++++++++++++++++++++++++++- 1 file changed, 59 insertions(+), 1 deletion(-) diff --git a/test/server/routes/index.test.js b/test/server/routes/index.test.js index b451854..1d56925 100644 --- a/test/server/routes/index.test.js +++ b/test/server/routes/index.test.js @@ -44,7 +44,6 @@ describe('routes', function () { send: (value) => { res.body = value }, } router = buildRoutes(buildMockRouter()) - db.resetToSeed() }) describe('GET /api/files', function () { @@ -69,6 +68,7 @@ describe('routes', function () { beforeEach(async function () { route = '/api/files' inputFile = buildUploadData({}) + db.resetToSeed() }) it('uploads a file and returns its metadata', async function () { @@ -145,4 +145,62 @@ describe('routes', function () { }) }) }) + + describe('Upload duplicates fill gaps, original file named', function () { + + this.beforeAll(async function () { + route = '/api/files' + db.resetToSeed() + inputFile = buildUploadData({}) + fileRepository.bulkInsertFiles([ + { + description: inputFile.description, + filename: inputFile.file.name, + mimetype: inputFile.file.mimetype, + src: inputFile.file.base64, + username, + }, + { + description: inputFile.description, + filename: `${testBaseFileName}(1).${testFileExtension}`, + mimetype: inputFile.file.mimetype, + src: inputFile.file.base64, + username, + }, + { + description: inputFile.description, + filename: `${testBaseFileName}(3).${testFileExtension}`, + mimetype: inputFile.file.mimetype, + src: inputFile.file.base64, + username, + }, + { + description: inputFile.description, + filename: `${testBaseFileName}(5).${testFileExtension}`, + mimetype: inputFile.file.mimetype, + src: inputFile.file.base64, + username, + }, + ]) + }) + + const testCases = [ + `${testBaseFileName}(2).${testFileExtension}`, + `${testBaseFileName}(4).${testFileExtension}`, + `${testBaseFileName}(6).${testFileExtension}`, + ] + + testCases.forEach((testCase) => { + it(`uploaded file name should be "${testCase}"`, async function () { + req.body = inputFile + await router.postRoutes[route](req, res) + + validateTest({ + inputFile, + expectedFilename: testCase, + responseBody: res.body, + }) + }) + }) + }) }) From b82d9702d2c7e09647a9055efc075ebe2106f128 Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 18:25:46 -0600 Subject: [PATCH 15/24] Added: add'l dup tests --- test/server/routes/index.test.js | 76 ++++++++++++++++++++++++++++++++ 1 file changed, 76 insertions(+) diff --git a/test/server/routes/index.test.js b/test/server/routes/index.test.js index 1d56925..0c743ea 100644 --- a/test/server/routes/index.test.js +++ b/test/server/routes/index.test.js @@ -142,8 +142,84 @@ describe('routes', function () { inputFile, expectedFilename: `${testBaseFileName}(1).${testFileExtension}`, responseBody: res.body, + }) + }) + + describe('Uploading duplicate files, non-duplicate syntax file names', function () { + + it('uploads duplicate file, original does not exists in db', async function () { + // adding original file to db, but with the syntax of a duplicate filename + fileRepository.insertFile({ + description: inputFile.description, + filename: `${testBaseFileName}(1).${testFileExtension}`, + mimetype: inputFile.file.mimetype, + src: inputFile.file.base64, + username, + }) + + // uploading original file + req.body = inputFile + await router.postRoutes[route](req, res) + + validateTest({ + inputFile, + expectedFilename: inputFile.file.name, + responseBody: res.body, + }) + }) + + it('uploads 1st duplicate, original exists in db', async function () { + // adding original file to db + fileRepository.insertFile({ + description: inputFile.description, + filename: inputFile.file.name, + mimetype: inputFile.file.mimetype, + src: inputFile.file.base64, + username, + }) + + // uploading a duplicate + req.body = inputFile + await router.postRoutes[route](req, res) + + validateTest({ + inputFile, + expectedFilename: `${testBaseFileName}(1).${testFileExtension}`, + responseBody: res.body, + }) + }) + + it('uploads 2nd duplicate file, original exists in db', async function () { + // inserting previous uploads + fileRepository.bulkInsertFiles([ + { + description: inputFile.description, + filename: inputFile.file.name, + mimetype: inputFile.file.mimetype, + src: inputFile.file.base64, + username, + }, + { + description: inputFile.description, + filename: `${testBaseFileName}(1).${testFileExtension}`, + mimetype: inputFile.file.mimetype, + src: inputFile.file.base64, + username, + }, + ]) + + // uploading original file + req.body = inputFile + await router.postRoutes[route](req, res) + + validateTest({ + inputFile, + expectedFilename: `${testBaseFileName}(2).${testFileExtension}`, + responseBody: res.body, + }) }) }) + }) describe('Upload duplicates fill gaps, original file named', function () { From a6512faed329943f4dea2c367d9693595b4a73c0 Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 18:29:49 -0600 Subject: [PATCH 16/24] Added: test for filenaming edgecases --- test/server/routes/index.test.js | 156 +++++++++++++++++++++++++++---- 1 file changed, 136 insertions(+), 20 deletions(-) diff --git a/test/server/routes/index.test.js b/test/server/routes/index.test.js index 0c743ea..d6d96df 100644 --- a/test/server/routes/index.test.js +++ b/test/server/routes/index.test.js @@ -125,26 +125,6 @@ describe('routes', function () { }) }) - it('uploads 1st duplicate, original exists in db for user with 100+ duplicate files', async function () { - fileRepository.insertFile({ - description: inputFile.description, - filename: inputFile.file.name, - mimetype: inputFile.file.mimetype, - src: inputFile.file.base64, - username, - }) - - // uploading a duplicate - req.body = inputFile - await router.postRoutes[route](req, res) - - validateTest({ - inputFile, - expectedFilename: `${testBaseFileName}(1).${testFileExtension}`, - responseBody: res.body, - }) - }) - describe('Uploading duplicate files, non-duplicate syntax file names', function () { it('uploads duplicate file, original does not exists in db', async function () { @@ -220,6 +200,75 @@ describe('routes', function () { }) }) + describe('Uploading duplicate files, names in duplicate syntax', function () { + beforeEach(function () { + inputFile = buildUploadData({ baseFileName: 'test(1)', fileExt: 'png' }) + }) + + it('uploads orignal file', async function () { + req.body = inputFile + await router.postRoutes[route](req, res) + + + validateTest({ + inputFile, + expectedFilename: inputFile.file.name, + responseBody: res.body, + }) + }) + + it('uploads 1st duplicate file (2nd uploaded)', async function () { + // inserting original into DB + fileRepository.insertFile({ + description: inputFile.description, + filename: 'test(1).png', + mimetype: inputFile.file.mimetype, + src: inputFile.file.base64, + username, + }) + + // uploading duplicate of test(1).png + req.body = inputFile + await router.postRoutes[route](req, res) + + validateTest({ + inputFile, + expectedFilename: 'test(1)(1).png', + responseBody: res.body, + }) + }) + + it('uploads 2nd duplicate file (3rd uploaded)', async function () { + // inseting files into DB + fileRepository.bulkInsertFiles([ + { + description: inputFile.description, + filename: 'test(1).png', + mimetype: inputFile.file.mimetype, + src: inputFile.file.base64, + username, + }, + { + description: inputFile.description, + filename: 'test(1)(1).png', + mimetype: inputFile.file.mimetype, + src: inputFile.file.base64, + username, + }, + ]) + + // uploading 3rd file (2nd duplicate) of test(1).png + req.body = inputFile + await router.postRoutes[route](req, res) + + validateTest({ + inputFile, + expectedFilename: 'test(1)(2).png', + responseBody: res.body, + }) + }) + }) + }) describe('Upload duplicates fill gaps, original file named', function () { @@ -279,4 +328,71 @@ describe('routes', function () { }) }) }) + + describe('Upload duplicates fill gaps, original file name is already in duplicate syntax', function () { + this.beforeAll(async function () { + const baseTestFilename = `${testBaseFileName}(1)` + route = '/api/files' + inputFile = buildUploadData({ baseFileName: baseTestFilename }) + db.resetToSeed() + fileRepository.bulkInsertFiles([ + { + description: inputFile.description, + filename: inputFile.file.name, + mimetype: inputFile.file.mimetype, + src: inputFile.file.base64, + username, + }, + { + description: inputFile.description, + filename: `${baseTestFilename}(2).${testFileExtension}`, + mimetype: inputFile.file.mimetype, + src: inputFile.file.base64, + username, + }, + { + description: inputFile.description, + filename: `${baseTestFilename}(3).${testFileExtension}`, + mimetype: inputFile.file.mimetype, + src: inputFile.file.base64, + username, + }, + { + description: inputFile.description, + filename: `${baseTestFilename}(5).${testFileExtension}`, + mimetype: inputFile.file.mimetype, + src: inputFile.file.base64, + username, + }, + { + description: inputFile.description, + filename: `${baseTestFilename}(8).${testFileExtension}`, + mimetype: inputFile.file.mimetype, + src: inputFile.file.base64, + username, + }, + ]) + }) + + const testCases = [ + `${testBaseFileName}(1)(1).${testFileExtension}`, + `${testBaseFileName}(1)(4).${testFileExtension}`, + `${testBaseFileName}(1)(6).${testFileExtension}`, + `${testBaseFileName}(1)(7).${testFileExtension}`, + `${testBaseFileName}(1)(9).${testFileExtension}`, + ] + + testCases.forEach((testCase) => { + it(`uploaded file name should be "${testCase}"`, async function () { + req.body = inputFile + await router.postRoutes[route](req, res) + + validateTest({ + inputFile, + expectedFilename: testCase, + responseBody: res.body, + }) + }) + }) + }) }) From abcfcd3e5d7190d1329b77bd3e666fa4c72fdeb0 Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 21:36:02 -0600 Subject: [PATCH 17/24] Updated: fileRepository and server index, display paginated results --- src/db/repositories/fileRepository.js | 52 +++++++++++++++++++++------ 1 file changed, 41 insertions(+), 11 deletions(-) diff --git a/src/db/repositories/fileRepository.js b/src/db/repositories/fileRepository.js index 15f1fe7..ef98dec 100644 --- a/src/db/repositories/fileRepository.js +++ b/src/db/repositories/fileRepository.js @@ -19,9 +19,24 @@ class FileRepository { mimetype TEXT NOT NULL, src TEXT NOT NULL, username TEXT NOT NULL - ) + ); + `).run() + + this.db.prepare(` + CREATE INDEX IF NOT EXISTS idx_filename ON files (filename); + `).run() + + this.db.prepare(` + CREATE INDEX IF NOT EXISTS idx_filename ON files (filename); + `).run() + + this.db.prepare(` + CREATE INDEX IF NOT EXISTS idx_username ON files (username); + `).run() + + this.db.prepare(` + CREATE INDEX IF NOT EXISTS idx_src ON files (src); `).run() - // this.db.prepare('ADD COLUMN IF NOT EXISTS') } /** @@ -95,13 +110,19 @@ class FileRepository { /** * Retrieves paginated files associated with a specific user. * @param {string} username - The username of the user. + * @param {number} size - The maximum number of files to retrieve. + * @param {number} page - The number of files to skip before starting to return records. * @returns {object} An object of file records uploaded by the user and total. */ - getUserFiles ({ username }) { + getUserFiles ({ username, page, size }) { + page = page || 1 + size = size || 30 + const offset = (page -1) * size return this.db.prepare(` SELECT * FROM files WHERE username = :username - `).all({ username }) + LIMIT :limit OFFSET :offset + `).all({ username, limit: size, offset }) } /** @@ -116,11 +137,15 @@ class FileRepository { /** * Retrieves all files stored in the database. - * @returns {Array} An array of all file records stored. + * @param {number} size - The maximum number of files to retrieve. + * @param {number} page - The number of files to skip before starting to return records.s * @returns {Array} An array of all file records. */ - getAllFiles () { - return this.db.prepare(`SELECT * FROM files`).all() + getAllFiles ({ page=1, size=30 }) { + page = page || 1 + size = size || 30 + const offset = (page - 1) * size + return this.db.prepare(`SELECT * FROM files LIMIT :limit OFFSET :offset`).all({ limit: size, offset }) } /** @@ -143,7 +168,7 @@ class FileRepository { */ getOriginalFile ({ username, filename, base64 }) { return this.db.prepare(` - SELECT * + SELECT id, filename, src FROM files WHERE filename == :filename @@ -157,27 +182,32 @@ class FileRepository { }) } - // getCountMatchingCriteria /** * Retrieves all files similar to the provided criteria. Used to get the list of duplicate document * @param {Object} criteria - An object containing the criteria (username, filename, fileExt, and base64) for finding similar files. * @returns {Array} An object with records similar to the provided criteria and total matching in DB. */ - findAllLike ({ username, filename, fileExt, base64 }) { + findAllLike ({ username, filename, fileExt, base64, size=30, page=1 }) { + page = page || 1 + size = size || 30 + const offset = (page - 1) * size return this.db.prepare(` - SELECT * + SELECT id, filename, src FROM files WHERE filename == :filename OR filename LIKE :similarFilename AND src == :src AND username == :username ORDER BY filename ASC + LIMIT :limit OFFSET :offset `) .all({ filename: `${filename}.${fileExt}`, similarFilename: `${filename}(%.${fileExt}`, src: base64, username, + offset, + limit: size, }) } } From 31b618c07986a8ea977f3cf51451777ccc238f2a Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 21:36:59 -0600 Subject: [PATCH 18/24] Added: pagination tests --- test/server/routes/index.test.js | 117 +++++++++++++++++++++++++------ 1 file changed, 97 insertions(+), 20 deletions(-) diff --git a/test/server/routes/index.test.js b/test/server/routes/index.test.js index d6d96df..6ec9028 100644 --- a/test/server/routes/index.test.js +++ b/test/server/routes/index.test.js @@ -1,6 +1,7 @@ const { expect } = require('chai') const db = require('db') const buildRoutes = require('server/routes') +// const buildFileRoutes = require('server/routes/files') const buildMockRouter = require('../buildMockRouter') const FileRepository = require('db/repositories/fileRepository') @@ -9,7 +10,7 @@ describe('routes', function () { const testBaseFileName = 'bobby-tables' const testFileExtension = 'jpg' - const username = 'testuser' + const testUsername = 'testuser' const fileRepository = new FileRepository(db.instance) function buildUploadData ({ @@ -36,9 +37,36 @@ describe('routes', function () { expect(responseBody.src).to.equal(inputFile.file.base64) } + function generateTestFiles ({ baseFileName, count, missingNums=[], username }) { + const initFile = buildUploadData({ baseFileName }) + const files = [ + { + description: initFile.description, + filename: initFile.file.name, + mimetype: initFile.file.mimetype, + src: initFile.file.base64, + username: username || testUsername, + }, + ] + for (let i= 1; i < count; i++) { + if (!missingNums.includes(i)) { + const f = buildUploadData({ baseFileName: `${baseFileName}(${i})` }) + files.push({ + description: f.description, + filename: f.file.name, + mimetype: f.file.mimetype, + src: f.file.base64, + username: username || testUsername, + }) + } + } + return [initFile, files] + } + beforeEach(async function () { req = { - username, + username: testUsername, + query: {}, } res = { send: (value) => { res.body = value }, @@ -49,6 +77,7 @@ describe('routes', function () { describe('GET /api/files', function () { beforeEach(async function () { route = '/api/files' + db.resetToSeed() totalSeedTestFiles = fileRepository.getTotal() }) @@ -62,6 +91,33 @@ describe('routes', function () { await router.getRoutes[route](req, res) expect(res.body).to.have.length(0) }) + + it('display 1st page of user files', async function () { + req.username = 'test1' + const [, testFiles] = generateTestFiles({ + baseFileName: 'test1', + count: 50, + username: req.username, + }) + fileRepository.bulkInsertFiles(testFiles) + await router.getRoutes[route](req, res) + expect(res.body).to.have.length(30) + }) + + it('display 1st page of user files', async function () { + req.username = 'test1' + const [, testFiles] = generateTestFiles({ + baseFileName: 'test1', + count: 50, + username: req.username, + }) + req.query = { + page: 2, + } + fileRepository.bulkInsertFiles(testFiles) + await router.getRoutes[route](req, res) + expect(res.body).to.have.length(20) + }) }) describe('POST /api/files', function () { @@ -69,6 +125,7 @@ describe('routes', function () { route = '/api/files' inputFile = buildUploadData({}) db.resetToSeed() + totalSeedTestFiles = fileRepository.getTotal() }) it('uploads a file and returns its metadata', async function () { @@ -100,7 +157,7 @@ describe('routes', function () { validateTest({ inputFile, - expectedFilename:input.file.name, + expectedFilename: input.file.name, responseBody: res.body, }) }) @@ -111,7 +168,7 @@ describe('routes', function () { filename: inputFile.file.name, mimetype: inputFile.file.mimetype, src: inputFile.file.base64, - username, + username: testUsername, }) // uploading a duplicate @@ -134,7 +191,7 @@ describe('routes', function () { filename: `${testBaseFileName}(1).${testFileExtension}`, mimetype: inputFile.file.mimetype, src: inputFile.file.base64, - username, + username: testUsername, }) // uploading original file @@ -155,7 +212,7 @@ describe('routes', function () { filename: inputFile.file.name, mimetype: inputFile.file.mimetype, src: inputFile.file.base64, - username, + username: testUsername, }) // uploading a duplicate @@ -177,14 +234,14 @@ describe('routes', function () { filename: inputFile.file.name, mimetype: inputFile.file.mimetype, src: inputFile.file.base64, - username, + username: testUsername, }, { description: inputFile.description, filename: `${testBaseFileName}(1).${testFileExtension}`, mimetype: inputFile.file.mimetype, src: inputFile.file.base64, - username, + username: testUsername, }, ]) @@ -224,7 +281,7 @@ describe('routes', function () { filename: 'test(1).png', mimetype: inputFile.file.mimetype, src: inputFile.file.base64, - username, + username: testUsername, }) // uploading duplicate of test(1).png @@ -246,14 +303,14 @@ describe('routes', function () { filename: 'test(1).png', mimetype: inputFile.file.mimetype, src: inputFile.file.base64, - username, + username: testUsername, }, { description: inputFile.description, filename: 'test(1)(1).png', mimetype: inputFile.file.mimetype, src: inputFile.file.base64, - username, + username: testUsername, }, ]) @@ -269,6 +326,26 @@ describe('routes', function () { }) }) + // WIP + it.skip('uploads 1st duplicate, original exists in db for user with 100+ duplicate files', async function () { + const skipDupNums = [30, 49] + const [initFile, testFiles] = generateTestFiles({ + baseFileName: 'test1', + count: 100, + missingNums: skipDupNums, + }) + fileRepository.bulkInsertFiles(testFiles) + + // uploading a duplicate + req.body = initFile + await router.postRoutes[route](req, res) + + validateTest({ + inputFile: initFile, + expectedFilename: `test1(30).${testFileExtension}`, + responseBody: res.body, + }) + }) }) describe('Upload duplicates fill gaps, original file named', function () { @@ -283,28 +360,28 @@ describe('routes', function () { filename: inputFile.file.name, mimetype: inputFile.file.mimetype, src: inputFile.file.base64, - username, + username: testUsername, }, { description: inputFile.description, filename: `${testBaseFileName}(1).${testFileExtension}`, mimetype: inputFile.file.mimetype, src: inputFile.file.base64, - username, + username: testUsername, }, { description: inputFile.description, filename: `${testBaseFileName}(3).${testFileExtension}`, mimetype: inputFile.file.mimetype, src: inputFile.file.base64, - username, + username: testUsername, }, { description: inputFile.description, filename: `${testBaseFileName}(5).${testFileExtension}`, mimetype: inputFile.file.mimetype, src: inputFile.file.base64, - username, + username: testUsername, }, ]) }) @@ -341,35 +418,35 @@ describe('routes', function () { filename: inputFile.file.name, mimetype: inputFile.file.mimetype, src: inputFile.file.base64, - username, + username: testUsername, }, { description: inputFile.description, filename: `${baseTestFilename}(2).${testFileExtension}`, mimetype: inputFile.file.mimetype, src: inputFile.file.base64, - username, + username: testUsername, }, { description: inputFile.description, filename: `${baseTestFilename}(3).${testFileExtension}`, mimetype: inputFile.file.mimetype, src: inputFile.file.base64, - username, + username: testUsername, }, { description: inputFile.description, filename: `${baseTestFilename}(5).${testFileExtension}`, mimetype: inputFile.file.mimetype, src: inputFile.file.base64, - username, + username: testUsername, }, { description: inputFile.description, filename: `${baseTestFilename}(8).${testFileExtension}`, mimetype: inputFile.file.mimetype, src: inputFile.file.base64, - username, + username: testUsername, }, ]) }) From 2928b88db95406c9bf1c2ae55b6a3b3a9a2572ce Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 21:37:33 -0600 Subject: [PATCH 19/24] Updated: db test file --- test/server/db.test.js | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/test/server/db.test.js b/test/server/db.test.js index 6770550..c7580fa 100644 --- a/test/server/db.test.js +++ b/test/server/db.test.js @@ -12,15 +12,16 @@ describe('db', function () { res = db.instance .prepare(` INSERT INTO files - (description, filename, mimetype, src) + (description, filename, mimetype, src, username) VALUES - (@description, @filename, @mimetype, @src) + (@description, @filename, @mimetype, @src, @username) `) .run({ description: 'My file', filename, mimetype: 'text/plain', src: 'abc', + username: 'test', }) expect(res).to.be.ok }) From 4a0f9dd47db4bfd0994b1453c6d2673f49c6b3b6 Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 21:39:24 -0600 Subject: [PATCH 20/24] Updated: endpoint get files, display paginated results --- src/server/routes/index.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/server/routes/index.js b/src/server/routes/index.js index f5c4f33..ab8b5e1 100644 --- a/src/server/routes/index.js +++ b/src/server/routes/index.js @@ -14,7 +14,8 @@ function buildRoutes (router) { }) router.get('/api/files', async (req, res) => { - const files = fileRepository.getUserFiles({ username: req.username }) + const { page = 0, size = 30 } = req.query + const files = fileRepository.getUserFiles({ username: req.username, page: parseInt(page), size: parseInt(size) }) return res.send(files) }) @@ -45,7 +46,6 @@ function buildRoutes (router) { // If there are multiple duplicate db files, finding the next duplicate number (fill the gaps) let i = 0 while (i < existingFiles.length - 1) { - iters += 1 // if current file name identifier does not match the current iteration, breakout out of loop. Missing a duplicate, gap should be filled if (existingFiles[i].filename !== `${fileUploader.fileBaseName}(${i+1}).${fileUploader.fileExt}`) { break From 044eba660da14f3565a4e26f9d2b91b8d21a3d92 Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 21:41:43 -0600 Subject: [PATCH 21/24] Updated: seed.db file --- src/db/seed.db | Bin 36864 -> 36864 bytes 1 file changed, 0 insertions(+), 0 deletions(-) diff --git a/src/db/seed.db b/src/db/seed.db index 990039931309ba28d6ae3757ec4090e3422ae389..7c27ad7102a735c0905582295e0c215a03ee24db 100644 GIT binary patch delta 503 zcmZozz|^pSX+joTIRoEGzVeNQ?0l1N@&$1D z1f--ECl_TFlw{`TDTKI2geds=197O2j}8}135-t5%t_5l%uOYvE;lnbwWP8DkGgn> zk;O&Hgv=@}PA$T;6UAwoybKHstlXRo{LA>g_|EYqaC2@f{LaPOc+8xYlOa`JvAH{1 zcC(YT9kXa?zHgFYL{2DB14|AIJ43pnyu0k=9=TY-lGNf7kX=>G>1&T0i=4trC54Q^dV7`JU delta 365 zcmZozz|^pSX+jp;O$L5#zMGo`6*BoIm-7X2H3l)Vi;IghHZxDY%%?q>gI}jUCAB!Y zD6^m>Ge1uu#5E#BAuqo~A+IziM~6!R4AL@lQu7jXQ?bkDX6B}rR2JZnEiOt%)dn_G zlb3;kftA~gfqy+edko)8z7}q?jfE=QJdMZnSUDI{K~n+{<0d3v)NKG<@L)07`gbF8}}l From 0e9485310bb7b678d1bd56c65881fe4119bdace5 Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 21:50:57 -0600 Subject: [PATCH 22/24] Added: todo for fix --- src/server/routes/index.js | 1 + 1 file changed, 1 insertion(+) diff --git a/src/server/routes/index.js b/src/server/routes/index.js index ab8b5e1..30c27f7 100644 --- a/src/server/routes/index.js +++ b/src/server/routes/index.js @@ -44,6 +44,7 @@ function buildRoutes (router) { filename = fileUploader.generateDuplicateFilename(dbOriginalFile.filename) } else { // If there are multiple duplicate db files, finding the next duplicate number (fill the gaps) + // TODO: fix for 10+ duplicate files let i = 0 while (i < existingFiles.length - 1) { // if current file name identifier does not match the current iteration, breakout out of loop. Missing a duplicate, gap should be filled From 7f230cffe02d812ee3c948211aeda0746f95cec3 Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Thu, 14 Mar 2024 21:53:05 -0600 Subject: [PATCH 23/24] Added: notes to README --- README.md | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/README.md b/README.md index dbb21df..cbd534e 100644 --- a/README.md +++ b/README.md @@ -52,3 +52,14 @@ You can add new files! Screen Shot 2020-05-07 at 12 30 03 PM That's all it does without your help! + + +#### Notes +I configured the application to only display files uploaded by the user +If there was functionality on the front end to look at files for a different user, that is possible +Some of changes I would make if this was truly for production: +1) Use pagination to display the files, have buttons for next, back and clicking on the individual pages +2) Allow user to sort by title and date uploaded (asc or desc) + - add updated_at and created_at columns to "files" table to facilitate +3) Have table for user and add `user_id` colmumn in "files" table +4) Implement authentication middleware and not get username from os From 98772ed6251446d95cd091c937ce7a8aafc585d0 Mon Sep 17 00:00:00 2001 From: ccornali611 Date: Fri, 15 Mar 2024 11:12:17 -0600 Subject: [PATCH 24/24] small fixes --- src/db/index.js | 3 +++ src/db/repositories/fileRepository.js | 13 ++++++++++--- src/server/routes/index.js | 2 +- test/server/routes/index.test.js | 1 - 4 files changed, 14 insertions(+), 5 deletions(-) diff --git a/src/db/index.js b/src/db/index.js index 0380a88..7cc2e65 100644 --- a/src/db/index.js +++ b/src/db/index.js @@ -1,6 +1,7 @@ const fs = require('fs') const path = require('path') const Database = require('better-sqlite3') +const FileRepository = require('db/repositories/fileRepository') const SEED_DB = path.join(__dirname, 'seed.db') const DEV_DB = path.join(__dirname, 'dev.db') @@ -27,5 +28,7 @@ module.exports = { resetToSeed () { copySeedDB({ force: true }) db = new Database(localDBPath) + const fileRepository = new FileRepository(db) + fileRepository.createTable() }, } diff --git a/src/db/repositories/fileRepository.js b/src/db/repositories/fileRepository.js index ef98dec..0497fcb 100644 --- a/src/db/repositories/fileRepository.js +++ b/src/db/repositories/fileRepository.js @@ -21,7 +21,15 @@ class FileRepository { username TEXT NOT NULL ); `).run() - + const schema = this.db.prepare("PRAGMA table_info('files')").all() + const usernameColumnExists = schema.some((column) => column.name === 'username') + + if (!usernameColumnExists) { + this.db.prepare(` + ALTER TABLE files ADD COLUMN username TEXT; + `).run() + } + // add column for duplicate count and orig_hashed_filename this.db.prepare(` CREATE INDEX IF NOT EXISTS idx_filename ON files (filename); `).run() @@ -53,7 +61,6 @@ class FileRepository { `) .run({ description, filename, mimetype, src, username }) const dbFile = this.getFileById(result.lastInsertRowid) - // console.log(`[INFO] FileRepository.insertFile(): "${filename}"`, dbFile) return dbFile } @@ -181,7 +188,7 @@ class FileRepository { src: base64, }) } - + /** * Retrieves all files similar to the provided criteria. Used to get the list of duplicate document * @param {Object} criteria - An object containing the criteria (username, filename, fileExt, and base64) for finding similar files. diff --git a/src/server/routes/index.js b/src/server/routes/index.js index 30c27f7..03c68c5 100644 --- a/src/server/routes/index.js +++ b/src/server/routes/index.js @@ -14,7 +14,7 @@ function buildRoutes (router) { }) router.get('/api/files', async (req, res) => { - const { page = 0, size = 30 } = req.query + const { page = 1, size = 30 } = req.query const files = fileRepository.getUserFiles({ username: req.username, page: parseInt(page), size: parseInt(size) }) return res.send(files) }) diff --git a/test/server/routes/index.test.js b/test/server/routes/index.test.js index 6ec9028..e94dbe6 100644 --- a/test/server/routes/index.test.js +++ b/test/server/routes/index.test.js @@ -1,7 +1,6 @@ const { expect } = require('chai') const db = require('db') const buildRoutes = require('server/routes') -// const buildFileRoutes = require('server/routes/files') const buildMockRouter = require('../buildMockRouter') const FileRepository = require('db/repositories/fileRepository')