pythin 远程 windows 服务器
更新: 2025/8/7 字数: 0 字 时长: 0 分钟
Windows 内置的服务「 winrm 」可以满足远程的需求
它是一种基于标准简单对象访问协议( SOAP )的防火墙友好协议,允许来自不同供应商的硬件和操作系统进行互操作
1. 被控端 windows
这里以 Windows 10 系统机器为例
具体操作步骤如下:
启动 winrm 服务
以管理员权限打开 CMD 命令窗口,输入下面命令启动 winrm 服务
bash
# 启动winrm服务
winrm quickconfig -q
如果运行报错,提示网络类型设置异常
可以快捷键 win+i 进入到网络和 Internet,更改网络配置文件,由公用切换为专用
- 检查 winrm 服务监听状态
继续在命令行输入下面命令,查看 winrm 服务的状态
PS:注意这里的端口号 Port 值后面连接会用到
bash
# 查看winrm服务的状态
winrm e winrm/config/listener
# 结果
Listener
Address = *
Transport = HTTP
Port = 5985
Hostname
Enabled = true
URLPrefix = wsman
CertificateThumbprint
ListeningOn = **
- 查看 winrm 配置信息(可选)
通过以下命令可以查看 winrm 全部配置信息、client 客户端配置信息、service 服务端配置信息
bash
# 全部
winrm get winrm/config
# Client
winrm get winrm/config/client
# Service
winrm get winrm/config/service
- 配置 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')