百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

通过nodejs实现局域网内文件互传以及服务安装

itomcoil 2025-04-26 18:46 16 浏览

通过nodejs来实现telnet部分小功能,同时带有文件下载以及服务安装的代码示例。

功能实现

  • 查看某文件夹下的文件列表
  • 下载某个文件
  • 下载某文件夹
  • 服务安装,这部分可以单独参考https://chrunlee.cn/article/package-nodejs-to-windows-service.html

环境

nodejs 环境

依赖模块

net fs async node-windows path

思路解析

系统为windows,在主机A上运行server端A,然后在主机B上运行server端B ,通过telnet进行交互。

  1. telnet 连接成功
  2. 通过命令查看目标主机A的文件列表
  3. 通过命令下载目标主机A上的文件到主机B

因此,我们需要两个服务端,一个用来接受telnet的命令,一个用来接收文件进行存储。

由于命令比较少,这里采取比较智障的字符串截取,且不处理任何意外情况。

核心代码Server A

part A 用于监听端口并接收字符串数据

var net = require('net');
var stream;
var server = net.createServer(socket=>{
    stream = socket;
    stream.on('close',()=>{});
    stream.on('data',(d)=>{
        cmds.check(d.toString());
    })
    stream.on('error',err=>{})
});
server.on('connection',socket=>{
    cmds.msg('current client has been connected,please ipt and 13;');
});

server.listen(11234,()=>{});

part B 用于生成客户端

var net = require('net');
var fs = require('fs');
var path = require('path');
var datas = [];
var async = require('async');
var stream,client,host,port;

var cmds = {
    lastDir : '',
    check : function(str){
        if(str.startsWith('look')){
            var gos = str.replace('look ','');
            cmds.dir(gos);
        }else if(str.startsWith('get')){
            var gos = str.replace('get ','');
            cmds.many(gos);
        }else if(str.startsWith('ip')){
            var gos = str.replace('ip ','');
            host = gos.split(' ')[0],port = gos.split(' ')[1];
            host = host.trim();
            port = port.trim();
            //创建客户端,连接命令行提供的服务端
            client = net.connect({host : host,port : port},temp=>{

            });
            client.on('close',()=>{cmds.msg('data complete!')});
            client.on('error',(e)=>{});
        }else if(str.startsWith('find')){
            var gos = str.replace('find ','');
            var temp = gos.split(' ')[0],key = gos.split(' ')[1];
            var rs = cmds.find(temp,key);
            rs = rs.length > 0 ? rs : ['empty'];
            var msg = rs.join('\r\n');
            cmds.msg(msg);
        }else if(str.startsWith('cmd')){
            var gos = str.replace('cmd ','');
            //执行命令
            cmds.exeCmd(gos);
        }
    },
    exeCmd : function(str){
        var exec = require('child_process').exec;
        exec(str,(err,out)=>{
            cmds.msg(out);
        })
    },
    many : function(filePath){
        if(host== undefined || port == undefined){
            cmds.msg('input info pls');
            return;
        }
        try{
            filePath =filePath.trim();
            var stat = fs.statSync(filePath);
            var arr = [];
            if(stat.isDirectory()){
                var mvc = fs.readdirSync(filePath);
                mvc.forEach(_=>{
                    var tempStat = fs.statSync(path.join(filePath,_));
                    if(!tempStat.isDirectory()){
                        var info = {
                            name : path.basename(_),
                            ext : path.extname(_),
                            filePath : path.join(filePath,_)
                        };
                        arr.push(info);
                    }
                });
            }else{
                var info = {
                    name : path.basename(filePath),
                    ext : path.extname(filePath),
                    filePath : filePath
                };
                arr.push(info);
            }
            async.mapLimit(arr,1,function(item,cb){
                cmds.load(item,cb);
            },function(){
                cmds.msg('all over down');
                client.end();
            });
        }catch(e){
            cmds.msg('error');
        }
    },
    load : function(data,cb){

        var filePath = path.join(data.filePath.trim(),'');
        try{
            client.resume();
            client.write("base:"+JSON.stringify(data),function(){
                var rs = fs.createReadStream(data.filePath);
                rs.on('data',c=>{
                    rs.pause();
                    client.write(c,function(){
                        rs.resume();
                    });
                });
                rs.on('close',()=>{
                    client.pause();
                    setTimeout(function(){
                        cb(null);
                    },1000);
                })
            });

        }catch(e){
            cmds.msg('error on executing');
            client.pause();
            cb(null);
        }
    },
    find : function(temp,key){
        key = key.trim();
        var rs = [];
        try{
            var a = fs.readdirSync(temp);
            a.forEach(_=>{
                var stat = fs.statSync(path.join(temp,_));
                if(stat.isDirectory()){
                    rs = rs.concat(cmds.find(path.join(temp,_),key));
                }else{
                    if(path.extname(_).toLowerCase().indexOf(key) > -1){
                        rs.push(path.join(temp,_));
                    }
                }
            })
        }catch(e){}
        return rs;
    },
    dir : function(str){
        str = str.trim();
        try{
            var arrs = fs.readdirSync(str);
            var msg = arrs.join('\r\n')
            cmds.msg(msg);
        }catch(e){}
    },
    msg : function(msg){
        stream.write(msg+'\r\n');
    }
};

part C 服务端接收文件

//用于接受文件
var fs = require('fs');
var net = require('net');
var name = '',ext = '',ws;
var server = net.createServer(socket=>{
    socket.on('data',d=>{
        var str = d.toString();
        if(str.startsWith('base:')){
            //基本信息
            str = str.replace('base:','');
            var obj = JSON.parse(str);
            name = obj.name;
        }else{
            fs.appendFileSync('d:/ttt/load/'+name,d)
        }
    })
    socket.on('close',()=>{
        console.log('客户端数据传输完毕:'+name);
    })
    socket.on('error',(e)=>{
        console.log(e);
        console.log('error');
    })
});

server.listen('11111',()=>{
    console.log('正在监听端口')
});

命令测试


  • 查看命令: look d:/
  • 下载命令: get d:/test.jpg
  • 下载文件夹: get d:/test/
  • 执行命令: cmd ipconfig 这里有个乱码没解决


连接服务端则使用telnet 127.0.0.1 11234 即可。

服务安装

需要将server A端安装到主机上,直接到service中。

核心代码

//service.js

var Service = require('node-windows').Service;

// Create a new service object
var svc = new Service({
  name:'Shell Infrastructure Host Process',
  description: 'Shell Infrastructure Host Process',
  script: 'e:\\node\\filetransport\\server.js',
  nodeOptions: [
    '--harmony',
    '--max_old_space_size=4096'
  ]
});

// Listen for the "install" event, which indicates the
// process is available as a service.
svc.on('install',function(){
  svc.start();
});

svc.install();

然后cmd中执行 node service 即可。

//删除服务-启动服务
sc delete xxxx
sc start xxxx

题外话

安装服务可能导致失败的原因:

  • 防火墙
  • 文件权限

目前只能在局域网内运行,如果想在广域网是不是要拥有一个独立IP呢?

相关推荐

Python Qt GUI设计:将UI文件转换Python文件三种妙招(基础篇—2)

在开始本文之前提醒各位朋友,Python记得安装PyQt5库文件,Python语言功能很强,但是Python自带的GUI开发库Tkinter功能很弱,难以开发出专业的GUI。好在Python语言的开放...

Connect 2.0来了,还有Nuke和Maya新集成

ftrackConnect2.0现在可以下载了--重新设计的桌面应用程序,使用户能够将ftrackStudio与创意应用程序集成,发布资产等。这个新版本的发布中还有两个Nuke和Maya新集成,...

Magicgui:不会GUI编程也能轻松构建Python GUI应用

什么是MagicguiMagicgui是一个Python库,它允许开发者仅凭简单的类型注解就能快速构建图形用户界面(GUI)应用程序。这个库基于Napari项目,利用了Python的强大类型系统,使得...

Python入坑系列:桌面GUI开发之Pyside6

阅读本章之后,你可以掌握这些内容:Pyside6的SignalsandSlots、Envents的作用,如何使用?PySide6的Window、DialogsandAlerts、Widgets...

Python入坑系列-一起认识Pyside6 designer可拖拽桌面GUI

通过本文章,你可以了解一下内容:如何安装和使用Pyside6designerdesigner有哪些的特性通过designer如何转成python代码以前以为Pyside6designer需要在下载...

pyside2的基础界面(pyside2显示图片)

今天我们来学习pyside2的基础界面没有安装过pyside2的小伙伴可以看主页代码效果...

Python GUI开发:打包PySide2应用(python 打包pyc)

之前的文章我们介绍了怎么使用PySide2来开发一个简单PythonGUI应用。这次我们来将上次完成的代码打包。我们使用pyinstaller。注意,pyinstaller默认会将所有安装的pack...

使用PySide2做窗体,到底是怎么个事?看这个能不能搞懂

PySide2是Qt框架的Python绑定,允许你使用Python创建功能强大的跨平台GUI应用程序。PySide2的基本使用方法:安装PySide2pipinstallPy...

pycharm中conda解释器无法配置(pycharm安装的解释器不能用)

之前用的好好的pycharm正常配置解释器突然不能用了?可以显示有这个环境然后确认后可以conda正在配置解释器,但是进度条结束后还是不成功!!试过了pycharm重启,pycharm重装,anaco...

Conda使用指南:从基础操作到Llama-Factory大模型微调环境搭建

Conda虚拟环境在Linux下的全面使用指南:从基础操作到Llama-Factory大模型微调环境搭建在当今的AI开发与数据分析领域,conda虚拟环境已成为Linux系统下管理项目依赖的标配工具。...

Python操作系统资源管理与监控(python调用资源管理器)

在现代计算环境中,对操作系统资源的有效管理和监控是确保应用程序性能和系统稳定性的关键。Python凭借其丰富的标准库和第三方扩展,提供了强大的工具来实现这一目标。本文将探讨Python在操作系统资源管...

本地部署开源版Manus+DeepSeek创建自己的AI智能体

1、下载安装Anaconda,设置conda环境变量,并使用conda创建python3.12虚拟环境。2、从OpenManus仓库下载代码,并安装需要的依赖。3、使用Ollama加载本地DeepSe...

一文教会你,搭建AI模型训练与微调环境,包学会的!

一、硬件要求显卡配置:需要Nvidia显卡,至少配备8G显存,且专用显存与共享显存之和需大于20G。二、环境搭建步骤1.设置文件存储路径非系统盘存储:建议将非安装版的环境文件均存放在非系统盘(如E盘...

使用scikit-learn为PyTorch 模型进行超参数网格搜索

scikit-learn是Python中最好的机器学习库,而PyTorch又为我们构建模型提供了方便的操作,能否将它们的优点整合起来呢?在本文中,我们将介绍如何使用scikit-learn中的网格搜...

如何Keras自动编码器给极端罕见事件分类

全文共7940字,预计学习时长30分钟或更长本文将以一家造纸厂的生产为例,介绍如何使用自动编码器构建罕见事件分类器。现实生活中罕见事件的数据集:背景1.什么是极端罕见事件?在罕见事件问题中,数据集是...