Authentication Flow
The way the authentication of the Server-side Rendered application is going to go is entirely through the help of JSON Web Tokens (JWTs) in cookies. JWTs are useful little identifiers that we can attach to our requests to and from the server so that we know what permissions the user has associated with them and what mutations/queries are valid for our GraphQL server to do.
They are ideally sent with every request, an a common way to do so is by setting them in the user's Local Storage of their web browser. The problem with this is that if our application is SSR'd, the initial request isn't sent with the JWT so we will actually send them an SSR'd view of the non-authenticated view, which will flicker in 1-2s to the authenticated view once our browser receives a request for its Locally Stored content.
By attaching the JWTs to our cookies, they will be sent to the server with every single request, meaning that our SSR'd initial view ill be the an authenticate one, i.e. with a cart, or profile picture, etc!
Sign Up
When a user signs up, there are a few things that need to be done in order to preserve this authentication flow. They are easy to forget and can be a hassle down the line if you have more and more users accessing your applications:
- Lowercase the email
- Hash the password
- Create the User in the database (with the new password/email values)
- Create a JWT for the user (preferably with their ID)
- Set the JWT on the response as a cookie
The following code snippet is an example of how to implement the sign-up resolver:
async signUp(parent, args, ctx, info) {
const email = args.email.toLowerCase();
const password = await bcrypt.hash(args.password, 10);
const user = await ctx.db.mutation.createUser(
{
data: {
...args,
email,
password,
permissions: {
set: ["USER"]
}
}
},
info
);
// Create the JWT for this specific uses
const token = jwt.sign({ userId: user.id }, process.env.APP_SECRET);
// Set the JWT on the response as a cookie
ctx.response.cookie("token", token, {
httpOnly: true,
maxAge: 1000 * 60 * 60 * 24 * 365 // 1 year
});
return user;
}
The permissions field above is formatted oddly simply because it is an enum
in GraphQL, you can see more about how these work in the other note: 2 - Server Side GraphQL.
Sign In/Out
The sign up is probably the most complicated mutation to write. To sign in, or sign out really doesn't require any writes to the data store, just some manipulation of the cookies. If a user signs in, we should check their password and username against our database, and assign them the proper credentials via JWT, and if they sign out, simply revoke the JwT their browser has assigned!
First things first, we are going to define these mechanisms as mutations
instead of queries
. The reason for this, is that something in our application is actually changing when we perform this request, even if it isn't related to the database. On the client-side, the UI and permissions are mutating, thus mutation
.
Sign In
async signIn(parent, args, ctx, info) {
// 1. Validate the incoming data
const email = args.email.toLowerCase()
const user = await ctx.db.query.user({ where: { email } }, `{id password}`)
if (!user) {
throw new Error(`😫 No user found with that email (${email})! 😫`)
}
const valid = await bcrypt.compare(args.password, user.password)
if (!valid) {
throw new Error(`❌ Invalid password, try again! ❌`)
}
// Create the JWT for this specific uses
const token = jwt.sign({ userId: user.id }, process.env.APP_SECRET)
// Set the JWT on the response as a cookie
ctx.response.cookie('sf-token', token, {
httpOnly: true,
maxAge: 1000 * 60 * 60 * 24 * 365 // 1 year
})
return ctx.db.query.user({ where: { email } }, info)
},
The cookie generation is done exactly like the sign up system, since the user is signed in after signing up.
Sign Out
Here is actually a good place to explain how custom types can be passed as a response. In some cases such as this, you'll want to just send a message back from the API rather than information about the user who just signed out, since it wouldn't be useful to the app.
In that case you just add it to the schema.graphql
file as another type:
type SuccessMessage {
message: String
}
type Mutation {
signOut: SuccessMessage
}
Now your resolver has to return the data with the matching shape:
async signOut(parent, args, ctx, info) {
ctx.response.clearCookie('sf-token', {
httpOnly: true,
maxAge: 1000 * 60 * 60 * 24 * 365 // 1 year
})
return { message: 'See ya!' }
}
Another important note is that the clearCookie
method attached to the response is actually requires the same parameters as initially provided to it in order to ensure the browser complies (Check out the Docs)
Request Security
One important aspect of building an full-scale GraphQL API with Prisma that you should take into account is the accessible fields available to the user. Since the client determines the information it can receive, you have to make sure that certain fields aren't available to the client, and only for the resolvers. These are things like account credentials, emails, 2FA sources, passwords, reset tokens.
This can easily pass you by if you just import the generated schema from Prisma:
# import * from './generated/prisma.graphql'
type Query {
...
}
type Mutation {
...
}
The comment acts as an import statement for the prisma types, while this is good in most cases, for verbosity, it can be harmful in others. Take a User
type for example:
type User implements Node {
id: ID!
name: String!
email: String!
password: String!
resetToken: String
resetTokenExpiry: Float
permissions: [Permission!]!
updatedAt: DateTime!
createdAt: DateTime!
}
In this case, any time a query returns a User
the, client can request the hashed password, or reset token.
This can be fixed by just removing the fields and overwriting the previous type:
# import * from './generated/prisma.graphql'
type User {
id: ID!
name: String!
email: String!
permissions: [Permission!]!
}
type Query {
...
}
type Mutation {
...
}
Password Reset Flow
Implementing a password reset flow can be boiled down to just implementing a few extra routes in the API. As long as you have the following mutations and steps, you should be good to go:
- Client-side email field to send the request
- Perform
requestReset
mutation - In this mutation, check if the email is valid
- Set a
randomBytes
token on the account (resetToken) - Set an expiry for the token in a timestamp (resetTokenExpiry)
- Send the token to the account holder's email via URL
- Setup the URL parsing on the client-side to read the token
- Check if the token is valid
- Check if the token is expired
- Prompt for a new password
- Update the new password, revoke the reset token and expiry
- Set the cookie, login the user
Sending Mail
There are really only a few parts to sending mail with Node.js:
- The templating strategy
- The mail servers
For development purposes, one of the easiest solutions is to go with a service such as mailtrap.io. They provide you with a demo mailbox to send mail from, and all you need to do is pass in credentials (which should be in an environment file) to your NodeMailer
client (a helpful library for easily sending mail):
const transport = nodemailer.createTransport({
host: process.env.MAIL_HOST // "smtp.mailtrap.io",
port: process.env.MAIL_PORT // 2525,
auth: {
user: process.env.MAIL_USER // "38700db53edc57",
pass: process.env.MAIL_PASS // "076761e33e4b71"
}
});
Without implementing a templating strategy, you're left to just setting basic inline styles on html
as a string, which works just fine. You can check out solutions such as MJML for scalable alternative templating strategies.
You can send basic emails as follows:
await transport.sendMail({
from: "[email protected]",
to: user.email,
subject: "🙌 Reset your Password! 🙌",
html: `
<div className="email" style="
border: 1px solid black;
padding: 20px;
">
<h2>Hey there!</h2>
<p>
Your password reset token is here! \n\n <a href="${
process.env.FRONTEND_URL
}/reset?resetToken=${resetToken}">Click here to reset your password!</a>
</p>
<p>✌, Leander Rodrigues</p>
</div>`,
})
Relationships
A common pain point for many applications is creating concrete relationships between items in the database, and the user's responsible for them. To do so with Prisma/GraphQL, you first need to make sure they are designated in your datamodel file:
type Item {
id: ID! @id
title: String!
description: String!
updatedAt: DateTime! @updatedAt
createdAt: DateTime! @createdAt
user: User! # This is the relationship declaration
}
After deploying this to your live API server (prisma deploy
), you should get the updated prisma graphql
file after the graphql get-schema -p prisma
hook runs.
You've now set up a relationship! To start using it, take a look at this mutation example:
async createItem(parent, args, ctx, info) {
// Check if the request has the userId on it (attached via cookies)
if (!ctx.request.userId) {
throw new Error('🙅♂️ You must be logged in to do that! 🙅♀️')
}
// Create the item and connect it with the user
return ctx.db.mutation.createItem(
{
data: {
user: {
connect: {
id: ctx.request.userId
}
},
...args
}
},
info
)
},
This syntax will connect the the item to the user by using the id
field as the relationship! Now, in queries, the following is possible to query:
query {
item(where: { id: { eq: "dajf9912l0asd" } }) {
id
title
user {
id
name
email
}
}
}
Creating a Gate Sign in Component
A gated sign-in component is really useful for creating routes that user's shouldn't have access to without being signed in. You'll find this a lot on bigger sites where you click a link to some resource, then find yourself at a login instead of the resource. After logging in, you'll be redirected to the resource, and now your account details are present in the app.
To accomplish the same thing in React, all you need to do is steal the component you originally used to sign in, and re-purpose it into a wrapper.
const PleaseSignIn = ({ children }) => (
<Query query={CURRENT_USER_QUERY}>
{({ data, loading }) => {
if (loading) return <p>⚡ Loading... ⚡</p>
if (!data.me) {
return (
<div>
<SignIn samePage header="Sign in to continue" />
</div>
)
}
return children
}}
</Query>
)
This wrapper component checks with a CURRENT_USER_QUERY
to see whether or not the current user is logged in. If so, it renders it's children, renderProps
style. If not though, the content that is wrapped by this component is replaced with a dialog prompting the user to sign in!
If there are fixes you need (e.g. routing after sign in, text content), you might have to refactor the component to include some dependent code based on the props you passed to the wrapper. You can see it up there, using the props declared in SignIn.js
:
class SignIn extends Component {
static propTypes = {
header: PropTypes.string,
samePage: PropTypes.boolean
}
render() {
const { header, samePage } = this.props
return (
<Mutation
mutation={SIGN_IN_MUTATION}
variables={this.state}
refetchQueries={[{ query: CURRENT_USER_QUERY }]}
>
{(signIn, { error, loading }) => {
return (
<Form
method="post"
onSubmit={async e => {
e.preventDefault()
await signIn()
this.setState({
email: '',
name: '',
password: ''
})
if (!samePage) Router.push('/')
}}
>
<fieldset disabled={loading} aria-busy={loading}>
<h2>{header ? header : 'Already have an account?'}</h2>
// FOR ENUMS UDPATE
permissions: {
set: args.permissions
}
passing GQL mutations to event handler functions
handlePermissionChange(event, mutationFunction)
setState(stateChange, callback) setState(stateChange, mutationFunction ) ---> neat, since it will ensure latest state is mutated