受楼主启发,我也写了个cloudflare worker版的。
大家可以在这里体验:
aiuuo.com
再次感谢楼主。
由于是纯前端实现,根本不走后端请求。所以可以把html抠出来自己用。
代码
export default {
async fetch(request, env, ctx) {
const html = `
<!DOCTYPE html>
<html lang="zh-CN-Hans">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<link rel="icon" href="" type="image/x-icon">
<title>简易解析ChatGPT的对话JSON</title>
<style>
html {font-size: 16px;}
@media (max-width: 600px) {html {font-size: 14px;}}
body {margin: 0; padding: 0; font-family: Arial, sans-serif; background-color: #f5f5f5;}
.container {
display: flex; flex-direction: column; align-items: center;
width: 100%; max-width: 800px; margin: 0 auto; padding: 2rem;
transition: padding 0.5s ease;
}
.container.loaded {align-items: flex-start; padding: 1rem;}
#title {
font-size: 2rem; text-align: center; margin-bottom: 2rem;
transition: all 0.5s ease;
}
.container.loaded #title {
font-size: 1.5rem; text-align: left; margin-bottom: 1rem;
}
.file-input-container {
display: flex; justify-content: center; align-items: center; margin-bottom: 2rem;
transition: opacity 0.5s ease;
}
.container.loaded .file-input-container {opacity: 0; height: 0; overflow: hidden; margin-bottom: 0;}
.file-input {
padding: 0.5rem 1rem; font-size: 1rem; cursor: pointer; border: 1px solid #ccc; border-radius: 0.25rem;
background-color: #f5f5f5; color: #333; transition: background-color 0.3s ease;
}
.file-input:hover {background-color: #e0e0e0;}
.message {
position: relative; background-color: rgba(255, 255, 255, 0.9); border-radius: 0.5rem; padding: 1rem; margin-bottom: 1rem;
box-shadow: 0 0 0.5rem rgba(0, 0, 0, 0.05); word-wrap: break-word;
overflow: hidden;
transition: max-height 0.3s ease;
}
.system {background-color: rgba(255, 255, 255, 0.9);}
.user {background-color: rgba(255, 224, 178, 0.3);}
.tool {background-color: rgba(220, 237, 200, 0.3); cursor: pointer;}
.label {font-weight: bold; font-size: 1.1rem;}
.header {display: flex; justify-content: space-between; align-items: center; margin-bottom: 0.5rem;}
.timestamp {font-size: 0.875rem; color: #666;}
.content {margin-top: 0.5rem; transition: max-height 0.3s ease;}
.collapsed .content {max-height: 6rem; overflow: hidden;}
.message.collapsed::after {
content: ''; position: absolute; bottom: 0; left: 0; width: 100%; height: 50%;
background: linear-gradient(to bottom, rgba(255,255,255,0), rgba(255,255,255,0.9));
pointer-events: none;
border-bottom-left-radius: 0.5rem;
border-bottom-right-radius: 0.5rem;
}
pre {background-color: #f0f0f0; padding: 1rem; border-radius: 0.5rem; overflow-x: auto; white-space: pre-wrap; word-wrap: break-word; position: relative;}
.copy-button {
position: absolute; top: 0.5rem; right: 0.5rem; padding: 0.2rem 0.5rem; font-size: 0.875rem;
background-color: #e0e0e0; border: none; border-radius: 0.25rem; cursor: pointer;
transition: background-color 0.3s ease, transform 0.3s ease;
z-index: 1;
}
.copy-button:hover {background-color: #d5d5d5;}
.copy-button.copied {
background-color: #a5d6a7;
transform: scale(1.1);
}
#title-container {
display: flex; justify-content: center; align-items: center; width: 100%;
transition: all 0.5s ease;
}
.hidden {display: none;}
</style>
</head>
<body>
<div class="container" id="container">
<div id="title-container">
<h1 id="title">简易解析ChatGPT的对话JSON</h1>
</div>
<div class="file-input-container">
<input type="file" id="fileInput" class="file-input" accept=".json" />
</div>
<div id="content"></div>
</div>
<script defer src="https://cdn.jsdelivr.net/npm/marked/marked.min.js"></script>
<script defer>
function formatTimestamp(ts) {
const date = new Date(ts * 1000);
const y = date.getFullYear();
const m = String(date.getMonth() + 1).padStart(2, '0');
const d = String(date.getDate()).padStart(2, '0');
const hh = String(date.getHours()).padStart(2, '0');
const mm = String(date.getMinutes()).padStart(2, '0');
const ss = String(date.getSeconds()).padStart(2, '0');
return \`\${y}年\${m}月\${d}日 \${hh}:\${mm}:\${ss}\`;
}
function addCopyButtons() {
const pres = document.querySelectorAll('pre');
pres.forEach(pre => {
const code = pre.querySelector('code');
if (code) {
const button = document.createElement('button');
button.classList.add('copy-button');
button.textContent = 'Copy';
button.addEventListener('click', function(event) {
event.stopPropagation();
navigator.clipboard.writeText(code.textContent).then(() => {
button.textContent = 'Copied!';
button.classList.add('copied');
setTimeout(() => {
button.textContent = 'Copy';
button.classList.remove('copied');
}, 2000);
});
});
pre.appendChild(button);
}
});
}
document.getElementById('fileInput').addEventListener('change', function(event) {
const file = event.target.files[0];
if (!file) return;
const reader = new FileReader();
reader.onload = function(e) {
try {
const json = JSON.parse(e.target.result);
const messages = json.messages || [];
const contentDiv = document.getElementById('content');
contentDiv.innerHTML = '';
messages.forEach(msg => {
const role = msg.author.role;
const parts = msg.content.parts || [];
const createTime = msg.create_time;
const timeStr = createTime ? formatTimestamp(createTime) : '';
parts.forEach(part => {
if (part.trim() === '') return;
const messageDiv = document.createElement('div');
messageDiv.classList.add('message');
let label = '';
if (role === 'system' || role === 'assistant') {messageDiv.classList.add('system'); label = 'ChatGPT';}
else if (role === 'user') {messageDiv.classList.add('user'); label = 'User';}
else if (role === 'tool') {messageDiv.classList.add('tool'); label = '(思考 ChatGPT)';}
const headerDiv = document.createElement('div');
headerDiv.classList.add('header');
const labelDiv = document.createElement('div');
labelDiv.classList.add('label');
labelDiv.textContent = label;
const timeDiv = document.createElement('div');
timeDiv.classList.add('timestamp');
timeDiv.textContent = timeStr;
headerDiv.appendChild(labelDiv);
headerDiv.appendChild(timeDiv);
const contentDivInner = document.createElement('div');
contentDivInner.classList.add('content');
contentDivInner.innerHTML = marked.parse(part);
messageDiv.appendChild(headerDiv);
messageDiv.appendChild(contentDivInner);
if (role === 'tool') {
messageDiv.classList.add('collapsed');
messageDiv.addEventListener('click', function() {
if (messageDiv.classList.contains('collapsed')) {
const contentHeight = contentDivInner.scrollHeight;
contentDivInner.style.maxHeight = contentHeight + 'px';
messageDiv.classList.remove('collapsed');
contentDivInner.addEventListener('transitionend', function handler() {
messageDiv.style.setProperty('--mask-display', 'none');
messageDiv.removeEventListener('transitionend', handler);
});
} else {
const currentHeight = contentDivInner.scrollHeight;
contentDivInner.style.maxHeight = currentHeight + 'px';
void contentDivInner.offsetHeight;
contentDivInner.style.maxHeight = '6rem';
messageDiv.classList.add('collapsed');
}
});
}
contentDiv.appendChild(messageDiv);
});
});
const container = document.getElementById('container');
container.classList.add('loaded');
addCopyButtons();
} catch (err) {
alert('无效的JSON文件');
}
};
reader.readAsText(file);
const title = document.getElementById('title');
const titleContainer = document.getElementById('title-container');
titleContainer.style.transition = 'all 0.5s ease';
titleContainer.style.display = 'flex';
titleContainer.style.justifyContent = 'center';
titleContainer.style.alignItems = 'center';
setTimeout(() => {
titleContainer.style.justifyContent = 'flex-start';
}, 10);
});
</script>
</body>
</html>`;
return new Response(html, {
headers: { 'Content-Type': 'text/html' },
});
},
};