-
Notifications
You must be signed in to change notification settings - Fork 3
Open
Description
I tried to run the example projects so that I could navigate the paths in the browser.
I couldn't open the responses in the browser.
I created a relay server. Previously, I passed the server to the transport, and the requests were sent. Now, using the http server, I can't open the paths from Express routes in the browser.
How can I change the example in the message so that the HTML path opens in the browser and web socket connections for the relay are processed?
/* eslint-disable no-console */
import {createServer} from 'node:http'
import {http} from '@libp2p/http'
import {pingHTTP} from '@libp2p/http-ping'
import {canHandle} from '@libp2p/http-server/node'
import {createLibp2p} from 'libp2p'
import {HTTP_TEST_PROTOCOL} from './common.js'
import express from 'express'
import process from "node:process";
import path from "path";
import cors from "cors";
import compression from "compression";
import {privateKeyFromProtobuf, privateKeyToProtobuf} from "@libp2p/crypto/keys";
import fs from "node:fs";
import * as dotenv from "dotenv";
import {identify, identifyPush} from "@libp2p/identify";
import {gossipsub} from "@chainsafe/libp2p-gossipsub";
import {circuitRelayServer, circuitRelayTransport} from "@libp2p/circuit-relay-v2";
import {ping} from "@libp2p/ping";
import {nodeServer} from "@libp2p/http-server";
import {kadDHT, removePrivateAddressesMapper, removePublicAddressesMapper} from "@libp2p/kad-dht";
import {webTransport} from "@libp2p/webtransport";
import {webSockets} from "@libp2p/websockets";
import {tcp} from "@libp2p/tcp";
import {noise} from "@chainsafe/libp2p-noise";
import {yamux} from "@chainsafe/libp2p-yamux";
import {PUBSUB_PEER_DISCOVERY} from "./docs/constants.js";
const app = express()
let __dirname = process.cwd();
const RENDER_EXTERNAL_HOSTNAME = process.env.RENDER_EXTERNAL_HOSTNAME ? process.env.RENDER_EXTERNAL_HOSTNAME: 'localhost'
// Путь для сохранения приватного ключа
const PRIVATE_KEY_PATH = path.join(process.cwd(), 'private-key.proto');
const peerId = await getOrCreatePrivateKey();
console.log('RENDER_EXTERNAL_HOSTNAME', RENDER_EXTERNAL_HOSTNAME)
// an HTTP server that listens on a port and receives incoming requests
const server = createServer()
/**
* Сохраняет приватный ключ на диск
* @param {Uint8Array} privateKey - Приватный ключ в бинарном формате
* @returns {Promise<boolean>} - Успешно ли сохранен ключ
*/
async function savePrivateKey(privateKey) {
try {
// Конвертируем приватный ключ в protobuf формат
const privateKeyProto = privateKeyToProtobuf(privateKey);
// Сохраняем на диск
fs.writeFileSync(PRIVATE_KEY_PATH, privateKeyProto);
console.log('Приватный ключ успешно сохранен:', PRIVATE_KEY_PATH);
return true;
} catch (error) {
console.error('Ошибка при сохранении приватного ключа:', error);
return false;
}
}
/**
* Читает приватный ключ с диска
* @returns {Promise<Uint8Array|null>} - Приватный ключ или null если ошибка
*/
async function readPrivateKey() {
try {
// Проверяем существует ли файл
if (!fs.existsSync(PRIVATE_KEY_PATH)) {
console.log('Файл приватного ключа не найден:', PRIVATE_KEY_PATH);
return null;
}
// Читаем файл
const buffer = fs.readFileSync(PRIVATE_KEY_PATH);
// Конвертируем из protobuf обратно в приватный ключ
const privateKey = privateKeyFromProtobuf(buffer);
console.log('Приватный ключ успешно загружен с диска');
return privateKey;
} catch (error) {
console.error('Ошибка при чтении приватного ключа:', error);
return null;
}
}
/**
* Генерирует и сохраняет новый приватный ключ, если его нет
* @returns {Promise<Uint8Array>} - Существующий или новый приватный ключ
*/
async function getOrCreatePrivateKey() {
// Пытаемся прочитать существующий ключ
let privateKey = await readPrivateKey();
// Если ключа нет, генерируем новый и сохраняем
if (!privateKey) {
console.log('Генерация нового приватного ключа...');
const {generateKeyPair} = await import('@libp2p/crypto/keys');
privateKey = await generateKeyPair('Ed25519');
// Сохраняем новый ключ
await savePrivateKey(privateKey);
}
return privateKey;
}
dotenv.config();
const PORT = process.env.PORT
? process.env.PORT
: 4839;
app.use(compression());
app.use(express.json());
app.use(await cors({credentials: true}));
app.use('/pubsub', express.static(path.join(__dirname, '/docs')));
// app.use('/assets', express.static(path.join(__dirname, '/dist/assets')));
app.use('/assets', express.static(path.join(__dirname, '/public')));
app.get(`/`, async (req, res) => {
const html = `<!DOCTYPE html>
<html lang="ru">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Relay Node</title>
<meta name="description" content="Relay Node Information Dashboard">
<link rel="shortcut icon" href="" type="image/png">
<style>
* {
margin: 0;
padding: 0;
box-sizing: border-box;
}
body {
font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif;
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
min-height: 100vh;
padding: 20px;
color: #333;
}
.container {
max-width: 1200px;
margin: 0 auto;
}
.header {
text-align: center;
background: rgba(255, 255, 255, 0.95);
padding: 30px;
border-radius: 15px;
margin-bottom: 20px;
box-shadow: 0 8px 32px rgba(0, 0, 0, 0.1);
}
.logo {
width: 80px;
height: 80px;
margin-bottom: 15px;
}
.dashboard {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(350px, 1fr));
gap: 20px;
margin-bottom: 20px;
}
.card {
background: rgba(255, 255, 255, 0.95);
padding: 25px;
border-radius: 15px;
box-shadow: 0 8px 32px rgba(0, 0, 0, 0.1);
}
.card h3 {
color: #4a5568;
margin-bottom: 15px;
border-bottom: 2px solid #e2e8f0;
padding-bottom: 8px;
}
.info-grid {
display: grid;
gap: 10px;
}
.info-item {
display: flex;
justify-content: space-between;
align-items: center;
padding: 8px 0;
border-bottom: 1px solid #f1f1f1;
}
.info-label {
font-weight: 600;
color: #4a5568;
display: flex;
width: 7dvw;
}
.info-value {
font-family: 'Courier New', monospace;
background: #f7fafc;
padding: 4px 8px;
border-radius: 4px;
font-size: 0.9em;
word-break: break-all;
// max-width: 60%;
text-align: right;
}
.copy-btn {
background: #4299e1;
color: white;
border: none;
padding: 6px 12px;
border-radius: 4px;
cursor: pointer;
font-size: 0.8em;
margin-left: 8px;
transition: background 0.3s;
height: fit-content;
align-self: center;
}
.copy-btn:hover {
background: #3182ce;
}
.copy-btn.copied {
background: #48bb78;
}
.status-indicator {
display: inline-block;
width: 10px;
height: 10px;
border-radius: 50%;
margin-right: 8px;
}
.status-online {
background: #48bb78;
}
.status-offline {
background: #f56565;
}
.peers-list {
max-height: 300px;
overflow-y: auto;
}
.peer-item {
background: #f7fafc;
padding: 10px;
margin: 5px 0;
border-radius: 6px;
border-left: 4px solid #4299e1;
font-family: 'Courier New', monospace;
font-size: 0.85em;
word-break: break-all;
}
.actions {
display: flex;
gap: 10px;
flex-wrap: wrap;
margin-bottom: 15px;
}
.btn {
background: #4299e1;
color: white;
border: none;
padding: 10px 20px;
border-radius: 6px;
cursor: pointer;
font-size: 0.9em;
transition: all 0.3s;
}
.btn:hover {
background: #3182ce;
transform: translateY(-2px);
}
.btn-secondary {
background: #718096;
}
.btn-secondary:hover {
background: #4a5568;
}
.stats-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
gap: 15px;
margin-top: 15px;
}
.stat-item {
text-align: center;
padding: 15px;
background: #f7fafc;
border-radius: 8px;
}
.stat-value {
font-size: 1.5em;
font-weight: bold;
color: #4299e1;
}
.stat-label {
font-size: 0.8em;
color: #718096;
margin-top: 5px;
}
.refresh-info {
text-align: center;
color: #718096;
font-size: 0.8em;
margin-top: 10px;
}
.full-width {
grid-column: 1 / -1;
margin-bottom: 20px;
}
@media (max-width: 768px) {
.dashboard {
grid-template-columns: 1fr;
}
.stats-grid {
grid-template-columns: repeat(2, 1fr);
}
.info-item {
flex-direction: column;
align-items: flex-start;
gap: 5px;
}
.info-value {
max-width: 100%;
text-align: left;
}
}
.container_peer_id {
display: flex;
width: 100%;
}
</style>
</head>
<body>
<div class="container">
<div class="header">
<img src="./assets/logo.png" alt="Logo" class="logo">
<h1>Relay Node</h1>
<p>Real-time information and monitoring dashboard</p>
</div>
<div class="dashboard">
<div class="card">
<h3>🆔 Node Identity</h3>
<div class="info-grid">
<div class="info-item">
<span class="info-label">Peer ID:</span>
<div class="container_peer_id">
<span class="info-value" id="peerId">${node.peerId.publicKey.toString()}</span>
<button class="copy-btn" onclick="copyToClipboard('peerId')">Copy</button>
</div>
</div>
<div class="info-item">
<span class="info-label">Node Status:</span>
<span class="info-value">
<span class="status-indicator status-online"></span>
<span id="nodeStatus">Online</span>
</span>
</div>
<div class="info-item">
<span class="info-label">Process ID:</span>
<span class="info-value" id="processId">${process.pid}</span>
</div>
<div class="info-item">
<span class="info-label">Port:</span>
<span class="info-value" id="nodePort">${PORT}</span>
</div>
</div>
</div>
<div class="card">
<h3>🌐 Network Addresses</h3>
<div class="info-grid" id="addressesList">
${node.getMultiaddrs().map((addr, index) =>
`<div class="info-item">
<span class="info-label">Address ${index + 1}:</span>
<span class="info-value address-item">${addr.toString()}</span>
</div>`
).join('')}
</div>
</div>
<div class="card">
<h3>📊 Node Statistics</h3>
<div class="stats-grid">
<div class="stat-item">
<div class="stat-value" id="peersCount">${node.getPeers().length}</div>
<div class="stat-label">Connected Peers</div>
</div>
<div class="stat-item">
<div class="stat-value" id="clientsCount">0</div>
<div class="stat-label">SSE Clients</div>
</div>
<div class="stat-item">
<div class="stat-value" id="dhtMode">${node.services.lanDHT?.getMode() || 'Unknown'}</div>
<div class="stat-label">DHT Mode</div>
</div>
<div class="stat-item">
<div class="stat-value" id="uptime">0s</div>
<div class="stat-label">Uptime</div>
</div>
</div>
</div>
<div class="card">
<h3>🔧 Services & Protocols</h3>
<div class="info-grid">
<div class="info-item">
<span class="info-label">DHT (LAN):</span>
<span class="status-indicator status-online"></span>
</div>
<div class="info-item">
<span class="info-label">DHT (Amino):</span>
<span class="status-indicator status-online"></span>
</div>
<div class="info-item">
<span class="info-label">Circuit Relay:</span>
<span class="status-indicator status-online"></span>
</div>
<div class="info-item">
<span class="info-label">PubSub:</span>
<span class="status-indicator status-online"></span>
</div>
<div class="info-item">
<span class="info-label">AutoNAT:</span>
<span class="status-indicator status-online"></span>
</div>
<div class="info-item">
<span class="info-label">Identify:</span>
<span class="status-indicator status-online"></span>
</div>
</div>
</div>
<div class="card">
<h3>🔄 Transports</h3>
<div class="info-grid">
<div class="info-item">
<span class="info-label">WebTransport:</span>
<span class="status-indicator status-online"></span>
</div>
<div class="info-item">
<span class="info-label">WebSockets:</span>
<span class="status-indicator status-online"></span>
</div>
<div class="info-item">
<span class="info-label">TCP:</span>
<span class="status-indicator status-online"></span>
</div>
</div>
</div>
<div class="card">
<h3>📝 Node Information</h3>
<div class="info-grid">
<div class="info-item">
<span class="info-label">Start Time:</span>
<span class="info-value" id="startTime">${new Date().toLocaleString()}</span>
</div>
<div class="info-item">
<span class="info-label">Environment:</span>
<span class="info-value">${process.env.NODE_ENV || 'development'}</span>
</div>
<div class="info-item">
<span class="info-label">Version:</span>
<span class="info-value" id="libp2pVersion">Loading...</span>
</div>
</div>
</div>
</div>
<div class="card full-width">
<h3>👥 Connected Peers</h3>
<div class="actions">
<button class="btn" onclick="refreshPeers()">🔄 Refresh Peers</button>
<button class="btn btn-secondary" onclick="copyAllAddresses()">📋 Copy All Addresses</button>
<button class="btn" onclick="exportNodeInfo()">💾 Export Node Info</button>
</div>
<div class="peers-list" id="peersList">
${node.getPeers().length > 0
? node.getPeers().map(peer =>
`<div class="peer-item">${peer.toString()}</div>`
).join('')
: '<div class="refresh-info">No peers connected</div>'
}
</div>
</div>
<div class="card">
<h3>📡 Bootstrap Address</h3>
<div class="info-item">
<span class="info-label">Primary Address:</span>
<div>
<span class="info-value" id="primaryAddress">${pathNode}</span>
<button class="copy-btn" onclick="copyToClipboard('primaryAddress')">Copy</button>
</div>
</div>
<div class="refresh-info">Use this address to connect other nodes to this relay</div>
</div>
</div>
<script>
let nodeData = {};
let startTime = Date.now();
// Utility functions
function copyToClipboard(elementId) {
const element = document.getElementById(elementId);
const text = element.textContent || element.innerText;
navigator.clipboard.writeText(text).then(() => {
// const btn = event.target;
// const originalText = btn.textContent;
// btn.textContent = '✓ Copied!';
// btn.classList.add('copied');
// setTimeout(() => {
// btn.textContent = originalText;
// btn.classList.remove('copied');
// }, 2000);
}).catch(err => {
console.error('Failed to copy: ', err);
alert('Failed to copy to clipboard');
});
}
function copyAllAddresses() {
const addresses = Array.from(document.querySelectorAll('.address-item'))
.map(item => item.textContent)
.join('\\n');
if (addresses) {
navigator.clipboard.writeText(addresses).then(() => {
showNotification('All addresses copied to clipboard!');
});
}
}
function exportNodeInfo() {
const nodeInfo = {
peerId: document.getElementById('peerId').textContent,
addresses: Array.from(document.querySelectorAll('.address-item')).map(item => item.textContent),
peers: Array.from(document.querySelectorAll('.peer-item')).map(item => item.textContent),
statistics: {
peersCount: document.getElementById('peersCount').textContent,
clientsCount: document.getElementById('clientsCount').textContent,
dhtMode: document.getElementById('dhtMode').textContent,
uptime: document.getElementById('uptime').textContent
},
exportTime: new Date().toISOString()
};
const dataStr = JSON.stringify(nodeInfo, null, 2);
const dataBlob = new Blob([dataStr], {type: 'application/json'});
const url = URL.createObjectURL(dataBlob);
const link = document.createElement('a');
link.href = url;
link.download = \`node-info-\${new Date().toISOString().split('T')[0]}.json\`;
document.body.appendChild(link);
link.click();
document.body.removeChild(link);
URL.revokeObjectURL(url);
showNotification('Node information exported!');
}
function showNotification(message) {
// Create notification element
const notification = document.createElement('div');
notification.style.cssText = \`
position: fixed;
top: 20px;
right: 20px;
background: #48bb78;
color: white;
padding: 12px 20px;
border-radius: 6px;
box-shadow: 0 4px 12px rgba(0,0,0,0.15);
z-index: 1000;
font-size: 0.9em;
\`;
notification.textContent = message;
document.body.appendChild(notification);
setTimeout(() => {
notification.remove();
}, 3000);
}
function formatUptime() {
const seconds = Math.floor((Date.now() - startTime) / 1000);
const days = Math.floor(seconds / 86400);
const hours = Math.floor((seconds % 86400) / 3600);
const minutes = Math.floor((seconds % 3600) / 60);
const secs = seconds % 60;
if (days > 0) return \`\${days}d \${hours}h \${minutes}m\`;
if (hours > 0) return \`\${hours}h \${minutes}m \${secs}s\`;
if (minutes > 0) return \`\${minutes}m \${secs}s\`;
return \`\${secs}s\`;
}
// Data fetching functions
async function refreshPeers() {
try {
const response = await fetch('/peers');
const data = await response.json();
nodeData = data;
updateDashboard();
showNotification('Peers list updated');
} catch (error) {
console.error('Error fetching peers:', error);
showNotification('Error updating peers list');
}
}
async function refreshClients() {
try {
const response = await fetch('/clients');
const clients = await response.json();
document.getElementById('clientsCount').textContent = clients.length;
} catch (error) {
console.error('Error fetching clients:', error);
}
}
function updateDashboard() {
// Update peers count and list
if (nodeData.peers) {
document.getElementById('peersCount').textContent = nodeData.peers.length;
const peersList = document.getElementById('peersList');
if (nodeData.peers.length > 0) {
peersList.innerHTML = nodeData.peers.map(peer =>
\`<div class="peer-item">\${peer}</div>\`
).join('');
} else {
peersList.innerHTML = '<div class="refresh-info">No peers connected</div>';
}
}
// Update DHT mode
if (nodeData.dhtMode) {
document.getElementById('dhtMode').textContent = nodeData.dhtMode;
}
}
// SSE connection for real-time updates
function setupEventSource() {
const events = new EventSource('/events');
events.onmessage = (event) => {
const data = JSON.parse(event.data);
// Peer ID уже установлен на сервере, но обновляем если приходит новый
if (data.peerId && data.peerId !== document.getElementById('peerId').textContent) {
// console.log('ddddddddddddd', data.peerId.publicKey.toString())
// document.getElementById('peerId').textContent = data.peerId.publicKey.toString();
}
};
events.onerror = (err) => {
console.log('SSE connection error:', err);
document.getElementById('nodeStatus').textContent = 'Connection Issues';
document.querySelector('#nodeStatus').previousElementSibling.className = 'status-indicator status-offline';
// Attempt reconnect after 5 seconds
setTimeout(setupEventSource, 5000);
};
}
// Initialize dashboard
document.addEventListener('DOMContentLoaded', function() {
setupEventSource();
refreshClients();
// Set libp2p version (this would need to be passed from server)
document.getElementById('libp2pVersion').textContent = '3.0.6';
// Update uptime every second
setInterval(() => {
document.getElementById('uptime').textContent = formatUptime();
}, 1000);
// Refresh data every 30 seconds
setInterval(() => {
refreshPeers();
refreshClients();
}, 30000);
});
</script>
</body>
</html>`;
res.status(200).send(html);
})
let addresses = process.env.PORT
? {
listen: [
`/ip4/0.0.0.0/tcp/${PORT}/wss`
],
announce: [
`/dns4/${process.env.RENDER_EXTERNAL_HOSTNAME}`,
`/dns4/${process.env.RENDER_EXTERNAL_HOSTNAME}/wss`
]
}
: {
listen: [
`/ip4/127.0.0.1/tcp/${PORT}/ws`
],
announce: [
`/dns4/127.0.0.1/tcp/${PORT}`,
`/dns4/127.0.0.1/tcp/${PORT}/ws`
]
}
// create a libp2p node with a HTTP service that can serve arbitrary HTTP
// protocols and a HTTP ping handler that implements the HTTP ping protocol
const node = await createLibp2p({
privateKey: peerId,
addresses: addresses,
transports: [
circuitRelayTransport(),
webTransport(),
webSockets(),
tcp(),
],
connectionEncryption: [
noise()
],
streamMuxers: [yamux()],
services: {
identify: identify(),
identifyPush: identifyPush(),
relay: circuitRelayServer,
http: http({
server: nodeServer(server)
}),
pingHTTP: pingHTTP(),
pubsub: gossipsub(),
// autoNAT: autoNATv2(),
}
})
node.services.pubsub.subscribe(PUBSUB_PEER_DISCOVERY)
// console.log(`Node started with id ${node.peerId.toString()}`)
let pathNode = ''
node.getMultiaddrs().forEach((ma, index) => {
pathNode = ma.toString()
console.log(`${index}::Listening on:`, pathNode)
})
// register a handler function for the passed protocol - it will be served at
// the protocol id path by default
node.services.http.handle(HTTP_TEST_PROTOCOL, {
handler: (req) => {
return new Response('Hello World!')
}
})
// this handler will return `true` if the request was handled by the libp2p
// node, otherwise it should be passed to an application server like express
// or otherwise handled
const handled = canHandle(node)
server.on('request', (req, res) => {
const isLibp2p = handled(req, res)
if (isLibp2p) {
return
}
app(req, res)
})
// server.listen(PORT, () => {
// console.info('Server listening on:')
// console.info(`http://127.0.0.1:${server.address().port}`)
// })
Browser
http://127.0.0.1:4839/
Only WebSocket connections are supported
Metadata
Metadata
Assignees
Labels
No labels