”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > Node.js 入门项目包括 GraphQL、Redis、JWT 和 Sequelize

Node.js 入门项目包括 GraphQL、Redis、JWT 和 Sequelize

发布于2024-12-22
浏览:514

Node.js Starter Project dengan GraphQL, Redis, JWT, dan Sequelize

此模板提供了一个 Node.js 入门项目,已配置 GraphQL 用于 API,Redis 用于缓存和临时数据存储,JWT 用于身份验证和授权,Sequelize 用于 ORM连接到关系数据库,例如 PostgreSQL 或 MySQL。该项目具有模块化结构,可让您立即开发具有集成和可扩展功能的现代 Web 应用程序。

?项目概况

该项目旨在让使用 GraphQL API 更轻松地开发后端应用程序,该 API 使用 Redis 进行数据缓存,并使用 JWT 来保护 API。 Sequelize 用作 ORM 来促进与关系数据库的交互。此外,还有中间件可以更轻松地处理身份验证、验证和日志记录。

主要特点

  • GraphQL API实现更灵活、高效的查询和数据突变
  • JWT 身份验证 用于基于令牌的安全身份验证
  • Redis用于数据缓存和提高应用程序性能
  • Sequelize ORM用于关系数据库管理
  • 中间件用于集中授权和请求处理
  • 模块化且结构良好可扩展且更易于维护

?️ 使用的技术

  • Node.js:使用 JavaScript 构建服务器端应用程序的平台。了解更多
  • GraphQL:API 的查询语言,可实现高效、灵活的数据检索。了解更多
  • Redis:临时数据存储(内存中),通常用于缓存和消息代理。了解更多
  • JWT:安全且简单的基于令牌的身份验证技术。了解更多
  • Sequelize:Node.js 的 ORM,支持 PostgreSQL、MySQL 和其他关系数据库。了解更多

?设置和运行项目的步骤

1. 克隆存储库

首先,将此模板存储库克隆到本地计算机:

git clone https://gitlab.com/dioarafi1/graphify-api.git
cd graphify-api

如果您从头开始,请使用以下命令初始化一个新项目:

mkdir blog-api
cd blog-api
npm init -y

2. 安装依赖项

克隆存储库或创建新项目后,运行命令安装所需的依赖项:

yarn install

这将安装 package.json 文件中列出的所有依赖项。


3. 环境配置

在项目根目录下创建.env文件,为RedisJWTDatabase添加如下配置:

DATABASE_URL="postgresql://user:password@localhost:5432/mydatabase"
JWT_SECRET="your_jwt_secret_key"
REDIS_HOST="localhost"
REDIS_PORT="6379"

根据您的数据库配置更改用户、密码和 mydatabase。


4. 使用 Sequelize 准备数据库

如果没有配置数据库,请运行命令初始化Sequelize并创建模型:

yarn sequelize init

这将在您的项目中创建配置、模型和迁移目录结构。接下来,为应用程序创建必要的模型,例如 UserPost,并执行迁移以在数据库中创建表。

yarn sequelize db:migrate

确保数据库正在运行(例如使用 PostgreSQL 或 MySQL)。


5. 设置 GraphQL Server

安装 Apollo Server 和 GraphQL 的依赖项:

yarn add apollo-server graphql

之后,创建 GraphQL 服务器配置文件、架构和解析器。您可以通过以下方式配置 GraphQL 服务器:

src/服务器.ts

import { ApolloServer } from 'apollo-server-express';
import express from 'express';
import { typeDefs, resolvers } from './graphql';
import { authenticateJWT } from './middlewares/auth';
import { sequelize } from './config/database';

const app = express();

// Gunakan middleware JWT
app.use(authenticateJWT);

// Inisialisasi Apollo Server
const server = new ApolloServer({
  typeDefs,
  resolvers,
  context: ({ req }) => ({ user: req.user }),
});

server.applyMiddleware({ app });

const PORT = process.env.PORT || 4000;

app.listen(PORT, async () => {
  console.log(`Server running at http://localhost:${PORT}${server.graphqlPath}`);
  await sequelize.authenticate();
  console.log('Database connected');
});

src/graphql/schema.ts

定义查询和突变的 GraphQL 架构:

import { gql } from 'apollo-server-express';

export const typeDefs = gql`
  type User {
    id: ID!
    username: String!
  }

  type Post {
    id: ID!
    title: String!
    content: String!
    user: User!
  }

  type Query {
    posts: [Post]
    post(id: ID!): Post
    users: [User]
  }

  type Mutation {
    createPost(title: String!, content: String!): Post
    register(username: String!, password: String!): User
    login(username: String!, password: String!): String # JWT token
  }
`;

src/graphql/resolvers.ts

为查询和突变实现解析器:

import { Post, User } from '../models';
import jwt from 'jsonwebtoken';
import bcrypt from 'bcryptjs';

export const resolvers = {
  Query: {
    posts: () => Post.findAll(),
    post: (_, { id }) => Post.findByPk(id),
    users: () => User.findAll(),
  },
  Mutation: {
    createPost: async (_, { title, content }, { user }) => {
      if (!user) throw new Error('Authentication required');
      const post = await Post.create({ title, content, userId: user.id });
      return post;
    },
    register: async (_, { username, password }) => {
      const hashedPassword = await bcrypt.hash(password, 10);
      const user = await User.create({ username, password: hashedPassword });
      return user;
    },
    login: async (_, { username, password }) => {
      const user = await User.findOne({ where: { username } });
      if (!user) throw new Error('User not found');

      const match = await bcrypt.compare(password, user.password);
      if (!match) throw new Error('Invalid password');

      const token = jwt.sign({ userId: user.id }, process.env.JWT_SECRET!, { expiresIn: '1h' });
      return token;
    },
  },
};

6. 运行开发服务器

要使用hot-reload在开发环境中运行服务器,请使用以下命令:

yarn dev

服务器将在http://localhost:4000运行,您可以访问GraphQL Playground来测试API查询和突变。


?项目目录结构

此项目目录结构旨在分隔应用程序的不同部分,使其更加模块化且易于维护:

/myapp
├── src
│   ├── middlewares      # Berisi middleware untuk otentikasi (JWT), caching (Redis), dan validasi
│   ├── routes           # Definisi endpoint API dan resolver GraphQL
│   ├── services         # Logika bisnis utama dan pengolahan data
│   ├── app.ts           # File utama untuk inisialisasi aplikasi dan middleware
│   ├── graphql          # Menyimpan konfigurasi GraphQL, schema, dan resolvers
│   ├── models           # Model Sequelize untuk mengelola database relasional
│   ├── config           # File konfigurasi global untuk Redis, JWT, database, dll
│   ├── index.ts         # Entry point aplikasi, menginisialisasi server dan middleware
│   ├── resolvers        # Berisi resolver GraphQL untuk query dan mutasi
│   ├── server.ts        # File untuk setup Apollo Server dan konfigurasi GraphQL
│   ├── schema           # Definisi schema GraphQL
│   ├── types            # TypeScript types dan interfaces untuk GraphQL dan lainnya
│   └── utils            # Berisi helper dan fungsi utility
├──

 .env                 # File konfigurasi environment (Redis, JWT Secret, Database URL)
├── package.json         # Metadata proyek dan dependensi
└── tsconfig.json        # Konfigurasi TypeScript

?部署

要准备生产项目,请使用以下命令将 TypeScript 构建为 JavaScript:

yarn build

输出将位于 dist/ 文件夹中并准备好部署到生产服务器。

为云平台做准备

此项目可以部署到 HerokuAWSDigitalOcean 等平台,步骤如下:

  1. 将代码推送到 Git 存储库(GitHub、GitLab 或其他)。
  2. 在所选云平台(Redis、JWT Secret、URL 数据库)上设置环境变量
  3. 使用云平台的命令或集成部署项目

?资源

  • GraphQL 文档
  • Redis 文档
  • 智威汤逊文档
  • 续集文档

按照上述步骤,您现在可以使用 RedisJWTSequelize 运行和开发 GraphQL API 应用程序。

版本声明 本文转载于:https://dev.to/dioarafi/nodejs-starter-project-dengan-graphql-redis-jwt-dan-sequelize-2l3i?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • 在 JavaScript 中使用 setTimeout 时如何保留“this”引用?
    在 JavaScript 中使用 setTimeout 时如何保留“this”引用?
    setTimeout 和 JavaScript 中难以捉摸的“this”使用 setTimeout 函数时,开发者经常会遇到后续调用方法丢失的问题他们的预期上下文,导致看似未定义的方法。这通常是由于丢失“this”引用引起的。问题:考虑以下代码:test.prototype.method = fun...
    编程 发布于2024-12-22
  • 尽管代码有效,为什么 POST 请求无法捕获 PHP 中的输入?
    尽管代码有效,为什么 POST 请求无法捕获 PHP 中的输入?
    解决 PHP 中的 POST 请求故障在提供的代码片段中:action=''而不是:action="<?php echo $_SERVER['PHP_SELF'];?>";?>"检查 $_POST数组:表单提交后使用 var_dump 检查 $_POST 数...
    编程 发布于2024-12-22
  • Bootstrap 4 Beta 中的列偏移发生了什么?
    Bootstrap 4 Beta 中的列偏移发生了什么?
    Bootstrap 4 Beta:列偏移的删除和恢复Bootstrap 4 在其 Beta 1 版本中引入了重大更改柱子偏移了。然而,随着 Beta 2 的后续发布,这些变化已经逆转。从 offset-md-* 到 ml-auto在 Bootstrap 4 Beta 1 中, offset-md-*...
    编程 发布于2024-12-22
  • 为什么使用 SwingUtilities.invokeLater?
    为什么使用 SwingUtilities.invokeLater?
    SwingUtilities.invokeLater:仔细观察在 Swing 编程领域,SwingUtilities.invokeLater 起着至关重要的作用。这个方法到底是做什么的,它与在主线程中直接执行代码有何不同?SwingUtilities.invokeLater,顾名思义,将给定 Run...
    编程 发布于2024-12-22
  • 插入数据时如何修复“常规错误:2006 MySQL 服务器已消失”?
    插入数据时如何修复“常规错误:2006 MySQL 服务器已消失”?
    插入记录时如何解决“一般错误:2006 MySQL 服务器已消失”介绍:将数据插入 MySQL 数据库有时会导致错误“一般错误:2006 MySQL 服务器已消失”。当与服务器的连接丢失时会出现此错误,通常是由于 MySQL 配置中的两个变量之一所致。解决方案:解决此错误的关键是调整wait_tim...
    编程 发布于2024-12-22
  • 如何在 Spring 中流式传输大型 MySQL 结果集以避免 OutOfMemoryError?
    如何在 Spring 中流式传输大型 MySQL 结果集以避免 OutOfMemoryError?
    使用 MySQL 流式处理大型结果集在 Spring 应用程序中处理大量 MySQL 表时,当驱动程序试图执行以下操作时,可能会出现 OutOfMemoryException:将整个表加载到内存中。设置statement.setFetchSize(Integer.MIN_VALUE);可能还不够,因...
    编程 发布于2024-12-22
  • 在 SQL 中更新同一个表后,如何安全地更新触发器内的表?
    在 SQL 中更新同一个表后,如何安全地更新触发器内的表?
    更新同一个表后更新触发器中的表在 SQL 中,更新触发器后更新表同一张表提出了一个潜在的问题。这是因为该表已针对更新操作锁定,并且尝试在作为同一事务的一部分执行的触发器内访问该表可能会导致冲突。要规避此限制,您可以更新通过使用 BEFORE 选项而不是 AFTER 来影响触发器中的列。这允许您在原始...
    编程 发布于2024-12-22
  • 使用PHP的DOMDocument提取节点时如何保留HTML标签?
    使用PHP的DOMDocument提取节点时如何保留HTML标签?
    使用 DOMDocument 提取 HTML 节点的问题简介DOMDocument 是一个 PHP 类,提供了一种便捷的解析方法以及操作 HTML 文档。然而,当尝试在提取内容时保留 HTML 标签时,用户可能会遇到困难。本文深入探讨了 DOM 的基本概念,并提出了应对这一挑战的解决方案。理解 DO...
    编程 发布于2024-12-22
  • 如何在 Python 类型提示中指定函数类型?
    如何在 Python 类型提示中指定函数类型?
    在类型提示中指定函数类型在Python中,类型提示用于提供有关变量和函数参数的预期类型的​​可选元数据。然而,将变量的类型提示指定为函数类型似乎不清楚。解决方案尽管相关中缺少“typing.Function” PEP 483,您可以使用以下命令将变量的类型提示指定为函数类型"typing....
    编程 发布于2024-12-22
  • 如何修复 macOS 上 Django 中的“配置不正确:加载 MySQLdb 模块时出错”?
    如何修复 macOS 上 Django 中的“配置不正确:加载 MySQLdb 模块时出错”?
    MySQL配置不正确:相对路径的问题在Django中运行python manage.py runserver时,可能会遇到以下错误:ImproperlyConfigured: Error loading MySQLdb module: dlopen(/Library/Python/2.7/site-...
    编程 发布于2024-12-22
  • 为什么我在发送 URL 编码的 POST 请求时收到 400 BAD REQUEST?
    为什么我在发送 URL 编码的 POST 请求时收到 400 BAD REQUEST?
    带有 URL 编码负载的 POST 请求当构造带有 URL 编码数据的 POST 请求时,经常会遇到 400 BAD REQUEST 响应,表明服务器无法理解负载。当有效负载提供不正确时,通常会出现此问题。标准库的 http.NewRequest(...) 方法期望将有效负载作为第三个参数提供,该参...
    编程 发布于2024-12-22
  • HTML 格式标签
    HTML 格式标签
    HTML 格式化元素 **HTML Formatting is a process of formatting text for better look and feel. HTML provides us ability to format text without us...
    编程 发布于2024-12-22
  • 为什么我不能直接从定义类中调用内联友元函数?
    为什么我不能直接从定义类中调用内联友元函数?
    内联友元函数的可见性范围尽管在类中定义了内联友元函数,但它们的实际范围超出了类的边界。 C 标准规定,内联友元函数的名称引用在最近的封闭命名空间范围中声明的函数。但是,在该命名空间内提供匹配声明之前,无法通过非限定或限定查找访问此声明的函数。范围。因此,从类本身、封闭作用域或成员函数直接调用友元函数...
    编程 发布于2024-12-22
  • ## [].forEach.call() 如何转换 JavaScript 类数组结构?
    ## [].forEach.call() 如何转换 JavaScript 类数组结构?
    JavaScript 中的数组转换:了解 [].forEach.call()使用 JavaScript 时,您可能会遇到使用以下代码片段[].forEach.call() 迭代元素列表。该技术允许使用数组函数扩展类似数组的结构,例如 NodeList。空数组 [] 用于访问 forEach 函数,该...
    编程 发布于2024-12-22
  • 如何使用 CSS 创建具有固定标题的可滚动表格?
    如何使用 CSS 创建具有固定标题的可滚动表格?
    如何使用 CSS 和固定标题制作可滚动表格在 Web 开发中,通常需要创建包含大量数据的表格需要滚动。然而,在滚动表格主体的同时保持固定的标题可能具有挑战性。以下是实现此效果的方法:HTML 结构首先,我们必须确保 HTML 结构正确。我们有一个带滚动条的外部 div,一个包含表格的内部 div,表...
    编程 发布于2024-12-22

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3