Files
nxxmdata/docs/API认证解决方案文档.md

17 KiB
Raw Blame History

API认证解决方案文档

1. 概述

本文档详细说明了如何解决https://ad.ningmuyun.com/bank/api/projects接口的认证问题401 Unauthorized和跨域资源共享CORS问题。文档提供了多种认证方案、实现示例和最佳实践帮助开发者在实际项目中解决类似问题。

2. API认证的常见解决方案

2.1 基本认证Basic Authentication

基本认证是最简单的HTTP认证方式通过在请求头中添加Base64编码的用户名和密码实现。

实现方式

// 在请求头中添加认证信息
const username = 'your-username';
const password = 'your-password';
const credentials = btoa(`${username}:${password}`);

fetch(url, {
  headers: {
    'Authorization': `Basic ${credentials}`
  }
})

优点:实现简单,易于理解 缺点:安全性较低,凭证可被解码,不适合生产环境

2.2 Bearer Token认证

Bearer Token是一种常用的令牌认证方式通常在OAuth 2.0流程中使用,通过在请求头中添加Bearer [token]实现。

实现方式

// 在请求头中添加Bearer Token
const token = 'your-access-token';

fetch(url, {
  headers: {
    'Authorization': `Bearer ${token}`
  }
})

优点:安全性较高,可设置过期时间,适合生产环境 缺点需要安全存储和管理token

2.3 OAuth 2.0认证

OAuth 2.0是一种开放标准的授权协议,提供多种授权流程(授权码、隐式、客户端凭证、资源所有者密码凭证)。

实现方式:根据具体授权流程实现,通常需要先获取授权码,再兑换访问令牌。

优点:安全性高,支持第三方授权,细粒度权限控制 缺点:实现复杂,学习成本较高

2.4 API Key认证

API Key是一种简单的认证方式通过在请求中添加特定的API密钥实现。

实现方式

// 在URL参数中添加API Key
const apiKey = 'your-api-key';
fetch(`${url}?apiKey=${apiKey}`)

// 或在请求头中添加API Key
fetch(url, {
  headers: {
    'X-API-Key': apiKey
  }
})

优点:实现简单,适合内部服务调用 缺点:密钥容易泄露,不适合复杂权限控制

3. Bearer Token认证实现示例

3.1 认证服务类实现

src/utils/api-auth-guide.js文件中实现完整的认证服务类:

/**
 * API认证服务类用于处理API请求的认证、授权和错误处理
 */
export class AuthService {
  constructor() {
    this.baseUrl = 'https://ad.ningmuyun.com/bank';
    this.tokenKey = 'auth_token';
    this.refreshTokenKey = 'refresh_token';
    this.tokenExpiryKey = 'token_expiry';
  }

  /**
   * 初始化认证服务
   */
  init() {
    // 检查token是否即将过期如果是则自动刷新
    this.checkTokenExpiry();
  }

  /**
   * 检查用户是否已认证
   */
  isAuthenticated() {
    const token = localStorage.getItem(this.tokenKey);
    const expiry = localStorage.getItem(this.tokenExpiryKey);
    
    // 检查token是否存在且未过期
    if (!token || !expiry) {
      return false;
    }
    
    // 检查token是否即将在5分钟内过期
    const now = Date.now();
    const willExpireSoon = parseInt(expiry) - now < 5 * 60 * 1000;
    
    if (willExpireSoon) {
      // 后台刷新token
      this.refreshToken().catch(err => {
        console.error('自动刷新token失败:', err);
      });
    }
    
    return now < parseInt(expiry);
  }

  /**
   * 用户登录
   * @param {string} username - 用户名
   * @param {string} password - 密码
   * @returns {Promise} 登录结果
   */
  async login(username, password) {
    try {
      const response = await fetch(`${this.baseUrl}/api/login`, {
        method: 'POST',
        headers: {
          'Content-Type': 'application/json'
        },
        body: JSON.stringify({ username, password })
      });

      if (!response.ok) {
        throw new Error(`登录失败: ${response.status}`);
      }

      const data = await response.json();
      
      // 存储token和过期时间
      if (data.token && data.refreshToken && data.expiresIn) {
        const expiryTime = Date.now() + data.expiresIn * 1000;
        localStorage.setItem(this.tokenKey, data.token);
        localStorage.setItem(this.refreshTokenKey, data.refreshToken);
        localStorage.setItem(this.tokenExpiryKey, expiryTime.toString());
        return true;
      }
      
      throw new Error('登录响应不包含有效token');
    } catch (error) {
      console.error('登录错误:', error);
      throw error;
    }
  }

  /**
   * 用户登出
   */
  logout() {
    localStorage.removeItem(this.tokenKey);
    localStorage.removeItem(this.refreshTokenKey);
    localStorage.removeItem(this.tokenExpiryKey);
  }

  /**
   * 刷新访问令牌
   * @returns {Promise} 刷新结果
   */
  async refreshToken() {
    try {
      const refreshToken = localStorage.getItem(this.refreshTokenKey);
      
      if (!refreshToken) {
        throw new Error('没有可用的刷新令牌');
      }

      const response = await fetch(`${this.baseUrl}/api/refresh-token`, {
        method: 'POST',
        headers: {
          'Content-Type': 'application/json'
        },
        body: JSON.stringify({ refreshToken })
      });

      if (!response.ok) {
        throw new Error(`刷新token失败: ${response.status}`);
      }

      const data = await response.json();
      
      if (data.token && data.expiresIn) {
        const expiryTime = Date.now() + data.expiresIn * 1000;
        localStorage.setItem(this.tokenKey, data.token);
        if (data.refreshToken) {
          localStorage.setItem(this.refreshTokenKey, data.refreshToken);
        }
        localStorage.setItem(this.tokenExpiryKey, expiryTime.toString());
        return true;
      }
      
      throw new Error('刷新token响应不包含有效数据');
    } catch (error) {
      console.error('刷新token错误:', error);
      // 刷新失败时,清除所有认证信息
      this.logout();
      throw error;
    }
  }

  /**
   * 发送带认证的请求
   * @param {string} url - 请求URL
   * @param {object} options - fetch选项
   * @returns {Promise} 请求结果
   */
  async request(url, options = {}) {
    // 确保URL是完整的
    const fullUrl = url.startsWith('http') ? url : `${this.baseUrl}${url}`;
    
    // 确保options有默认值
    options = { ...options };
    options.headers = { ...options.headers };
    
    // 检查是否已认证
    if (!this.isAuthenticated()) {
      throw new Error('用户未认证,请先登录');
    }
    
    // 添加认证头
    const token = localStorage.getItem(this.tokenKey);
    options.headers['Authorization'] = `Bearer ${token}`;
    
    try {
      const response = await fetch(fullUrl, options);
      
      // 处理401错误尝试刷新token后重试
      if (response.status === 401) {
        try {
          // 尝试刷新token
          await this.refreshToken();
          
          // 刷新成功后使用新token重新请求
          options.headers['Authorization'] = `Bearer ${localStorage.getItem(this.tokenKey)}`;
          const retryResponse = await fetch(fullUrl, options);
          return this.handleResponse(retryResponse);
        } catch (refreshError) {
          // 刷新token失败抛出原始401错误
          console.error('刷新token后重试失败:', refreshError);
          throw new Error('认证失败,请重新登录');
        }
      }
      
      return this.handleResponse(response);
    } catch (error) {
      console.error('API请求错误:', error);
      throw error;
    }
  }

  /**
   * 处理响应
   * @param {Response} response - fetch响应对象
   * @returns {Promise} 处理后的响应数据
   */
  async handleResponse(response) {
    if (!response.ok) {
      throw new Error(`请求失败: ${response.status} ${response.statusText}`);
    }
    
    try {
      return await response.json();
    } catch (e) {
      throw new Error('响应数据格式错误');
    }
  }

  /**
   * 检查token是否即将过期
   */
  checkTokenExpiry() {
    const checkInterval = setInterval(() => {
      if (!this.isAuthenticated()) {
        // token已过期或无效
        clearInterval(checkInterval);
        // 这里可以触发登录提示
        console.warn('认证已过期,请重新登录');
      }
    }, 60000); // 每分钟检查一次
  }
}

3.2 在组件中使用认证服务

Home.vue组件中集成和使用认证服务:

// 导入认证服务
import { AuthService } from '../utils/api-auth-guide.js';

// 在组件中使用
export default {
  data() {
    return {
      // ...现有数据
      authService: null
    };
  },
  created() {
    // 初始化认证服务
    this.authService = new AuthService();
    this.authService.init();
    
    // 加载项目数据
    this.fetchProjectsData();
  },
  methods: {
    async fetchProjectsData() {
      try {
        // 检查是否已认证,未认证则尝试登录
        if (!this.authService.isAuthenticated()) {
          try {
            // 注意:实际项目中应该通过用户界面获取凭据,而不是硬编码
            await this.authService.login('admin', 'password');
          } catch (loginError) {
            console.error('登录失败:', loginError);
            // 登录失败时使用模拟数据
            this.setEnhancedMockProjectsData();
            return;
          }
        }
        
        // 使用认证服务调用API
        try {
          const data = await this.authService.request('https://ad.ningmuyun.com/bank/api/projects?page=1&limit=12&search=&status=');
          
          if (data.success && data.data && data.data.items) {
            console.log('成功获取API数据共', data.data.items.length, '条项目');
            this.projectsData = data.data.items.map(project => ({
              id: project.id,
              name: project.name || '未知项目',
              status: project.status || 'pending',
              statusText: this.projectStatusMap[project.status] || '未知状态',
              amount: project.amount ? `${project.amount}万元` : '0万元',
              expiryDate: project.expiryDate || '暂无日期'
            }));
          } else {
            console.warn('API返回数据格式不符合预期使用模拟数据...');
            this.setEnhancedMockProjectsData();
          }
        } catch (apiError) {
          console.error('API请求失败:', apiError);
          this.setEnhancedMockProjectsData();
        }
      } catch (error) {
        console.error('获取项目数据时发生异常:', error);
        this.setEnhancedMockProjectsData();
      }
    }
  }
};

4. CORS跨域问题的处理方案

4.1 了解CORS问题

跨域资源共享CORS是一种浏览器安全机制用于限制Web页面从不同域请求资源。当http://localhost:5175(前端开发服务器)尝试访问https://ad.ningmuyun.comAPI服务器如果服务器没有正确配置CORS响应头浏览器会阻止请求。

错误表现:

Access to fetch at 'https://ad.ningmuyun.com/bank/api/projects' from origin 'http://localhost:5175' has been blocked by CORS policy: The 'Access-Control-Allow-Origin' header has a value that is not equal to the supplied origin.

4.2 解决方案

4.2.1 服务器端配置CORS

最佳方案是在API服务器端配置CORS允许来自前端域名的请求

Nginx配置示例

location / {
    # 允许的源(可以是具体域名或*
    add_header Access-Control-Allow-Origin 'http://localhost:5175' always;
    # 允许的请求头
    add_header Access-Control-Allow-Headers 'Authorization,Content-Type' always;
    # 允许的方法
    add_header Access-Control-Allow-Methods 'GET,POST,PUT,DELETE,OPTIONS' always;
    # 允许携带凭证
    add_header Access-Control-Allow-Credentials 'true' always;
    
    # 处理OPTIONS预检请求
    if ($request_method = 'OPTIONS') {
        return 204;
    }
    
    # 其他配置...
}

Express.js配置示例

const express = require('express');
const cors = require('cors');
const app = express();

// 配置CORS
app.use(cors({
  origin: 'http://localhost:5175', // 允许的源
  methods: ['GET', 'POST', 'PUT', 'DELETE', 'OPTIONS'], // 允许的方法
  allowedHeaders: ['Authorization', 'Content-Type'], // 允许的请求头
  credentials: true // 允许携带凭证
}));

// 其他路由和中间件...

4.2.2 使用代理服务器

在开发环境中可以使用代理服务器转发API请求绕过浏览器的CORS限制

Vite配置示例vite.config.js

import { defineConfig } from 'vite';

export default defineConfig({
  // ...其他配置
  server: {
    proxy: {
      '/api': {
        target: 'https://ad.ningmuyun.com/bank',
        changeOrigin: true,
        rewrite: (path) => path.replace(/^\/api/, '')
      }
    }
  }
});

配置后,前端可以将请求发送到/api/projectsVite会自动将其代理到https://ad.ningmuyun.com/bank/projects

4.2.3 使用Chrome扩展或命令行参数

在开发环境中可以使用Chrome扩展如Allow CORS: Access-Control-Allow-Origin临时绕过CORS限制或使用以下命令行参数启动Chrome

chrome.exe --disable-web-security --user-data-dir="C:/ChromeDevSession"

注意:这仅适用于开发测试环境,生产环境中不应该使用。

4.2.4 使用fetch的credentials选项

在前端fetch请求中添加credentials选项允许跨域请求携带凭证

fetch(url, {
  credentials: 'include' // 发送跨域请求时携带凭证如cookie、HTTP认证信息等
})

5. 实际项目中的最佳实践

5.1 安全性考虑

  1. 不要在客户端存储敏感信息避免在localStorage或cookie中存储明文密码等敏感信息

  2. 使用HTTPS确保所有API请求都通过HTTPS加密传输防止中间人攻击

  3. token安全管理

    • 为token设置合理的过期时间
    • 实现自动刷新token机制
    • 在用户登出或检测到异常时及时清除token
  4. 避免硬编码凭证:不要在代码中硬编码用户名和密码,应通过安全的方式(如环境变量或用户输入)获取

5.2 错误处理

  1. 统一的错误处理机制实现全局错误处理中间件统一处理API请求错误

  2. 明确的错误信息:向用户提供清晰的错误提示,同时在控制台记录详细的错误信息

  3. 降级策略在API不可用或认证失败时提供适当的降级方案如使用模拟数据

  4. 重试机制:对于临时性错误(如网络波动),实现合理的重试逻辑

5.3 性能优化

  1. 缓存策略对不经常变化的数据实施客户端缓存减少API调用次数

  2. 懒加载和分页:对大量数据使用懒加载和分页,提高加载速度和用户体验

  3. 批量请求将多个相关API请求合并为批处理请求减少网络开销

  4. 监控和日志添加API请求监控和日志记录及时发现和解决性能问题

6. 临时解决方案(模拟数据)

在开发和测试过程中如果API认证或CORS问题暂时无法解决可以使用模拟数据作为临时替代方案

// 增强版模拟数据生成函数
setEnhancedMockProjectsData() {
  // 生成模拟项目数据
  this.projectsData = [
    {
      id: 1,
      name: '宁夏银川肉牛养殖基地项目',
      status: 'active',
      statusText: '正常运行',
      amount: '1280万元',
      expiryDate: '2025-12-31'
    },
    {
      id: 2,
      name: '中卫市活体羊抵押融资项目',
      status: 'warning',
      statusText: '即将到期',
      amount: '850万元',
      expiryDate: '2024-06-15'
    },
    // ...更多模拟数据
  ];
}

注意模拟数据仅适用于开发测试阶段生产环境中应使用真实API。

7. 总结

解决API认证和CORS问题需要前后端协同工作

  1. 选择合适的认证方案根据项目需求选择Basic Auth、Bearer Token、OAuth 2.0或API Key

  2. 正确实现认证逻辑:使用安全的方式存储和管理认证凭证,实现自动刷新机制

  3. 配置服务器端CORS在API服务器端正确配置CORS响应头允许来自前端的请求

  4. 使用代理和开发工具在开发环境中使用代理服务器或浏览器扩展绕过CORS限制

  5. 实现降级策略在API不可用时提供模拟数据确保应用仍能正常运行

通过以上措施可以有效解决API认证和CORS问题确保前后端通信的安全和稳定。