In this guide, we will walk through how to build a full-stack application with RedwoodJS.
You may have seen a lot of tutorials and guides around building a full-stack application with x framework (or) technology, but, RedwoodJS is different and beneficial in certain ways, including:
- RedwoodJS includes Typescript, GraphQL, Prisma, and a testing framework
- Startups can build and prototype a product since it provides modules such as authentication, authorization, and CRUD operations. All we need to do is to design business logic for our requirements
- CLI is one of the best features of RedwoodJS; it makes the development process faster and easier
Take a look at this demo to what the final app will look like from this tutorial:
Redwood_LogRocket.mp4
Dropbox is a free service that lets you bring your photos, docs, and videos anywhere and share them easily. Never email yourself a file again!
Here we are going to build a forum to understand how RedwoodJS apps are built. It includes all the functionalities that help you understand all the frameworks’ functions.
The functionalities that we are going to build are:
- Login and signup
- Create, read, and update posts
- Commenting system
- User-based access on posts
Along with RedwoodJS, we will use Typescript for type checking and TailwindCSS for styling.
Table of Contents
- RedwoodJS installation and setup
- Setting up TailwindCSS
- Connecting database
- Designing database
- Prisma migration
- Authentication
- Creating posts
- Post details
- User-based access
RedwoodJS installation and setup
RedwoodJS uses yarn as a package manager. Once you install it, you can create a new project using the following command:
yarn create redwood-app --ts ./redwoodblog
It scaffolds all the modules to build a full-stack application. Here, you can see the complete structure of a RedwoodJS application.
There are three major directories. They are api
, scripts
, and web
. Let’s discuss them in detail.
.redwood
: Contains the build of the application.api
: Serves the backend of the application. It mainly containsdb
, which serves the database schema of the application. All backend functionalities will be insrc
directorysrc
: Contains all your backend code. It contains five directories, which are as follows:directives
: Contains GraphQL schema directives to control access to GraphQL queriesfunctions
: RedwoodJS runs the GraphQL API as serverless functions. It auto-generatesgraphql.ts
; you can add additional serverless function on top of itgraphql
: Contains GraphQL schema written in schema definition language (SDL)lib
: Contains all the reusable functions across the backend API. for example, authenticationservices
: Contains business logic related to your data. It runs the functionality related to the API and returns the results
Setting up TailwindCSS
Installing TailwindCSS is straightforward; run the following command in the root directory:
yarn rw setup ui tailwindcss
To confirm the installation of TailwindCSS, go to web/src/index.css
and see the Tailwind classes in that file.
Connecting database
To connect the Postgres database, we will use Docker for local development.
(Note: To install docker, see the documentation from official docker website)
Create docker-utils/postgres-database.sh
in root directory and add the following script:
#!/bin/bash
set -e
set -u
function create_user_and_database() {
local database=$1
echo " Creating user and database '$database'"
psql -v ON_ERROR_STOP=1 --username "$POSTGRES_USER" <<-EOSQL
CREATE USER $database;
CREATE DATABASE $database;
GRANT ALL PRIVILEGES ON DATABASE $database TO $database;
EOSQL
}
if [ -n "$POSTGRES_MULTIPLE_DATABASES" ]; then
echo "Multiple database creation requested: $POSTGRES_MULTIPLE_DATABASES"
for db in $(echo $POSTGRES_MULTIPLE_DATABASES | tr ',' ' '); do
create_user_and_database $db
done
echo "Multiple databases created"
fi
This script implements a function to create a user and database in Postgres. Once you create the script, you can use docker-compose up
to run the Postgres database.
Create docker-compose.yml
and add the following code:
version: "3.6" services: postgres: image: postgres restart: unless-stopped
If you want to create different versions of docker-compose up
based on the environment, you can do that as well. To do this, create docker-compose.override.yml
and add the following code:
version: "3" services: postgres: image: postgres environment: - POSTGRES_USER=api - POSTGRES_PASSWORD=development_pass - POSTGRES_MULTIPLE_DATABASES="redwoodforum-api","redwoodforum-api-testing" volumes: - ./docker-utils:/docker-entrypoint-initdb.d - redwoodforum_api_data:/data/postgres ports: - 5440:5432 volumes: redwoodforum_api_data: {}
Once you add the script, you can run the database using this command:
docker-compose up
To connect a Redwood application to Postgres, change the Prisma configuration to a PostgreSQL provider and add database URL in an environment variable.
Go to api/db/schema.prisma
and change the db provider to postgresql
. Add DATABASE_URL
in your .env.
DATABASE_URL=postgres://api:development_pass@localhost:5440/redwoodforum-api
Designing database
As you can see in the demo, we want to build a forum. However, before we implement the functionality, Here are the key things we want users to be able to do in our application:
- Users can log in/signup into the app
- Once users log in, they can create a post in the forum
- Users can comment on any post, and the owner can delete any comments
- User can view their post and go to the Home page to view all posts
Let’s design an ER diagram for the application.
Here we have the user
, post
, and comment
schemas.
user
and post
have a one-to-many relationship, and post
and comment
have a one-to-many relationship, while comment
and user
have a one-to-one relationship.
Now we have the ER diagram for the application. let’s create the schema for the database. For that, go to api/db/schema.prisma
.
(Note: RedwoodJS uses Prisma for database. If you’re new to Prisma world, check out their documentation for more information)
Now, create the schemas in a Prisma file:
model User { id Int @id @default(autoincrement()) email String @unique name String? hashedPassword String salt String resetToken String? resetTokenExpiresAt DateTime? posts Post[] comments Comment[] } model Post { id Int @id @default(autoincrement()) title String body String comments Comment[] author User @relation(fields: [authorId], references: [id]) authorId Int createdAt DateTime @default(now()) updatedAt DateTime @default(now()) } model Comment { id Int @id @default(autoincrement()) body String post Post @relation(fields: [postId], references: [id]) postId Int author User @relation(fields: [authorId], references: [id]) authorId Int createdAt DateTime @default(now()) updatedAt DateTime @default(now()) }
As you can see, we have a relationship between the User
, Post
, and Comment
schemas.
Defining a relationship in Prisma is simple. You can refer to the documentation to learn more in detail.
Once you define a schema in Prisma, you must run the migration to create those schemas as a table in Postgres.
Prisma migration
One of the features of Prisma is that you can manage migration for different stages. Here, we are going to run the migration only for development. For that, you can use this command:
yarn redwood prisma migrate dev
To check if the migration was successful, you can go to Prisma Studio and see all the tables after migration. you can see all the tables and columns inside of each table by visiting http://localhost:5555.
yarn redwood prisma studio
Now, we have database and schema for the API and frontend, let’s create authentication for the application.
Authentication
RedwoodJS provides authentication out of the box. A single CLI command will get you everything you need to get authentication working.
yarn rw setup auth dbAuth
It will create a auth.ts
serverless function that checks the cookie if the user exists in the database and token expiry. Then, it returns the response based on that to a client.
It also creates lib/auth.ts
to handle functionalities, such as getCurrent user from session, check if authenticated, require authentication etc.
So far, we have the authentication functionality for the API and database. Let’s create pages for login, signup, and forgot password. Then, you can use the command to scaffold the login, signup, and forgot password pages.
yarn rw g dbAuth
It will create all the pages for authentication. You can check those pages at web/src/pages
.
For styling the pages, you can use the components from the source code and customize them based on your preferences. Here is the complete login page from the implementation:
To connect an API for login and signup functionality, RedwoodJS provides hooks that do all the magic under the hood.
import { useAuth } from '@redwoodjs/auth' // provides login and signup functionality out of the box const { isAuthenticated, signUp, logIn, logOut } = useAuth()
In the form onSubmit
function, we can use that signup
and logIn
to make the API request and send the payload.
const onSubmit = async (data) => { const response = await signUp({ ...data }) if (response.message) { toast(response.message) } else if (response.error) { toast.error(response.error) } else { // user is signed in automatically toast.success('Welcome!') } }
Once the user signs up or logs in, you can access the user information across the application using currentUser
.
const { currentUser } = useAuth()
Now, we have the user logged in to the application. Next, let’s build the functionality to post and comment.
Once the users log in, they land on the home page, where we need to show all the posts in the forum. Then, the user can create a new post and update a post.
To implement the listing page, create a route with the Home page component and fetch the data from the API to show it on the client side.
Luckily, RedwoodJS provides scaffolding that generates all the implementation for us. Let’s say you want to scaffold all the pages, including GraphQL backend implementation, you can use the following command:
yarn redwood g scaffold post
It will generate pages, SDL, and service for the post model. You can refer to all the RedwoodJS commands in their documentation.
Since we are going to customize the pages. Let’s scaffold SDL and services only. Use this command:
yarn redwood g sdl --typescript post
It will create post domain files in graphql/posts.sdl.ts
and services/posts
— let’s create pages on the web.
Even though we customize the pages and components, we don’t need to create everything from scratch. Instead, we can use scaffolding and modify it based on our requirements.
Let’s create a Home page using this command:
yarn redwood g page home
It will create a home page and add that page inside the Routes.tsx
. So now, you have the basic home page component.
Now, to list all the posts on the home page, you need to fetch the data from api and show it on the pages. To make this process easier, RedwoodJS provides cells. Cells are a declarative approach to data fetching — it executes a GraphQL query and manages its lifecycle.
To generate cells, use this command:
yarn rw generate cell home
It will create a GraphQL query and its lifecycle:
import type { FindPosts } from 'types/graphql' import { Link } from '@redwoodjs/router' import type { CellSuccessProps, CellFailureProps, CellLoadingProps, } from '@redwoodjs/web' export const QUERY = gql` query FindPosts { posts { id title body comments { id } createdAt updatedAt } } ` export const Loading: React.FC<CellLoadingProps> = () => <div>Loading...</div> export const Empty = () => <div>No posts found</div> export const Failure = ({ error }: CellFailureProps) => ( <div>Error loading posts: {error.message}</div> ) export const Success = ({ posts }: CellSuccessProps<FindPosts>) => { return ( <div> <ul> {posts.map((post) => ( <li key={post.id}> {' '} <Link to={`/posts/${post.id}`}> <div className="p-2 my-2 rounded-lg shadow cursor-pointer"> <h4 className="text-xl font-medium">{post.title}</h4> <p>{post.body}</p> </div> </Link> </li> ))} </ul> </div> ) }
Protected route
To protect the route in the RedwoodJS application, you can use Private
from @redwoodjs/router
and wrap everything inside the route.
<Private unauthenticated="login"> <Set wrap={NavbarLayout}> <Set wrap={ContainerLayout}> <Route path="/new" page={NewpostPage} name="newpost" /> <Set wrap={SidebarLayout}> <Route path="/" page={HomePage} name="home" /> // routes come here </Set> </Set> </Set> </Private>
Creating posts
To create a new post, scaffold a new post page using the following command:
yarn redwood g page newpost /new
If you want to customize the route URL, you can pass that as a parameter here. RedwoodJS adds routes based on the provided name. RedwoodJS provides forms and validations out of the box,
import { FieldError, Form, Label, TextField, TextAreaField, Submit, SubmitHandler, } from '@redwoodjs/forms'
Once a user submits form, you can call the GraphQL mutation to create a post.
const CREATE_POST = gql` mutation CreatePostMutation($input: CreatePostInput!) { createPost(input: $input) { id } } ` const onSubmit: SubmitHandler<FormValues> = async (data) => { try { await create({ variables: { input: { ...data, authorId: currentUser.id }, }, }) toast('Post created!') navigate(routes.home()) } catch (e) { toast.error(e.message) } }
Post details
Create a post details page and cell for data fetching to view post details. You can follow the same process that we did before.
yarn redwood g page postdetails
This will create the page and route in routes.tsx
. To pass URL params in the route, you can modify it like this:
<Route path="/posts/{id:Int}" page={PostDetails} name="postdetails" />
You can pass ID into the component as props. Then, create a cell to fetch the post details and render it inside the component.
yarn redwood g cell post
Add the following code to fetch the data and comments for a specific post:
import type { FindPosts } from 'types/graphql' import { format } from 'date-fns' import { useAuth } from '@redwoodjs/auth' import type { CellSuccessProps, CellFailureProps, CellLoadingProps, } from '@redwoodjs/web' export const QUERY = gql` query FindPostDetail($id: Int!) { post: post(id: $id) { id title body author { id } comments { id body author { id name } createdAt } createdAt updatedAt } } ` export const Loading: React.FC<CellLoadingProps> = () => <div>Loading...</div> export const Empty = () => <div>No posts found</div> export const Failure = ({ error }: CellFailureProps) => ( <div>Error loading posts: {error.message}</div> ) export const Success = ({ post }: CellSuccessProps<FindPosts>) => { const { currentUser } = useAuth() return ( <div> <div> <h2 className="text-2xl font-semibold">{post.title}</h2> <p className="mt-2">{post.body}</p> </div> <div className="mt-4 "> <hr /> <h3 className="my-4 text-lg font-semibold text-gray-900">Comments</h3> {post.comments.map((comment) => ( <div key={comment.id} className="flex justify-between sm:px-2 sm:py-2 border rounded-lg" > <div className="my-4 flex-1 leading-relaxed"> <strong>{comment.author.name}</strong>{' '} <span className="text-xs text-gray-400"> {format(new Date(comment.createdAt), 'MMM d, yyyy h:mm a')} </span> <p>{comment.body}</p> </div> {currentUser && currentUser.id === post.author.id && ( <div className="m-auto"> <button type="button" className="focus:outline-none text-white bg-red-700 hover:bg-red-800 focus:ring-4 focus:ring-red-300 font-medium rounded-lg text-sm px-5 py-2.5 mr-2 mb-2 dark:bg-red-600 dark:hover:bg-red-700 dark:focus:ring-red-900" > Delete </button> </div> )} </div> ))} </div> </div> ) }
An important thing to note here is the condition to check if the current logged in user is the author of post. In that case, we provide an option to delete comments.
User-based access
To provide user-based access inside the application, you can get the current user using useAuth
hooks and add conditions on it. For example, To show a list of posts created by the user, you can use the current user ID to fetch posts by author.
const { currentUser } = useAuth()
MyPostCell.tsx
import { Link } from '@redwoodjs/router' import type { FindPosts } from 'types/graphql' import type { CellSuccessProps, CellFailureProps, CellLoadingProps, } from '@redwoodjs/web' export const QUERY = gql` query FindMyPosts($id: Int!) { user: user(id: $id) { id name posts { id title body } } } ` export const Loading: React.FC<CellLoadingProps> = () => <div>Loading...</div> export const Empty = () => <div>No posts found</div> export const Failure = ({ error }: CellFailureProps) => ( <div>Error loading posts: {error.message}</div> ) export const Success = ({ user }: CellSuccessProps<FindPosts>) => { return ( <div> <ul> {user.posts.map((post) => ( <li key={post.id}> {' '} <Link to={`/posts/${post.id}`}> <div className="shadow rounded-lg p-2 my-2 cursor-pointer"> <h4 className="text-xl font-medium">{post.title}</h4> <p>{post.body}</p> </div> </Link> </li> ))} </ul> </div> ) }
Conclusion
RedwoodJS provides everything out of the box. It’s all about building the application based on our requirements. Some important concepts are cells, pages, Prisma schema and migration, and understanding how the system works.
Once you understand RedWoodJS, you can build a full-stack application with very little time, as we’ve seen in this post. You can find the source code for this tutorial here.
The post How to build a full-stack app in RedwoodJS appeared first on LogRocket Blog.
from LogRocket Blog https://ift.tt/EDZGVg7
Gain $200 in a week
via Read more