1 Star 0 Fork 1

吕立军/node-hiprint-transit

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
index.js 11.46 KB
一键复制 编辑 原始数据 按行查看 历史
xavier9896 提交于 2024-07-22 19:14 . release 0.0.4
/*
* @Date: 2023-09-28 19:28:42
* @LastEditors: admin@54xavier.cn
* @LastEditTime: 2024-07-22 16:38:53
* @FilePath: /node-hiprint-transit/index.js
*/
import path from "node:path";
import http from "node:http";
import https from "node:https";
import { readFileSync } from "node:fs";
import { fileURLToPath } from "node:url";
import { totalmem, freemem } from "node:os";
import chalk from "chalk";
import { I18n } from "i18n";
import { Server } from "socket.io";
import forge from "node-forge";
import { toUnicode } from "punycode";
import log from "./src/log.js";
import { readConfig, getIPAddress } from "./src/config.js";
import packageJson from "./package.json" assert { type: "json" };
// ES Module need use fileURLToPath to get __dirname
const __filename = fileURLToPath(import.meta.url);
const __dirname = path.dirname(__filename);
// Setup i18n
const i18n = new I18n({
locales: ["en", "zh"],
directory: path.join(__dirname, "./src/locales"),
defaultLocale: "en",
});
const CLIENT = new Map();
// Read config first and then start serve
readConfig().then((CONFIG) => {
const { port, token, useSSL, lang } = CONFIG;
var ipAddress = `http://${getIPAddress()}:${port}`;
i18n.setLocale(lang);
var server;
if (useSSL) {
const key = readFileSync("./src/ssl.key", "utf-8");
const cert = readFileSync("./src/ssl.pem", "utf-8");
// Check SSL certificate
if (!key || !cert) {
console.error(chalk.red(i18n.__("SSL certificate is missing")));
process.exit(1);
}
const certificate = forge.pki.certificateFromPem(cert);
// Check SSL certificate is expired
if (new Date(certificate.validity.notAfter) < new Date()) {
console.warn(chalk.red(i18n.__("SSL certificate has expired")));
}
server = https.createServer({
key,
cert,
});
// Get all domains from certificate
const domains = certificate.extensions
.find(({ name }) => name === "subjectAltName")
.altNames.map(({ value }) => toUnicode(value));
ipAddress = domains.map((value) => `https://${value}:${port}`).join("\n");
} else {
server = http.createServer();
}
// Setup socket.io
const io = new Server(server, {
cors: {
origin: "*",
methods: ["GET", "POST"],
},
});
server.listen(port, () => {
log(i18n.__("Serve is start"));
console.log(
chalk.green(`node-hiprint-transit version: ${packageJson.version}\n`)
);
console.log(
i18n.__(
"Serve is running on\n%s\n\nPlease make sure that the ports have been opened in the security group or firewall.\ntoken: %s",
chalk.green.underline(ipAddress),
chalk.green(token)
)
);
});
// Authentication
io.use((socket, next) => {
const tokenRegex = new RegExp(`^${token.replace(/\*/g, "\\S+")}$`);
if (token && tokenRegex.test(socket.handshake.auth.token)) {
next();
} else {
log(i18n.__("Authentication failed for %s", socket.id));
next(new Error("Authentication failed"));
}
});
// Socket.io Add event listener
io.on("connection", (socket) => {
const sToken = socket.handshake.auth.token;
if (!CLIENT.has(sToken)) {
CLIENT.set(sToken, {});
}
// Send server info to client
const serverInfo = {
// Server version
version: packageJson.version,
// Number of Electron-hiprint clients for the current socket's token
currentClients: Object.keys(CLIENT.get(sToken)).length,
// Number of all Electron-hiprint clients
allClients: Array.from(io.sockets.sockets.values()).filter(
({ handshake }) => handshake.query?.client === "electron-hiprint"
)?.length,
// Number of web clients for the current socket's token
webClients: Array.from(io.sockets.sockets.values()).filter(
({ handshake }) =>
handshake.auth.token === sToken &&
handshake.query?.client !== "electron-hiprint"
)?.length,
// Number of all web clients
allWebClients: Array.from(io.sockets.sockets.values()).filter(
({ handshake }) => handshake.query?.client === "electron-hiprint"
)?.length,
// Server total memory
totalmem: totalmem(),
// Server free memory
freemem: freemem(),
};
socket.emit("serverInfo", serverInfo);
if (socket.handshake.query.test !== "true") {
if (socket.handshake.query.client === "electron-hiprint") {
log(
i18n.__(
"Client connected: %s",
`${socket.id} | ${sToken} | (electron-hiprint)`
)
);
// Join electron-hiprint room
socket.join(`${sToken}_electron-hiprint`);
} else {
log(
i18n.__(
"Client connected: %s",
`${socket.id} | ${sToken} | (web-client)`
)
);
// Join web-client room
socket.join(`${token}_web-client`);
// Send client list to web client
socket.emit("clients", CLIENT.get(sToken));
// Send all printer list to web client
const allPrinterList = [];
const clients = CLIENT.get(sToken);
Object.keys(clients).forEach((key) => {
const client = clients[key];
client.printerList.forEach((printer) => {
allPrinterList.push({
...printer,
server: Object.assign({}, client, {
clientId: key,
printerList: undefined,
}),
});
});
});
socket.emit("printerList", allPrinterList);
}
} else {
log(i18n.__("Client connected: %s", `${socket.id} | ${sToken} | (test)`));
// Wait for the serverInfo event to be emitted, then disconnect.
setTimeout(() => {
socket.disconnect(true);
}, 1000 * 5);
}
// Get client info
socket.on("clientInfo", (data) => {
CLIENT.get(sToken)[socket.id] = Object.assign(
{
clientId: socket.id,
},
CLIENT.get(sToken)[socket.id],
data,
);
io.to(`${sToken}_web-client`).emit("clients", CLIENT.get(sToken));
});
// Get client printer list
socket.on("printerList", (printerList) => {
CLIENT.get(sToken)[socket.id] = Object.assign(
{},
CLIENT.get(sToken)[socket.id],
{ printerList }
);
});
// Get all client list
socket.on("getClients", () => {
socket.emit("clients", CLIENT.get(sToken));
});
// Get all clients printer list
socket.on("refreshPrinterList", () => {
io.to(`${sToken}_electron-hiprint`).emit("refreshPrinterList");
// Just wait 2 seconds for the client to update the printer list
// Of course, this is not a good way to do it. But it’s not like it can’t be used 🤪
setTimeout(() => {
const allPrinterList = [];
const clients = CLIENT.get(sToken);
Object.keys(clients).forEach((key) => {
const client = clients[key];
client.printerList.forEach((printer) => {
allPrinterList.push({
...printer,
server: Object.assign({}, client, {
clientId: key,
printerList: undefined,
}),
});
});
});
socket.emit("printerList", allPrinterList);
}, 1000 * 2);
});
// Get client address info, is not supported
socket.on("address", () => {
socket.emit(
"address",
"Address is not supported in transit server, you should use getClients."
);
});
// Make a ipp print to electron-hiprint client
socket.on("ippPrint", (options) => {
if (options.client) {
if (!CLIENT.get(sToken)[options.client]) {
socket.emit("error", {
msg: "Client is not exist.",
});
return;
}
socket
.to(options.client)
.emit("ippPrint", { ...options, replyId: socket.id });
log(i18n.__("%s send ippPrint to %s", socket.id, options.client));
} else {
socket.emit("error", {
msg: "Client must be specified.",
});
}
});
// Make a ipp printer connected event to reply client
socket.on("ippPrinterConnected", (options) => {
if (options.replyId && options.printer) {
socket.to(options.replyId).emit("ippPrinterConnected", options.printer);
}
});
// Make a ipp printer callback to reply client
socket.on("ippPrinterCallback", (options, res) => {
if (options.replyId) {
socket.to(options.replyId).emit("ippPrinterCallback", options, res);
}
});
// Make a ipp request to electron-hiprint client
socket.on("ippRequest", (options) => {
if (options.client) {
if (!CLIENT.get(sToken)[options.client]) {
socket.emit("error", {
msg: "Client is not exist.",
});
return;
}
socket
.to(options.client)
.emit("ippRequest", { ...options, replyId: socket.id });
log(i18n.__("%s send ippRequest to %s", socket.id, options.client));
} else {
socket.emit("error", {
msg: "Client must be specified.",
});
}
});
// Make a ipp request callback to reply client
socket.on("ippRequestCallback", (options, res) => {
if (options.replyId) {
socket.to(options.replyId).emit("ippRequestCallback", options, res);
}
});
// Make a news to electron-hiprint client
socket.on("news", (options) => {
if (options.client) {
if (!CLIENT.get(sToken)[options.client]) {
socket.emit("error", {
msg: "Client is not exist.",
templateId: options.templateId,
});
return;
}
socket
.to(options.client)
.emit("news", { ...options, replyId: socket.id });
log(i18n.__("%s send news to %s", socket.id, options.client));
} else {
socket.emit("error", {
msg: "Client must be specified.",
templateId: options.templateId,
});
}
});
// Make a success callback to reply client
socket.on("success", (options) => {
if (options.replyId) {
socket.to(options.replyId).emit("success", options);
log(
i18n.__(
"%s client: print success, templateId: %s",
socket.id,
options.templateId
)
);
}
});
// Make a error callback to reply client
socket.on("error", (options) => {
if (options.replyId) {
socket.to(options.replyId).emit("error", options);
log(
i18n.__(
"%s client: print error, templateId: %s",
socket.id,
options.templateId
)
);
}
});
// Client disconnect
socket.on("disconnect", () => {
if (socket.handshake.query.test !== "true") {
log(i18n.__("Client disconnected: %s", socket.id));
// Remove electron-hiprint client from CLIENT
if (socket.handshake.query.client === "electron-hiprint") {
delete CLIENT.get(sToken)[socket.id];
// Send client list to web client
io.to("web-client").emit("clients", CLIENT.get(sToken));
}
}
});
});
// Retrieve the client print list every 10 minutes.
setInterval(() => {
log(i18n.__("Retrieve the client print list"));
CLIENT.forEach((_, key) => {
io.to(`${key}_electron-hiprint`).emit("refreshPrinterList");
});
}, 1000 * 60 * 10);
});
// Close serve
process.on("SIGINT", () => {
log(i18n.__("Serve is closed")).then(() => {
process.exit(0); // 退出进程
});
});
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
JavaScript
1
https://gitee.com/f749893/node-hiprint-transit.git
git@gitee.com:f749893/node-hiprint-transit.git
f749893
node-hiprint-transit
node-hiprint-transit
main

搜索帮助