Skip to content

Playing with Next.js and building a sample file upload app for educational purposes

License

Notifications You must be signed in to change notification settings

lmammino/next-sample-file-upload-app

Repository files navigation

Luciano Mammino - 2020-12-06

An attempt to build a simple Next.js application to manage file uploads to a folder in just 48 hours.

Installation

This is a next.js application. To run it locally you have to first install all the project dependencies with yarn install, then you can run it in dev mode with yarn dev.

The app will start at http://localhost:3000/

You can also run the the test suite (lint and unit tests) with yarn test

Security

  • This application does NOT provide any authentication/authorization mechanism.
  • This application does NOT provide any other mechanism (e.g. CSRF) to prevent malicious bodies to retrieve the list of files directly (curl http://localhost:3000/api/fs/ls | jq . will show all the available files). Of course, this doesn't make the app ready to be shipped to a remote server.
  • The application validates file uploads (max file size / file type) on both frontend and backend.
  • On the backend, the upload is carried over only if files are valid. Data is initially stored in a temp folder and only moved to the final destination if all the validation rules pass. Mime type validation is done as soon as the file is started to be uploaded, while file size is checked in a streaming fashion: as soon as the stream goes over the maximum expected size, the stream is interrupted.
  • The frontend application does not set HTML dynamically (Normal react data binding should take care of most sanitization needs).
  • During the upload, the original file name is used to compute the actual destination name in the final data folder. The storage is kept flat. In order to avoid attempts to escape the final destination folder, every / or \ in the original file name are replaced to _.
  • If a file already exists with a given name, the final file is given a random suffix to avoid overriding the original file.

Improvements

Some potential improvements that are currently missing:

  • Some dialogs (errors, confirm messages) have been implemented using alert and confirm. This is not ideal for many reasons (bad UI/UX, blocking the event loop, etc.), so with more time I would have used dedicated react components to handle these interactions (e.g. a toast component like react-toast-notifications).
  • With more time I would have added some sort of Authentication mechanism or at least some CSRF mechanism to make the backend API more secure.
  • Backend tests are missing.
  • E2E tests are missing (something like Cypress might have been nice here).
  • For simplicity of implementation, the UI will always have a partial view of the content of the data folder. It is loaded at startup and then kept in sync while the various delete or upload operations happen. If files are added or deleted manually (or from another UI) these changes won't be immediately reflected in the UI. It could have been nice to explore a different architecture using websockets to keep the UI always in sync. A simpler alternative (not implemented here), could be to poll the API layer to retrieve the list of files at given intervals of time rather than doing it only once at startup time.
  • With a bit more time investment, it would have been nice to implement support for uploading files via drag and drop.
  • For simplicity and speed of development I kept the bulk of the business logic in the Home component. This is to all effects a mega-component and should probably be broken down into smaller and more reusable components.
  • Tests have been written in a rush and they are not the most polished. With a bit more time it would be nice to refactor them and make them more content-independent by using data-testid attributes rather than matching by content.

Libraries

  • next: Next.js, to quickly scaffold a React application with backend APIs.
  • prop-types: To validate props being passed to components.
  • pretty-bytes: To easily convert byte units into more readable strings (e.g. 1000 -> 1 kb ).
  • formidable: To handle multipart form submissions in a streaming way on the backend.
  • eslint: To make sure coding style is consistent (using StandardJS style) and to spot common JavaScript mistakes.
  • jest: Test runner.
  • babel-jest: To be able to transpile files at runtime while running Jest.
  • @testing-library/jest-dom: To simplify writing React tests with Jest.
  • identity-obj-proxy: a utility to mock CSS modules in tests.

API

Note: the API will manage files in the data folder (upload, delete, list operations). The file system is effectively used as a poor man's database from the API layer.

The API layout is not particularly RESTful. I could have used some more advanced capabilities of Next.js to have more freedom on how to structure the API, but I went for speed rather than for compliance with the common RESTful conventions.

GET /api/fs/ls

Lists all the files available in the data folder.

Example

curl http://localhost:3000/api/fs/ls | jq .
200 OK
{
  "files": [
    {
      "name": "1999229.png",
      "size": 43090
    },
    {
      "name": "457cc040ca23e98d84f70d215111255c copy.jpg",
      "size": 11572
    },
    {
      "name": "9b822e70053cb1f65dfd7a99fcd6d98a-best-quality-badge-by-vexels copy.png",
      "size": 28655
    }
  ]
}  

DELETE /api/fs/delete/:filename

Deletes a file from the filesystem

Input

  • filename (path parameter): the name of the file to delete (e.g. 1999229.png)

Example

curl http://localhost:3000/api/fs/delete/1999229.png | jq .
200 OK
{
  "success": true,
  "deleted": "1999229.png"
}

POST /api/fs/upload

Uploads a new file to the file system

Input

  • file (body parameter): A file attachment using multipart/form-data for encoding

Example

curl --form file='@filename.jpg' http://localhost:3000/api/fs/upload | jq .
201 CREATED
{
  "name": "filename.jpg",
  "size": 13234444
}

Other notes

About

Playing with Next.js and building a sample file upload app for educational purposes

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published