」工欲善其事,必先利其器。「—孔子《論語.錄靈公》
首頁 > 程式設計 > 使用express js和react js unsing jwt進行身份驗證

使用express js和react js unsing jwt進行身份驗證

發佈於2024-11-01
瀏覽:229

Authentication with express js and react js unsing jwt

要在前端使用 React.js、在后端使用 Express.js 创建身份验证系统,我们需要实现以下内容:

  • 前端(React.js with Pulsy):处理登录和注销,维护用户身份验证状态,并保留令牌。
  • 后端 (Express.js):提供身份验证端点(例如登录、注销、用户验证)。

第 1 步:后端 (Express.js) 设置

让我们从后端开始处理用户身份验证和令牌生成。

安装所需的软件包


npm install express bcryptjs jsonwebtoken cors


后端代码 (Express.js)

创建 authController.js 文件来处理身份验证逻辑:


// authController.js
const bcrypt = require('bcryptjs');
const jwt = require('jsonwebtoken');

// Mock user data (in production, you would use a real database)
const users = [
  {
    id: 1,
    username: 'john',
    password: '$2a$10$O1s8iLKRLPbPqhc1uTquLO.xODTC1U/Z8xGoEDU6/Dc0PAQ3MkCKy', // hashed password for 'password123'
  },
];

// JWT Secret
const JWT_SECRET = 'supersecretkey';

exports.login = (req, res) => {
  const { username, password } = req.body;

  const user = users.find((u) => u.username === username);

  if (!user) {
    return res.status(401).json({ error: 'Invalid credentials' });
  }

  bcrypt.compare(password, user.password, (err, isMatch) => {
    if (isMatch) {
      // Create a token
      const token = jwt.sign({ id: user.id, username: user.username }, JWT_SECRET, { expiresIn: '1h' });
      return res.json({ token });
    } else {
      return res.status(401).json({ error: 'Invalid credentials' });
    }
  });
};

exports.validateToken = (req, res) => {
  const token = req.header('Authorization').replace('Bearer ', '');

  if (!token) {
    return res.status(401).json({ error: 'No token provided' });
  }

  try {
    const decoded = jwt.verify(token, JWT_SECRET);
    res.json({ user: { id: decoded.id, username: decoded.username } });
  } catch (err) {
    res.status(401).json({ error: 'Invalid token' });
  }
};


接下来,创建用于设置 Express 的主 server.js 文件:


// server.js
const express = require('express');
const cors = require('cors');
const { login, validateToken } = require('./authController');

const app = express();
app.use(express.json());
app.use(cors());

// Authentication routes
app.post('/api/login', login);
app.get('/api/validate', validateToken);

// Start the server
const PORT = 5000;
app.listen(PORT, () => {
  console.log(`Server running on port ${PORT}`);
});


  • POST /api/login:对用户进行身份验证并返回 JWT 令牌。
  • GET /api/validate:验证令牌并返回用户信息。

第 2 步:前端(React.js 和 Pulsy)

现在让我们使用 React.jsPulsy 设置前端来处理身份验证状态。

安装所需的软件包


npm install axios pulsy


脉冲商店设置

我们将创建一个 Pulsy 商店来管理全球身份验证状态。


// authStore.js
import { createStore, addMiddleware } from 'pulsy';
import axios from 'axios';

// Create a store to hold the user and token
createStore('auth', {
  user: null,
  token: null,
}, { persist: true }); // Persist the auth state in localStorage

// Middleware to add Authorization header for authenticated requests
addMiddleware('auth', (nextValue, prevValue, storeName) => {
  if (nextValue.token) {
    axios.defaults.headers.common['Authorization'] = `Bearer ${nextValue.token}`;
  } else {
    delete axios.defaults.headers.common['Authorization'];
  }
  return nextValue;
});


此存储将保留身份验证状态(用户和令牌)并自动为经过身份验证的请求应用授权标头。

认证功能

创建辅助函数来处理登录和验证请求:


// authService.js
import { setStoreValue } from 'pulsy';
import axios from 'axios';

const API_URL = 'http://localhost:5000/api';

export const login = async (username, password) => {
  try {
    const response = await axios.post(`${API_URL}/login`, { username, password });
    const { token } = response.data;

    // Set token and user info in Pulsy store
    setStoreValue('auth', { token, user: { username } });

    return true;
  } catch (error) {
    console.error('Login failed', error);
    return false;
  }
};

export const validateToken = async () => {
  try {
    const response = await axios.get(`${API_URL}/validate`);
    const user = response.data.user;

    // Update the store with the user info
    setStoreValue('auth', { user, token: localStorage.getItem('auth_token') });
    return true;
  } catch (error) {
    console.error('Token validation failed', error);
    return false;
  }
};

export const logout = () => {
  setStoreValue('auth', { user: null, token: null });
  localStorage.removeItem('pulsy_auth');
};


第3步:创建身份验证组件

现在让我们为登录和经过身份验证的视图创建 React 组件。

登录组件


// Login.js
import React, { useState } from 'react';
import { login } from './authService';

const Login = () => {
  const [username, setUsername] = useState('');
  const [password, setPassword] = useState('');
  const [error, setError] = useState('');

  const handleLogin = async (e) => {
    e.preventDefault();
    const success = await login(username, password);
    if (!success) {
      setError('Invalid credentials. Try again.');
    }
  };

  return (
    

Login

setUsername(e.target.value)} placeholder="Username" required /> setPassword(e.target.value)} placeholder="Password" required />
{error &&

{error}

}
); }; export default Login;

已验证组件


// Dashboard.js
import React from 'react';
import { usePulsy } from 'pulsy';
import { logout } from './authService';

const Dashboard = () => {
  const [auth] = usePulsy('auth');

  const handleLogout = () => {
    logout();
    window.location.reload(); // Simple page refresh to redirect to login
  };

  return (
    

Welcome, {auth.user.username}!

); }; export default Dashboard;

第 4 步:应用程序组件

在 App.js 组件中,您需要检查用户是否经过身份验证并有条件地呈现登录名或仪表板。


// App.js
import React, { useEffect } from 'react';
import { usePulsy } from 'pulsy';
import { validateToken } from './authService';
import Login from './Login';
import Dashboard from './Dashboard';

function App() {
  const [auth] = usePulsy('auth');

  useEffect(() => {
    // Check token validity on app load
    if (auth.token) {
      validateToken();
    }
  }, [auth.token]);

  return (
    
{auth.user ? : }
); } export default App;

第 5 步:运行应用程序

现在我们已经设置了后端前端,您可以运行该应用程序了。

  1. 启动 Express 服务器:

   node server.js


  1. 启动React前端:

   npm start


两者都运行后:

  • 您可以访问http://localhost:3000查看登录页面。
  • 登录后,身份验证令牌将被保存,您将被重定向到仪表板。
  • 如果令牌有效,您将保持登录状态,否则,您将被重定向回登录页面。

概括

此示例展示了如何将 Pulsy 与 Express.js API 支持的 React 身份验证系统集成。 Pulsy 可帮助您管理身份验证的全局状态,包括身份验证令牌和跨会话的用户数据的持久性,使其成为功能强大且易于使用的状态管理工具。

版本聲明 本文轉載於:https://dev.to/ng_dream_3e53e6a868268e4d/authentication-with-express-js-and-react-js-unsing-jwt-4nbp?1如有侵犯,請聯絡[email protected]刪除
最新教學 更多>
  • 您應該避免的錯誤(以及如何修復它們)
    您應該避免的錯誤(以及如何修復它們)
    身為 React 開發人員,很容易陷入某些編碼模式,這些模式一開始看起來很方便,但最終可能會導致問題。在這篇文章中,我們將探討 5 個常見的 React 錯誤,並討論如何避免它們,確保您的程式碼保持高效、可維護和可擴展。 1. 濫用關鍵道具 錯誤: {myList.map((ite...
    程式設計 發佈於2024-11-07
  • 如何在 PHP 中存取 JavaScript 變數值?
    如何在 PHP 中存取 JavaScript 變數值?
    在PHP 中使用JavaScript 變數值使用同時涉及JavaScript 和PHP 的Web 應用程式時,通常需要在兩者之間交換資料兩種語言。然而,由於語言的執行環境不同,直接在 PHP 中存取 JavaScript 變數是不可能的。 PHP 在伺服器端執行,而 JavaScript 在客戶端運...
    程式設計 發佈於2024-11-07
  • Popver API VS 對話方塊模態:相同但不同
    Popver API VS 對話方塊模態:相同但不同
    我在閱讀一些科技新聞部落格時偶然發現標題 Popover API 登陸 Baseline。我很困惑,在我最近深入前端開發期間,我最近很難習慣在 HTML 中使用 Elements。在瀏覽部落格時,我一直對到目前為止我如何使用該元素感到困惑。 長話短說 選擇: 需要使用者焦點的模態彈...
    程式設計 發佈於2024-11-07
  • Go中不嵌入結構體可以實現方法繼承嗎?
    Go中不嵌入結構體可以實現方法繼承嗎?
    嵌入式結構:方法繼承的探索理解Go 中的方法繼承 In在Go 中,將方法從一種類型繼承到另一種類型的能力主要是透過嵌入結構來實現的。此技術涉及將一個結構嵌入另一個結構,允許外部結構存取和利用嵌入結構的方法。 嵌入結構的範例考慮以下內容程式碼片段:type Properties map[string]...
    程式設計 發佈於2024-11-07
  • 如何在 PHP 中的 Foreach 迴圈中檢索數組鍵
    如何在 PHP 中的 Foreach 迴圈中檢索數組鍵
    在Foreach 循環期間檢索數組鍵:PHP在PHP 中使用數組時,通常需要檢索其中的鍵和值foreach 循環。 key() 函數提供了一種在迭代期間存取當前鍵的便捷方法。但是,在某些情況下,它可能不會產生所需的結果。 考慮以下程式碼,其目的是從範例陣列產生 HTML 表:foreach($sam...
    程式設計 發佈於2024-11-07
  • 在 JavaScript 中建立物件的方法
    在 JavaScript 中建立物件的方法
    介紹 在 JavaScript 中建立物件的方法有很多種。 對象字面量 Object() 建構子 Object.create() 建構子 ES6 類 對象字面量 這可能是在 JavaScript 中建立物件最快、最簡單的方法。這也稱為物件初始值設定項,是一個由零對...
    程式設計 發佈於2024-11-07
  • 如何在 JavaScript 中擴充自訂異常的錯誤物件?
    如何在 JavaScript 中擴充自訂異常的錯誤物件?
    擴充JavaScript 中的錯誤物件在JavaScript 中拋出例外時,可能想要擴充內建Error 物件以建立自訂錯誤類型。這允許更具體和資訊豐富的異常處理。 在JavaScript 中,繼承不是透過子類化與Python 不同,在Python 中,異常通常是從Exception 基類進行子類化的...
    程式設計 發佈於2024-11-07
  • MySQL如何保證並發操作時資料的完整性?
    MySQL如何保證並發操作時資料的完整性?
    MySQL 並發:確保資料完整性如果您的MySQL 資料庫使用InnoDB 儲存引擎,您可能會擔心在執行過程中潛在的並發問題。同時記錄更新或插入。本文探討了 MySQL 如何處理並發以及是否需要在應用程式中加入額外的處理。 MySQL 的同時處理MySQL 採用原子性,這意味著單獨的 SQL 語句是...
    程式設計 發佈於2024-11-07
  • 如何使用 Go 在 SQL 查詢中有效連接字串和值?
    如何使用 Go 在 SQL 查詢中有效連接字串和值?
    在Go 中有效地製作SQL 查詢在Go 中將字串與文字SQL 查詢中的值連接起來可能有點棘手。與 Python 不同,Go 的字串格式化語法行為不同,導致常見錯誤,如此處遇到的錯誤。 元組語法錯誤初始程式碼片段嘗試使用 Python -style 元組,Go 中不支援。這會導致語法錯誤:query ...
    程式設計 發佈於2024-11-07
  • 為什麼 json_encode() 無法使用 Latin1 編碼對 MySQL 資料庫中的重音字元進行編碼?
    為什麼 json_encode() 無法使用 Latin1 編碼對 MySQL 資料庫中的重音字元進行編碼?
    MySQL 中UTF-8 字元的JSON 編碼難題當嘗試使用latin1_swedish_ci 編碼從資料庫中檢索重音字元並使用json_encode() 將它們編碼為JSON 時,結果可能出乎意料。預期結果(例如“Abord â Plouffe”)會轉換為“null”,從而使編碼的 JSON 無效...
    程式設計 發佈於2024-11-07
  • 如何在 MySQL 中將行轉置為列:綜合指南
    如何在 MySQL 中將行轉置為列:綜合指南
    在MySQL 中將行轉換為列在MySQL 中將行轉換為列在MySQL 查詢中將行轉換為列需要在應用程式中執行複雜的查詢或手動操作。 GROUP_CONCAT 解雖然 GROUP_CONCAT 可以將行轉換為單列,但它不提供整個結果集所需的轉置。 手動查詢方法對於更複雜的轉置,需要細緻的查詢,從原始行...
    程式設計 發佈於2024-11-07
  • 如何解決iOS後台模式下未收到GCM通知的問題
    如何解決iOS後台模式下未收到GCM通知的問題
    當應用程式在iOS 上處於後台模式時未收到GCM 通知當iOS 在後台收到通知但不處理時,會出現此問題它們在使用者介面中。若要解決此問題,請確保您的應用程式:啟用後台推播通知:檢查您的應用程式是否已要求並取得在背景接收推播通知的權限。 設定徽章應用程式圖示:驗證是否在應用程式的「設定」>「通知」部分...
    程式設計 發佈於2024-11-07
  • 為什麼在 Windows 7 中使用 CLASSPATH 時出現 ClassNotFoundException?
    為什麼在 Windows 7 中使用 CLASSPATH 時出現 ClassNotFoundException?
    儘管使用CLASSPATH 環境變數仍解決java.lang.ClassNotFoundException在Windows 7 中嘗試使用Java 連線至MySQL 資料庫時,設定CLASSPATH 環境變數以包含JDBC 驅動程式jar 檔案的路徑似乎無法解決java.lang.ClassNotF...
    程式設計 發佈於2024-11-07
  • 開發人員需要了解免費外匯 API
    開發人員需要了解免費外匯 API
    如果您是一名开发人员,您一定正在寻找可以帮助您更轻松地工作的工具,对吗?免费的外汇 API 就是其中之一!它使您无需支付任何费用即可获取外汇汇率。但是,许多开发人员对这些 API 不太了解。因此,本文旨在解释什么是免费外汇 API、它为何有用以及如何为您的项目选择一个 API。 什么是免费外汇 A...
    程式設計 發佈於2024-11-07
  • 如何使用 JavaScript 將字串中每個單字的首字母大寫?
    如何使用 JavaScript 將字串中每個單字的首字母大寫?
    使用JavaScript 將字串中每個單字的首字母大寫在JavaScript 中,將字串中每個單字的首字母大寫可以透過多種方法來實現。一種常見的方法是使用將給定字串轉換為標題大小寫的函數。 讓我們探索一個示範此技術的程式碼範例:function titleCase(str) { var spli...
    程式設計 發佈於2024-11-07

免責聲明: 提供的所有資源部分來自互聯網,如果有侵犯您的版權或其他權益,請說明詳細緣由並提供版權或權益證明然後發到郵箱:[email protected] 我們會在第一時間內為您處理。

Copyright© 2022 湘ICP备2022001581号-3