后端部分,使用node_js:
const express = require('express');
const mysql = require('mysql2/promise'); // 使用 promise 版本
const cors = require('cors');
const bodyParser = require('body-parser');
const rateLimit = require('express-rate-limit');
const app = express();
// 设置 trust proxy
app.set('trust proxy', 1);
// 配置请求频率限制
const limiter = rateLimit({
windowMs: 15 * 60 * 1000,
max: 100,
message: '请求过于频繁,请稍后再试。',
standardHeaders: true,
legacyHeaders: false,
});
app.use(limiter);
app.use(cors({
origin: '*',
methods: ['GET', 'POST'],
allowedHeaders: ['Content-Type']
}));
app.use(bodyParser.json());
app.use(express.static('public'));
// 数据库连接配置
const dbConfig = {
host: process.env.DB_HOST || 'localhost',
port: process.env.DB_PORT || 3306,
user: process.env.DB_USER || 'root',
password: process.env.DB_PASSWORD || '',
database: process.env.DB_NAME || 'mydb',
connectTimeout: 60000,
acquireTimeout: 60000,
timeout: 60000,
connectionLimit: 10,
waitForConnections: true,
queueLimit: 0
};
// 创建连接池
const pool = mysql.createPool(dbConfig);
// 测试数据库连接
async function testConnection() {
try {
const connection = await pool.getConnection();
console.log('已成功连接到 MySQL 数据库');
connection.release();
} catch (err) {
console.error('数据库连接失败:', err);
setTimeout(testConnection, 5000);
}
}
// 初始测试连接
testConnection();
// API 路由
app.post('/api/get-article', async (req, res) => {
try {
const { articleId, password } = req.body;
// 输入验证
if (!articleId || !password) {
return res.status(400).json({
success: false,
error: '缺少必要参数'
});
}
// 验证 articleId 是否为有效的整数
if (!Number.isInteger(Number(articleId)) || articleId <= 0) {
return res.status(400).json({
success: false,
error: '无效的文章ID'
});
}
// 执行查询
const [rows] = await pool.execute(
'SELECT id, title, content FROM articles WHERE id = ? AND password = ?',
[articleId, password]
);
if (!rows || rows.length === 0) {
return res.status(401).json({
success: false,
error: '密码错误或文章不存在'
});
}
// 成功响应
res.json({
success: true,
article: {
id: rows[0].id,
title: rows[0].title,
content: rows[0].content
}
});
} catch (error) {
console.error('查询错误:', error);
res.status(500).json({
success: false,
error: '服务器内部错误'
});
}
});
//关闭连接池
process.on('SIGINT', async () => {
try {
await pool.end();
console.log('数据库连接池已关闭');
process.exit(0);
} catch (err) {
console.error('关闭连接池时出错:', err);
process.exit(1);
}
});
// 启动服务器
const PORT = process.env.PORT || 3000;
app.listen(PORT, () => {
console.log(`服务器运行在 http://localhost:${PORT}`);
});
数据库部分,使用mysql
-- 创建数据库
CREATE DATABASE IF NOT EXISTS blog_db;
USE blog_db;
-- 创建文章表
CREATE TABLE articles (
id INT PRIMARY KEY AUTO_INCREMENT,
title VARCHAR(255) NOT NULL,
content TEXT NOT NULL,
password VARCHAR(50) NOT NULL,
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
);
-- 插入测试数据
INSERT INTO articles (title, content, password)
VALUES ('测试文章', '这是一篇受密码保护的文章内容。这里可以是很长的文章内容...', '1234');
前端部分(支持markdown代码渲染)在verifyPassword
函数中配置后端连接信息
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>加密文章访问</title>
<!-- 使用 ES6 模块方式加载 marked -->
<script type="module">
import { marked } from 'https://cdn.jsdelivr.net/npm/marked/lib/marked.esm.js';
window.marked = marked;
marked.setOptions({
breaks: true,
gfm: true,
headerIds: true,
mangle: false,
sanitize: false
});
</script>
<!-- 加载 highlight.js -->
<script src="https://cdn.jsdelivr.net/npm/[email protected]/lib/highlight.min.js"></script>
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/[email protected]/styles/github.min.css">
<style>
/* 基础变量定义 */
:root {
--bg-color: inherit;
--bg-secondary: inherit;
--text-color: inherit;
--text-secondary: #476582;
--border-color: #eaecef;
--input-bg: #ffffff;
--focus-color: #3eaf7c;
--focus-shadow: rgba(62, 175, 124, 0.25);
--code-bg: #f6f8fa;
--code-block-bg: #282c34;
--code-block-color: #abb2bf;
--code-color: #476582;
--blockquote-color: #6a737d;
--blockquote-border: #42b983;
--table-alt-bg: #f6f8fa;
--link-color: #3eaf7c;
--error-bg: #fef0f0;
--error-color: #f56c6c;
--header-bg: linear-gradient(120deg, #155799, #159957);
--header-color: #fff;
--button-hover: #42d392;
--scrollbar-thumb: #c1c1c1;
--scrollbar-track: #f1f1f1;
--card-shadow: 0 4px 6px rgba(0, 0, 0, 0.1);
}
/* 全局样式重置 */
*, *::before, *::after {
box-sizing: border-box;
margin: 0;
padding: 0;
}
/* 滚动条美化 */
::-webkit-scrollbar {
width: 8px;
height: 8px;
}
::-webkit-scrollbar-track {
background: var(--scrollbar-track);
border-radius: 4px;
}
::-webkit-scrollbar-thumb {
background: var(--scrollbar-thumb);
border-radius: 4px;
}
::-webkit-scrollbar-thumb:hover {
background: #a8a8a8;
}
/* 基础样式 */
html {
font-size: 16px;
line-height: 1.6;
overflow-y: scroll;
}
body {
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell,
'Fira Sans', 'Droid Sans', 'Helvetica Neue', sans-serif;
width: 100%;
min-height: 100vh;
margin: 0;
padding: 0;
background-color: var(--bg-color);
color: var(--text-color);
}
/* 页面容器 */
.page-container {
width: 100%;
min-height: 100vh;
display: flex;
flex-direction: column;
}
/* 页头样式 */
.header {
color: var(--header-color);
padding: clamp(2rem, 5vw, 4rem) clamp(1rem, 3vw, 2rem);
text-align: center;
margin-bottom: 1rem;
}
.header h1 {
font-size: clamp(1.5rem, 4vw, 2.5rem);
margin-bottom: 1rem;
font-weight: 600;
}
.header p {
font-size: clamp(1rem, 2vw, 1.25rem);
opacity: 0.9;
max-width: 800px;
margin: 0 auto;
}
/* 主容器样式 */
.container {
width: 100%;
max-width: min(90vw, 1200px);
min-width: min(300px, 90vw);
margin: 0 auto;
padding: clamp(1rem, 3vw, 2rem);
flex: 1;
}
/* 卡片样式 */
.card {
background: var(--bg-color);
border-radius: 8px;
box-shadow: var(--card-shadow);
padding: clamp(1rem, 3vw, 2rem);
margin-bottom: 1rem;
transition: transform 0.3s ease, box-shadow 0.3s ease;
}
.card:hover {
transform: translateY(-2px);
box-shadow: 0 6px 12px rgba(0, 0, 0, 0.15);
}
/* 密码输入区域样式 */
.password-section {
max-width: 600px;
margin: 0 auto;
text-align: center;
}
.password-section h2 {
font-size: clamp(1.25rem, 3vw, 1.75rem);
margin-bottom: 1.5rem;
color: var(--text-color);
margin-top: 0;
}
.password-form {
display: flex;
flex-direction: column;
gap: 1rem;
align-items: center;
}
/* 输入框样式 */
.input-group {
width: 100%;
max-width: 400px;
position: relative;
}
input[type="password"] {
width: 100%;
padding: 0.75rem 1rem;
font-size: 1rem;
border: 2px solid var(--border-color);
border-radius: 6px;
background: var(--input-bg);
color: var(--text-color);
transition: all 0.3s ease;
}
input[type="password"]:focus {
outline: none;
border-color: var(--focus-color);
box-shadow: 0 0 0 3px var(--focus-shadow);
}
/* 按钮样式 */
.btn {
padding: 0.75rem 2rem;
font-size: 1rem;
font-weight: 500;
color: white;
background: var(--focus-color);
border: none;
border-radius: 6px;
cursor: pointer;
transition: all 0.3s ease;
min-width: 200px;
}
.btn:hover {
background: var(--button-hover);
transform: translateY(-1px);
}
.btn:active {
transform: translateY(0);
}
.btn:disabled {
background: var(--border-color);
cursor: not-allowed;
transform: none;
}
/* 错误消息样式 */
.error-message {
background: var(--error-bg);
color: var(--error-color);
padding: 0.75rem 1rem;
border-radius: 6px;
margin-top: 1rem;
font-size: 0.875rem;
display: none;
animation: fadeInUp 0.3s ease;
}
@keyframes fadeInUp {
from {
opacity: 0;
transform: translateY(10px);
}
to {
opacity: 1;
transform: translateY(0);
}
}
/* 文章内容区域样式 */
.article-content {
width: 100%;
max-width: 100%;
opacity: 0;
transform: translateY(20px);
transition: all 0.5s ease;
display: none;
}
.article-content.visible {
opacity: 1;
transform: translateY(0);
display: block;
}
.article-title {
font-size: clamp(1.5rem, 4vw, 2.5rem);
margin-bottom: 2rem;
color: var(--text-color);
text-align: center;
}
/* Markdown 内容样式 */
.markdown-content {
width: 100%;
max-width: 100%;
margin: 0 auto;
color: var(--text-color);
line-height: 1.8;
font-size: clamp(1rem, 1.1vw, 1.2rem);
}
.markdown-content h1,
.markdown-content h2,
.markdown-content h3,
.markdown-content h4,
.markdown-content h5,
.markdown-content h6 {
margin: 2rem 0 1rem;
line-height: 1.4;
color: var(--text-color);
font-weight: 600;
}
.markdown-content h1 { font-size: clamp(1.75rem, 2.5vw, 2rem); }
.markdown-content h2 { font-size: clamp(1.5rem, 2vw, 1.75rem); }
.markdown-content h3 { font-size: clamp(1.25rem, 1.8vw, 1.5rem); }
.markdown-content h4 { font-size: clamp(1.1rem, 1.6vw, 1.25rem); }
.markdown-content h5 { font-size: clamp(1rem, 1.4vw, 1.1rem); }
.markdown-content h6 { font-size: clamp(0.9rem, 1.2vw, 1rem); }
.markdown-content p {
margin: 1rem 0;
line-height: 1.8;
}
.markdown-content a {
color: var(--link-color);
text-decoration: none;
border-bottom: 1px solid transparent;
transition: all 0.3s ease;
}
.markdown-content a:hover {
border-bottom-color: var(--link-color);
}
.markdown-content code {
background: var(--code-bg);
color: var(--code-color);
padding: 0.2em 0.4em;
border-radius: 3px;
font-family: 'SF Mono', Monaco, Menlo, Consolas, 'Ubuntu Mono', monospace;
font-size: 0.9em;
}
.markdown-content pre {
background: var(--code-block-bg);
color: var(--code-block-color);
padding: 1rem 1.5rem;
border-radius: 6px;
overflow-x: auto;
margin: 1.5rem 0;
position: relative;
}
.markdown-content pre code {
background: transparent;
color: inherit;
padding: 0;
font-size: 0.9em;
line-height: 1.6;
}
.markdown-content blockquote {
margin: 1.5rem 0;
padding: 0.5rem 1.5rem;
border-left: 4px solid var(--blockquote-border);
background: var(--bg-secondary);
color: var(--blockquote-color);
border-radius: 0 6px 6px 0;
}
.markdown-content img {
max-width: 100%;
height: auto;
border-radius: 6px;
margin: 1.5rem 0;
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1);
}
.markdown-content table {
width: 100%;
margin: 1.5rem 0;
border-collapse: collapse;
border-spacing: 0;
display: block;
overflow-x: auto;
}
.markdown-content table th,
.markdown-content table td {
padding: 0.75rem 1rem;
border: 1px solid var(--border-color);
text-align: left;
}
.markdown-content table th {
background: var(--bg-secondary);
font-weight: 600;
}
.markdown-content table tr:nth-child(2n) {
background: var(--table-alt-bg);
}
.markdown-content ul,
.markdown-content ol {
padding-left: 1.5rem;
margin: 1rem 0;
}
.markdown-content li {
margin: 0.5rem 0;
}
.markdown-content hr {
height: 1px;
background: var(--border-color);
border: none;
margin: 2rem 0;
}
/* 响应式布局 */
@media screen and (max-width: 768px) {
.container {
padding: 1rem;
}
.card {
padding: 1.25rem;
}
.password-form {
padding: 0 1rem;
}
.btn {
width: 100%;
}
.markdown-content pre {
margin: 1rem -1.25rem;
border-radius: 0;
}
.markdown-content blockquote {
margin: 1rem -1.25rem;
border-radius: 0;
}
}
/* 打印样式 */
@media print {
.header, .password-section {
display: none;
}
.container {
max-width: none;
padding: 0;
}
.card {
box-shadow: none;
padding: 0;
}
.markdown-content {
font-size: 12pt;
}
.markdown-content pre,
.markdown-content code {
white-space: pre-wrap;
}
}
</style>
<div class="page-container">
<header class="header">
<h1>加密文章访问</h1>
<p>请输入密码以访问加密内容</p>
</header>
<main class="container">
<div class="card">
<div id="passwordSection" class="password-section">
<h2>验证访问密码</h2>
<div class="password-form">
<div class="input-group">
<input type="password" id="passwordInput" placeholder="请输入密码" autocomplete="off">
</div>
<button class="btn" onclick="verifyPassword()">验证密码</button>
<div id="errorMessage" class="error-message"></div>
</div>
</div>
<div id="articleContent" class="article-content">
<h1 id="articleTitle" class="article-title"></h1>
<div id="articleBody" class="markdown-content"></div>
</div>
</div>
</main>
</div>
<script>
// 更新主题和代码高亮
function updateTheme() {
const isDarkMode = window.matchMedia('(prefers-color-scheme: dark)').matches;
const highlightTheme = isDarkMode ?
'https://cdn.jsdelivr.net/npm/[email protected]/styles/atom-one-dark.min.css' :
'https://cdn.jsdelivr.net/npm/[email protected]/styles/github.min.css';
let highlightStylesheet = document.querySelector('link[data-highlight-theme]');
if (!highlightStylesheet) {
highlightStylesheet = document.createElement('link');
highlightStylesheet.setAttribute('rel', 'stylesheet');
highlightStylesheet.setAttribute('data-highlight-theme', 'true');
document.head.appendChild(highlightStylesheet);
}
highlightStylesheet.setAttribute('href', highlightTheme);
}
// 初始化主题
updateTheme();
// 监听系统主题变化
window.matchMedia('(prefers-color-scheme: dark)').addListener(updateTheme);
// 等待 marked 加载完成
window.addEventListener('load', () => {
if (typeof window.marked === 'undefined') {
console.error('marked 库加载失败');
return;
}
console.log('marked 库加载成功');
document.getElementById('passwordInput').focus();
});
function showError(message) {
const errorElement = document.getElementById('errorMessage');
errorElement.textContent = message;
errorElement.style.display = 'block';
if (errorElement.fadeTimeout) {
clearTimeout(errorElement.fadeTimeout);
}
errorElement.fadeTimeout = setTimeout(() => {
errorElement.style.display = 'none';
}, 3000);
if (message.includes('密码错误')) {
const passwordInput = document.getElementById('passwordInput');
passwordInput.value = '';
passwordInput.focus();
}
}
function showArticle(title, content) {
console.log('显示文章:', { title, content });
const passwordSection = document.getElementById('passwordSection');
passwordSection.style.opacity = '0';
passwordSection.style.transform = 'translateY(-20px)';
const header = document.querySelector('.header');
header.style.opacity = '0';
header.style.transform = 'translateY(-20px)';
setTimeout(() => {
passwordSection.style.display = 'none';
header.style.display = 'none';
// 设置标题
document.getElementById('articleTitle').textContent = title;
// 渲染 Markdown 内容
try {
const articleBody = document.getElementById('articleBody');
if (typeof window.marked === 'undefined') {
articleBody.textContent = content;
console.error('marked 库未加载,显示原始内容');
} else {
const htmlContent = window.marked.parse(content);
articleBody.innerHTML = htmlContent;
// 高亮代码块
articleBody.querySelectorAll('pre code').forEach((block) => {
hljs.highlightBlock(block);
});
}
} catch (error) {
console.error('Markdown 渲染错误:', error);
document.getElementById('articleBody').textContent = content;
}
const articleContent = document.getElementById('articleContent');
articleContent.style.display = 'block';
setTimeout(() => {
articleContent.classList.add('visible');
}, 10);
}, 300);
}
function verifyPassword() {
const password = document.getElementById('passwordInput').value;
const articleId = 1; // 这里可以根据需要修改文章 ID
const submitButton = document.querySelector('.btn');
submitButton.disabled = true;
submitButton.textContent = '验证中...';
//修改成自己的api
fetch('https://se.991198.xyz/api/get-article', {
method: 'POST',
headers: {
'Content-Type': 'application/json',
'Accept': 'application/json'
},
mode: 'cors',
credentials: 'omit',
body: JSON.stringify({ articleId, password })
})
.then(response => {
if (response.status === 401) {
return response.json().then(data => {
throw new Error(data.error || '密码错误或文章不存在');
});
}
if (!response.ok) {
throw new Error(`服务器错误 (${response.status})`);
}
return response.json();
})
.then(data => {
if (data.success) {
console.log('接收到的数据:', data);
showArticle(data.article.title, data.article.content);
}
})
.catch(error => {
console.error('请求失败:', error);
if (error.message.includes('密码错误') || error.message.includes('文章不存在')) {
showError(error.message);
} else if (error.name === 'TypeError') {
showError('无法连接到服务器,请检查网络连接');
} else {
showError('服务器错误,请稍后重试');
}
})
.finally(() => {
submitButton.disabled = false;
submitButton.textContent = '验证密码';
});
}
// 添加回车键提交功能
document.getElementById('passwordInput').addEventListener('keypress', function(e) {
if (e.key === 'Enter') {
verifyPassword();
}
});
// 错误处理
window.onerror = function(msg, url, line, col, error) {
console.error('全局错误:', {
message: msg,
url: url,
line: line,
column: col,
error: error
});
};
</script>
效果展示:
密码:2025
加密文章访问
请输入密码以访问加密内容
评论区