Skip to content
有问题可以在公众号反馈 ✨
微信二维码

扫码留言

pythin 远程 windows 服务器

更新: 2025/8/7 字数: 0 字 时长: 0 分钟

Windows 内置的服务「 winrm 」可以满足远程的需求

它是一种基于标准简单对象访问协议( SOAP )的防火墙友好协议,允许来自不同供应商的硬件和操作系统进行互操作

1. 被控端 windows

这里以 Windows 10 系统机器为例

具体操作步骤如下:

  1. 启动 winrm 服务

    以管理员权限打开 CMD 命令窗口,输入下面命令启动 winrm 服务

bash
# 启动winrm服务
winrm quickconfig -q

如果运行报错,提示网络类型设置异常

alt text

可以快捷键 win+i 进入到网络和 Internet,更改网络配置文件,由公用切换为专用

alt text

  1. 检查 winrm 服务监听状态

继续在命令行输入下面命令,查看 winrm 服务的状态

PS:注意这里的端口号 Port 值后面连接会用到

bash
# 查看winrm服务的状态
winrm e winrm/config/listener

# 结果
Listener
    Address = *
    Transport = HTTP
    Port = 5985
    Hostname
    Enabled = true
    URLPrefix = wsman
    CertificateThumbprint
    ListeningOn = **
  1. 查看 winrm 配置信息(可选)

通过以下命令可以查看 winrm 全部配置信息、client 客户端配置信息、service 服务端配置信息

bash
# 全部
winrm get winrm/config

# Client
winrm get winrm/config/client

# Service
winrm get winrm/config/service
  1. 配置 winrm client
bash
# 配置winrm client
winrm set winrm/config/client @{AllowUnencrypted="true"}

winrm set winrm/config/client @{TrustedHosts="*"}

winrm set winrm/config/client/auth @{Basic="true"}

1-5   配置 winrm service

在配置完 winrm service 和 winrm client 后,我们通过通过步骤 1-3 查看配置文件,确保配置文件已生效

bash
# 配置winrm service
winrm set winrm/config/service @{AllowUnencrypted="true"}

winrm set winrm/config/service/auth @{Basic="true"}

2. 控制端

在控制端,比如:Mac OSX、Linux,我们只需要安装「 pywinrm 」依赖包即可

# 控制端安装依赖包
pip3 install pywinrm

3. 简单测试案例

接下来用 ip 地址、端口号、用户名、密码连接 Windows 被控端

python
# 连接windows
import winrm

...
# ip地址:端口号
# winrm server端口号
# auth:用户名和密码
self.session = winrm.Session("192.168.**.**:5985", auth=('username', 'password'), transport='ntlm')
...

4. 实际案例

python
import winrm

# 目标服务器的 IP 和端口(通常使用 5985 或 5986 端口)
session = winrm.Session('http://**.**.**.**:5985/wsman', auth=('账号', '密码'), server_cert_validation='ignore')

# 执行命令,获取磁盘使用情况
result = session.run_ps('Get-PSDrive -PSProvider FileSystem')

# 尝试使用 utf-8 解码,如果失败则尝试 gbk 或其他编码
try:
    output = result.std_out.decode('utf-8')  # 尝试以 UTF-8 解码
except UnicodeDecodeError:
    try:
        output = result.std_out.decode('gbk')  # 如果 utf-8 解码失败,尝试 gbk
    except UnicodeDecodeError:
        output = result.std_out.decode('latin1')  # 如果 gbk 也失败,尝试 latin1

# 输出原始内容
print(output)
python
import winrm

session = winrm.Session(
    'http://****.****.****.****:5985/wsman',
    auth=('账号', '密码'),
    server_cert_validation='ignore'
)

# 获取 IIS 网站列表
result = session.run_ps('Import-Module WebAdministration; Get-Website | Select-Object Name, State, PhysicalPath, Bindings')
# 停止指定的 IIS 网站
session.run_ps('Import-Module WebAdministration; Stop-Website -Name "***.****.cn"')
# 启动指定的 IIS 网站
session.run_ps('Import-Module WebAdministration; Start-Website -Name "****.****.cn"')

# 尝试解码输出
try:
    output = result.std_out.decode('utf-8')
except UnicodeDecodeError:
    try:
        output = result.std_out.decode('gbk')
    except UnicodeDecodeError:
        output = result.std_out.decode('latin1')

print(output)
python
import winrm

# 初始化 WinRM 会话
session = winrm.Session(
    'http://****.****.****.****:5985/wsman',
    auth=('账号', '密码'),
    server_cert_validation='ignore'
)

# 解码输出,解决中文乱码问题
def decode_output(output_bytes):
    for encoding in ('utf-8', 'gbk', 'gb2312', 'big5', 'latin1'):
        try:
            return output_bytes.decode(encoding)
        except UnicodeDecodeError:
            continue
    return output_bytes.decode('utf-8', errors='ignore')

# 包装执行函数,自动加关闭进度条代码
def run_and_print(title, ps_script):
    # 屏蔽 PowerShell 进度条,避免 CLIXML 垃圾信息
    full_script = "$ProgressPreference = 'SilentlyContinue'\n" + ps_script
    result = session.run_ps(full_script)

    output = decode_output(result.std_out)
    error = decode_output(result.std_err)

    print(f"\n{title}】")
    if output.strip():
        print("输出:")
        print(output.strip())
    if error.strip():
        print("错误:")
        print(error.strip())

    return output

# 1. 查询 es某个索引 总数
run_and_print("查询 es某个索引 总数", '(Invoke-WebRequest -Uri "http://****.****.****.****:9200/es某个索引/_count" -UseBasicParsing).Content')

# 2. 删除 es某个索引 索引
run_and_print("删除 es某个索引 索引", 'Invoke-RestMethod -Uri "http://****.****.****.****:9200/es某个索引" -Method Delete')

# 3. 调用初始化接口(请根据实际替换 where 参数)
wheresql = ""
init_url = f"http://****.******.****.****" #可以直接在远程服务器执行接口调用,本接口是用来同步es
init_script = f'''
$ProgressPreference = 'SilentlyContinue'
$response = Invoke-WebRequest -Uri "{init_url}" -UseBasicParsing
$bytes = [System.Text.Encoding]::UTF8.GetBytes($response.Content)
[System.Text.Encoding]::UTF8.GetString($bytes)
'''

run_and_print("调用 initESLog 初始化接口", init_script)

# 4. 设置 es某个索引 索引分页限制(PUT 操作)
put_script = '''
$body = @{
    "index" = @{
        "max_result_window" = 999999999
    }
} | ConvertTo-Json -Depth 10

Invoke-RestMethod -Uri "http://****.****.****.*:9200/es某个索引/_settings" -Method Put -Body $body -ContentType "application/json"
'''
run_and_print("设置 es某个索引 分页限制", put_script)

run_and_print("查询 es某个索引 总数", '(Invoke-WebRequest -Uri "http://****.****.****.*/es某个索引/_count" -UseBasicParsing).Content')