Based on Dcycle Node.js starterkit.
- About
- Strategies, credentials, and accounts
- Quickstart
- Let's Encrypt on a server
- Creating new users
- Adding arbitrary unique and non-unique fields, such as email addresses, to users
- Sending emails
- Dcycle Node Starterkit design patterns
- Component-based modular system
- Some components require initialization
- Components can require dependencies at runtime
- Defining which modules, and their configuration, to load via a yaml file
- Defining unversioned configuration for environment-specific configuration and sensitive data
- Components's class names are the same as their directory names but start with an uppercase letter
- Plugins: how modules can share information with each other
- Components can define classes
- The Node.js command line interface (CLI)
- MongoDB crud (create - read - update - delete)
- Mongoose vs MongoDB
- Logging in with GitHub
- GitHub Apps
- Security tokens
- REST API
- Access to content by permission
- Whatsapp Message Send/Recieve Functionality
- Typechecking
- Troubleshooting
- Resources
This project is a quick starter for Node applications on Docker. We have implemented a very simple chat application with authentication (see "Resources", below) using Socket.io, Express, and MongoDB.
This project uses Passport for authentication along with the "Username/password" strategy.
You can create a new account or regenerate a random password for an existing account by typing, on the command line:
./scripts/reset-password.sh some-username
This means that if your appliation is restarted or crashes, you won't have to log back in.
Install Docker and run:
./scripts/deploy.sh
This will give you a URL, username and password.
Now log on using the the credentials provided.
You will be able to use a simple chat application, and log out.
(This does not apply to local development, only to publicly-accessible servers.)
We will follow the instructions in the following blog posts:
- Letsencrypt HTTPS for Drupal on Docker, October 03, 2017, Dcycle Blog
- Deploying Letsencrypt with Docker-Compose, October 06, 2017, Dcycle Blog
Here are the exact steps:
-
Figure out the IP address of your server, for example 1.2.3.4.
-
Make sure your domain name, for example example.com, resolves to 1.2.3.4. You can test this by running:
ping example.com
You should see something like:
PING example.com (1.2.3.4): 56 data bytes
64 bytes from 1.2.3.4: icmp_seq=0 ttl=46 time=28.269 ms
64 bytes from 1.2.3.4: icmp_seq=1 ttl=46 time=25.238 ms
Press control-C to get out of the loop.
- Run your instance (./scripts/deploy.sh)
- edit the file .env
- replace the line VIRTUAL_HOST=localhost with VIRTUAL_HOST=example.com
- Run ./scripts/deploy.sh again
If you have Let's Encrypt already set up for another project on the same server, move on to "Figure out the network name", below. Otherwise, set up Let's Encrypt as per the above blog posts:
mkdir -p "$HOME"/certs
docker run -d -p 80:80 -p 443:443 \
--name nginx-proxy \
-v "$HOME"/certs:/etc/nginx/certs:ro \
-v /etc/nginx/vhost.d \
-v /usr/share/nginx/html \
-v /var/run/docker.sock:/tmp/docker.sock:ro \
--label com.github.jrcs.letsencrypt_nginx_proxy_companion.nginx_proxy \
--restart=always \
jwilder/nginx-proxy
docker run -d \
--name nginx-letsencrypt \
-v "$HOME"/certs:/etc/nginx/certs:rw \
-v /var/run/docker.sock:/var/run/docker.sock:ro \
--volumes-from nginx-proxy \
--restart=always \
jrcs/letsencrypt-nginx-proxy-companion
Figure out the network name
docker network ls
It is something like "starterkit_node_default".
Connect your network and restart the Let's Encrypt container:
docker network connect starterkit_node_default nginx-proxy
docker restart nginx-letsencrypt
After 120 seconds the security certificate should work. Now your site should work with LetsEncrypt.
You can run:
./scripts/reset-password.sh some-new-user
By default users only have (and must have) a username which needs to be unique.
It is also possible to add other information to user records.
Here is an example:
By default an "admin" user exists, and we can see its record in the database by running:
./scripts/mongo-cli.sh
...
show dbs
use login
show collections
db.userInfo.find();
This will show the record associated with the user admin and potentially other users.
In a new terminal window, create a new user to demonstrate unique vs. non-unique fields:
./scripts/reset-password.sh some-new-user
Back in the Mongo CLI, you will now see two users.
In yet another terminal window, open a CLI for the Node app.
./scripts/node-cli.sh
To add a non-unique field to both users, run:
await app.c('authentication')
.addNonUniqueFieldToUser('admin', 'hello', 'world');
await app.c('authentication')
.addNonUniqueFieldToUser('some-new-user', 'hello', 'world');
The same code can be used to modify an existing field. The code does nothing if the user does not exist.
You can also remove this field:
await app.c('authentication')
.removeFieldFromUser('admin', 'hello');
await app.c('authentication')
.removeFieldFromUser('some-new-user', 'hello');
Back the Node CLI, you can add a unique field to the admin user:
await app.c('authentication')
.addUniqueFieldToUser('admin', 'hello', 'world');
Now, if you try to add the same field and value to another user, you will get an error:
await app.c('authentication')
.addUniqueFieldToUser('some-new-user', 'hello', 'world');
> Uncaught:
Error: Cannot add unique field hello to user some-new-user with value world because a different user, admin, already has that value in the same field.
at /usr/src/app/app/authentication/index.js:154:15
./scripts/node-cli.sh
const u = await app.c('authentication').user('admin');
app.c('authentication').userFieldValue(u, 'view-content-permission-xyz', '0');
Your node application can send emails using SMTP. For that you need an SMTP server. In development, we use MailHog. Here is how it works:
Start your instance using ./scripts/deploy.sh
.
Once you have a running instance you will have access to mailhog.
You can send an email by running:
./scripts/node-cli.js
Then, on the prompt:
app.component('./mail/index.js').sendMailInDefaultServer({from: '[email protected]', to: '[email protected]', subject: 'Hello World', html: '<p>Hello</p>', text: 'Hello'}, (error, info) => { console.log(error); console.log(info); });
Then, you can run:
docker compose ps
And visit the URL for MailHog, and you will see your message.
If you would like to use a real SMTP mail server, for production for example, then create a new file ./app/config/unversioned.yml
based on ./app/config/unversioned.example.yml
, and in the myServer section, put your actual SMTP information. The ./app/config/unversioned.example.yml
is not in version control, so you need to edit it directly on your production server.
In your own project, you are welcome to delete everything in ./app/code except ./app/code/server.js and put your own code in ./app/code/server.js.
If you are interested in keeping the structure of the current project, here are some design patterns we have used to make things easier.
We have split our code in a series of components which are our custom node modules; they are all singleton class objects. The simplest one is ./app/code/random/index.js. It is self-contained and self explanatory; it serves to make random numbers.
You can try it by running:
echo 'app.c("random").random()' | ./scripts/node-cli.sh
Components like ./app/code/database/index.js require initialization before use. That is why ./app/code/server.js calls app.init() before app.run(). app.init() initializes all components that need to be initialized before the application can be run.
Some components, like ./app/code/chatWeb/index.js, require that other components be initiliazed before they themselves can bre initialized and eventually run.
In the case of ChatWeb, its dependency chain is as follows:
- ChatWeb depends on Express and ChatApi
- Express has no dependencies
- ChatApi depends on Express and Chat
- Chat depends on Database and BodyParser
- BodyParser depends on Express
- Database depends on Env
We use a simple dependency manager, ./app/code/dependencies/index.js, to calculate the dependency chain. You can try it at:
echo "app.c('dependencies').getInOrder(['./chatWeb/index.js'], app);" | ./scripts/node-cli.sh
This should give you a result or ordered dependencies:
{
errors: [],
results: [
'./express/index.js',
'./bodyParser/index.js',
'./env/index.js',
'./database/index.js',
'./chat/index.js',
'./chatApi/index.js',
'./chatWeb/index.js'
]
}
This is used internally to initialize dependencies in the correct order. For example, in this example the database needs to be fully initialized beofre chatWeb (the web interface of our chat program) can be used.
You can change which components are used by changing the yaml file ./app/config/versioned.yml, and, optionally, ./app/config/unversioned.yml, the latter being ignored in version control.
Different modules can have configuration. For example, ChatWeb needs to know on which path it should be active. That is why you will see, in ./app/config/versioned.yml, the following:
modules:
...
./chatWeb/index.js:
path: '/'
This tells our system that we want chatWeb to load; and, furthermore, we want to tell it that its path should be '/'. You can install the chat application on a different path if you want by changing that.
Configuration can differ between environments. Here are some examples:
- The default mail server might be the included MailHog test server by default, but, on production, you'd use your own server.
- Certain components might require API keys. This can be achieved using environment variables, but you can also define unversioned configuration in ./app/config/unversioned.yml
Take a look at ./app/config/unversioned.example.yml which is an example for a file you can create called ./app/config/unversioned.example.yml.
It shows you how to change the default mail server, and include API keys if you so desire.
For example, the class defined in ./app/code/staticPath/index.js is called StaticPath. This is more than a convention: all classes must have the same name as their directory except that they start with an uppercase letter. All our code, particularly loading plugins, depends on this.
Some components, such ./dashboardApi/index.js, can request information from other components. In the case of dashboardApi, it can attempt to get all information that other components wish to expose on a dashboard. For example, Chat may want to expose the current total number of messages, and Authentication may wish to expose the total number of user account.
You can invoke plugins like this:
app.invokePlugin('dashboardApi', 'all', function(component, result) {
console.log(component + ' responds:');
console.log(result);
});
Indeed this is what DashboardApi does.
In this case, the system will look in each of its components, including its dependencies, for files that look like:
./app/code/*/plugins/dashboardApi/all.js
For example ./app/code/chat/plugins/dashboardApi/all.js fits the bill, as does ./app/code/authentication/plugins/dashboardApi/all.js, but there could eventually be others.
Some components, such as dashboardApi, can define classes:
- ./app/code/dashboardApi/src/dashboardSingleNumber.js
- ./app/code/dashboardApi/src/dashboardElement.js
Objects of these classes can be created by calling a very primitive autoloader:
const dashboardSingleNumber = app.class('dashboardApi/dashboardSingleNumber');
const myObject = new dashboardSingleNumber('hello', 100);
myObject.getTitle();
// hello
myObject.getNumber();
// 100
There are two ways to interact with Node.js:
Whether or not your application has been started using ./scripts/deploy.sh (see Quickstart, above), you can type:
docker compose run --rm node /bin/sh -c 'node'
This allows you to test Javascript in isolation and does not interact with your running application. The simplest example is running:
1 + 1;
If you want to run code against your running application once you have deployed it (see Quickstart, above), thus having access to your database, as well as any information stored in memory by your app's process, you can use the app CLI:
./scripts/node-cli.sh
We achieve this using the Node REPL (see the Resources section below for further reading on the technical aspects of this).
To demonstrate this, you can first log into your application using the credentials provided after running the ./scripts/deploy.sh, at http://0.0.0.0:8428, and you will see something like:
User(s) currently online: 1
The purpose of the app CLI is to have access to this information in your running application instance. Here is how.
./scripts/node-cli.sh
app.component('./numUsers/index.js').numUsers();
This should give you the same number of users online as you see in the web interface.
You can pipe commands to the cli, like this:
echo 'app.c("random").random()' | ./scripts/node-cli.sh
If in your code you use something like:
console.log('hello');
Then you can access this by running:
docker compose logs node
To create a record with {hello: "world"} in a collection "arbitraryCollection" in a database called "arbitraryDatabase", you can log into the node CLI (see above) and type:
await app.c('database').client().db('arbitraryDatabase').collection('arbitraryCollection').insert({hello: "world"});
{
acknowledged: true,
insertedCount: 1,
insertedIds: { '0': new ObjectId("634447e509ac94b6c97ecac3") }
}
Now you can, in a separate terminal window, log into the Mongo CLI and see what happened:
./scripts/mongo-cli.sh
Show databases by running:
show dbs;
...
arbitraryDatabase 0.000GB
...
use arbitraryDatabase
switched to db arbitraryDatabase
show collections;
arbitraryCollection
db.arbitraryCollection.find();
{ "_id" : ObjectId("634447e509ac94b6c97ecac3"), "hello" : "world" }
The ID will be different in your case but let's assume that it is 634447e509ac94b6c97ecac3.
In your Node.js code, if you know the ID, you can find your record by running:
const ObjectId = require('mongodb').ObjectID;
await app.c('database').client().db('arbitraryDatabase').collection('arbitraryCollection').find({_id: ObjectId("634447e509ac94b6c97ecac3")}).toArray();
[ { _id: new ObjectId("634447e509ac94b6c97ecac3"), hello: 'world' } ]
If you want to find all records where 'hello' == 'world', you can run:
await app.c('database').client().db('arbitraryDatabase').collection('arbitraryCollection').find({hello: "world"}).toArray();
[ { _id: new ObjectId("634447e509ac94b6c97ecac3"), hello: 'world' } ]
If you want to attach some arbitrary information to record 634447e509ac94b6c97ecac3, you can run:
/** if ObjectId is already defined do not redefine it here **/
const ObjectId = require('mongodb').ObjectID;
await app.c('database').client().db('arbitraryDatabase').collection('arbitraryCollection').updateOne({_id: ObjectId("634447e509ac94b6c97ecac3")}, {$set:{some_extra_information: {arbitrary: "extra information"}}});
The "$set" property tells mongoDB that we want to add information to the record.
Now, if you go back to the terminal window where you are connected to the MongoDB CLI, you can run:
db.arbitraryCollection.find();
{ "_id" : ObjectId("634447e509ac94b6c97ecac3"), "hello" : "world", "some_extra_information" : { "arbitrary" : "extra information" } }
You can also update collections not by ID but by property, for example:
await app.c('database').client().db('arbitraryDatabase').collection('arbitraryCollection').updateMany({hello: "world"}, {$set:{yet_more_extra_information: {arbitrary: "extra information"}}});
Now, in the command line for MongoDB, you will find:
db.arbitraryCollection.find();
{ "_id" : ObjectId("634447e509ac94b6c97ecac3"), "hello" : "world", "some_extra_information" : { "arbitrary" : "extra information" }, "yet_more_extra_information" : { "arbitrary" : "extra information" } }
We can now remove all this information in the Node CLI:
await app.c('database').client().db('arbitraryDatabase').collection('arbitraryCollection').updateMany({hello: "world"}, {$unset:{yet_more_extra_information: "", some_extra_information: ""}});
Now the extra fields are gone in the MongoDB CLI:
db.arbitraryCollection.find();
{ "_id" : ObjectId("634447e509ac94b6c97ecac3"), "hello" : "world" }
To completely delete the object you can run, in the Node CLI:
await app.c('database').client().db('arbitraryDatabase').collection('arbitraryCollection').deleteMany({hello: "world"});
Now confirm these are deleted in the MongoDB CLI:
db.arbitraryCollection.find();
# Nothing found
We use both npm mongoose and npm mongodb.
Mongodb is very unstructured and lets you do almost anything; we use it in the above example. Mongoose allows you to define schemas and is used for user storage. I find its learning curve a lot steeper (I still don't fully understand it), but the code works for storing users; and it was taken from the resources in the "Resrouces" section, below.
It is possible to log in with GitHub.
Here is how it works:
- Make sure you have a publicly-accessible, https domain, for example https://www.example.com.
- Make sure you have a GitHub account
- Fill in the form at https://github.com/settings/applications/new
- Application name: 'MY APPLICATION NAME'
- Application URL: https://www.example.com
- Application description: 'MY APPLICATION DESCRIPTION'.
- Authorization callback URL: https://www.example.com/auth/github/callback
- Enable device flow: not enabled.
- Generate a new client secret and take note of the client ID and client secret.
- Make sure you have a file called ./app/config/unversioned.yml; in the file, have a section with your client id and secret:
# This can be used for API keys or anything which differs from one
# environment to another.
---
modules:
./loginWithGitHub/index.js:
client: 'client_id'
secret: 'secret'
baseUrl: 'https://www.example.com'
Now go to https://www.example.com/auth/github and you will be able to log in with GitHub.
Logging in with GitHub will provide access to the GitHub username, email and some other data. GitHub Apps provide acecss to a lot more data, for example private repos.
Here is how to set up a GitHub App:
- Go to https://github.com/settings/apps/new
- Enter, as a name, "Example App"
- Check Request user authorization (OAuth) during installation
- Enter the homepage URL https://whatsapp-communication.dcycleproject.org
- Enter the callback URL https://whatsapp-communication.dcycleproject.org
- Generate a client secret
- Copy your client ID and client secret
At this point an authorization token will be provided to your app, allowing you to use the GitHub API endpoints.
For example, if you want your app to be able to access your visitors' public repositories, you can call:
curl -u GITHUB_USERNAME:ACCESS_TOKEN "https://api.github.com/user/repos?visibility=public"
Security tokens are used to access data, notably via the REST API.
Tokens are generated in one of two ways.
If you are logged in to the system, you can visit this URL:
/token/request
It will give you a token that lasts 5 minutes.
You can check a token's validity by logging in and running:
/token/check-valid?token=MY_TOKEN
This will tell you whether the token is valid or not, and why.
All tokens need to be associated with a user.
To find your user ID, you can log into the Mongo CLI:
./scripts/mongo-cli.sh
And run:
db.userInfo.find();
To create a token for a given user for 60 seconds, log into the node cli:
./scripts/node-cli.sh
And run:
app.c('token').token('some-user-id', 60, {arbitrary: 'options'})
You can verify that the token is valid by typing:
await app.c('token').tokenStringToObject(t).toObjectAboutValidity();
Tokens are not revocable.
A REST API is defined at the following endpoing:
/api/v1
If you simply visit /api/v1, you will see documentation about the API.
Only endpoints that publicly accessible are currently supported, for example:
/api/v1/endpoints
Sometimes, only certain authenticated users should have access to certain content. That's what the restrictedByPermission module does. Here's how it works.
By default files of app/private/restricted-by-permission/permission-{permissionId}/access/* folder are restrcited to authenticated user and anonymous user. If you try to access restricted by permission folders then app/private/restricted-by-permission/permission-{permissionId}/no-access/index.html content will be displayed with 403 status.
If admin or any authenticated user wants to access files for example:- app/private/restricted-by-permission/permission-xyz/access/index.html or app/private/restricted-by-permission/permission-xyz/access/styles.css .... then we have to assign a permission to the respective user based on permissionId.
permissionId should be the part after pemission- in folder name.
example:- from above example permission-xyz/access/* is the restricted folder, xyz
is the permission id.
By running below command in terminal, you are giving permission to admin to access permission-xyz/access/* folder.
./scripts/node-cli.sh
// Load admin user.
const u = await app.c('authentication').user('admin');
// Enable permission to access files of permission-xyz folder.
app.c('authentication').userFieldValue(u, 'view-content-permission-xyz', '1');
With this admin can access files of folder permission-xyz/access.
Disable permission to user:- Run below command to remove permission to access files of permission-xyz/access folder for admin user.
./scripts/node-cli.sh
// Load admin user.
const u = await app.c('authentication').user('admin');
// Remove permission to access files of permission-xyz folder.
app.c('authentication').userFieldValue(u, 'view-content-permission-xyz', '0');
Send WhatsApp Message:
To send a WhatsApp message, ensure the following environment variables are present and valid in the .env
file:
-
TWILIO_USER
-
TWILIO_PASS
-
WHATSAPP_FROM
-
WHATSAPP_DEV_MODE
-
If
WHATSAPP_DEV_MODE=true
(development environment), the message is saved to./unversioned/output/whatsapp-send.json
. -
If
WHATSAPP_DEV_MODE=false
(production environment), the message is sent to the specifiedsendTo
number.
Ensure WHATSAPP_DEV_MODE=true
in the development environment.
Testing WhatsApp Message Sending Functionality in Terminal:
-
Access the Node.js client:
./scripts/node-cli.sh
-
Run the following code, replacing
<country code>
and<phone number>
:>> await app.c('whatsAppSend').parsepropertySendMessage('{"message": "<Message content>", "sendTo":"<country code><phone number>"}');
Example:
>> await app.c('whatsAppSend').parsepropertySendMessage('{"message": "This is a test message", "sendTo":"+150XXXXXXX"}');
Sending media message:
>> await app.c('whatsAppSend').parsepropertySendMessage('{"message": "This is a test message", "sendTo":"+150XXXXXXX","mediaUrl": "<valid url of a image or video or excel or csv >"');
Testing WhatsApp Message Sending Functionality Using curl:
-
In Development Environment:
curl -X POST \ -H "Content-Type: application/json" \ --data '{"message": "This is a test message000", "sendTo": "+XXXXXXXXXX"}' \ http://0.0.0.0:8792/whatsappmessage/send/<WHATSAPPSENDM_API_TOKEN>
-
In Production Environment:
curl -X POST \ -H "Content-Type: application/json" \ --data '{"message": "This is a test message", "sendTo": "+XXXXXXXXXX"}' \ https://whatsapp-communication.dcycleproject.org/whatsappmessage/send/<WHATSAPPSENDM_API_TOKEN>
modify message and sendTo according to your requirement.
- If you are a authorised user then access .env and copy WHATSAPPSENDM_API_TOKEN value and replace in above command.
-
Sending media message :
curl -X POST -H "Content-Type: application/json" --data '{"message": "<media caption message or leave empty>", "sendTo": "+91XXXXXXXXXX","mediaUrl": "<valid url of a image or video or excel or csv >"}' <base url>/whatsappmessage/send/<WHATSAPPSENDM_API_TOKEN>
Receive WhatsApp Message:
Whenever a WhatsApp message is sent to the WHATSAPP_FROM
number, it is saved to ./unversioned/output/whatsapp.json
. If the message's account SID equals to TWILIO_USER
, then the message saved to the whatsappmessages
collection in the database.
You can verify whether the message is saved to the database:
-
Send a message:
WHATSAPP_TO=[PUT YOUR WHATSAPP NUMBER HERE] cd ~/whatsapp-communication curl "https://api.twilio.com/2010-04-01/Accounts/${TWILIO_USER}/Messages.json" -X POST \ --data-urlencode "To=whatsapp:${WHATSAPP_TO}" \ --data-urlencode "From=whatsapp:${WHATSAPP_FROM}" \ --data-urlencode 'Body=This is a reply' \ -u ${TWILIO_USER}:${TWILIO_PASS}
-
In a separate terminal window, log into the Mongo CLI and check what happened:
./scripts/mongo-cli.sh Show databases by running:
show dbs; ... arbitraryDatabase 0.000GB ... use arbitraryDatabase switched to db arbitraryDatabase show collections; arbitraryCollection db.whatsappmessages.find();
Verify your message record exist.
In January, 2024, we moved away from using Flow and moved to the approach described in Using Typescript without compilation, by Pascal Schilp, dev.to, Mar 26, 2023 and Type Safe JavaScript with JSDoc, by TruckJS, Medium, Sep. 4, 2018.
For this we are using https://github.com/dcycle/docker-typescript.
The reasons are:
- I prefer the syntax of Typescript/JSDoc to the unweildy syntax of Flow.
- Flow seems to require type definitions, whereas Typescript will surmise type definitions if they don't exist.
In some cases you might run into an issue where you cannot successfully start the node service. docker-compose logs node
might give you an error which looks like:
ENOSPC: System limit for number of file watchers reached
If such is the case you might want to increase the number of file watchers on the Docker host machine.
To see how many file watchers you have:
- On mac OS, run
sysctl kern.maxfiles
. On my system I getkern.maxfiles: 491520
. - On Ubuntu, run
cat /proc/sys/fs/inotify/max_user_watches
. On DigitalOcean I was getting 8192, which worked fine with a single instance of a Node Starterkit-based app, however when I tried to create a new one app, I was getting ENOSPC: System limit for number of file watchers reached.
To increase the number of file watchers, for example, to 524288 (this number comes from a comment from user wellsman on a GitHub issue, and the number 524288 appears a lot in different sources pertaining to this issue, however it's not clear whence this precise number comes). In any event this works on Ubuntu:
echo fs.inotify.max_user_watches=524288 | sudo tee -a /etc/sysctl.conf
sudo sysctl -p
Again, it is important to do this on your Docker host, not on the container!
- How to build a real time chat application in Node.js using Express, Mongoose and Socket.io, July 30, 2018, Free Code Camp.
- Local Authentication Using Passport in Node.js, Beardscript, April 8, 2020, Sitepoint.
- Everything you need to know about the
passport-local
Passport JS Strategy, Zach Gollwitzer, Jan 11, 2020, Level Up Coding (Medium). - Mastering the Node.js REPL (part 3), Roman Coedo, Aug 27, 2018, Medium
- Setup Github OAuth With Node and Passport JS, by Sjlouji, Sept. 22, 2020.
- How to Use the GitHub API to List Repositories, Carlos Schults, 7 May 2022, Fisebit
- Authorizing GitHub Apps, GitHub docs
- How to Build a Secure Node js REST API: 4 Easy Steps, November 3rd, 2021, Hevo
- Connect to a MongoDB Database Using Node.js, Lauren Schaefer, Feb 04, 2022, Updated Sep 23, 2022
- MongoDB and Node.js Tutorial - CRUD Operations, Lauren Schaefer, Feb 04, 2022, Updated Sep 23, 2022, MongoDB
- How To Use JSON Web Tokens (JWTs) in Express.js, Danny Denenberg, February 18, 2020, Updated on March 22, 2021, DigitalOcean