Compare commits
4 Commits
280066436a
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
| f76f0bcdf2 | |||
| 44c492313c | |||
| b4839fde66 | |||
| 67f3f8440b |
@ -2,4 +2,9 @@
|
|||||||
|
|
||||||
roles_path = /var/jenkins_home/ansible-windows/roles
|
roles_path = /var/jenkins_home/ansible-windows/roles
|
||||||
|
|
||||||
ansible_root = /var/jenkins_home/ansible-windows
|
ansible_root = /var/jenkins_home/ansible-windows
|
||||||
|
|
||||||
|
# show verbose time details in log
|
||||||
|
stdout_callback = default
|
||||||
|
callback_whitelist = profile_roles, profile_tasks
|
||||||
|
callbacks_enabled = profile_roles, profile_tasks
|
||||||
|
|||||||
1
inventory/WinRM/CosmosRM.bat
Normal file
1
inventory/WinRM/CosmosRM.bat
Normal file
@ -0,0 +1 @@
|
|||||||
|
powershell -executionpolicy bypass -Command \\home.cosmos\netlogon\cosmosrm.ps1
|
||||||
17
inventory/WinRM/cosmosrm.ps1
Normal file
17
inventory/WinRM/cosmosrm.ps1
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
# script for setting ansible service account to registry key
|
||||||
|
$username = "cosmos-ansible"
|
||||||
|
$ansible_registry = "HKLM:\SOFTWARE\Cosmos\Ansible"
|
||||||
|
$password_key = "Password"
|
||||||
|
$password = (Get-ItemProperty $ansible_registry).$password_key
|
||||||
|
# This is what the thing needs to set the password
|
||||||
|
$securePassword = ConvertTo-SecureString $password -AsPlainText -Force
|
||||||
|
# Set password
|
||||||
|
$UserAccount = Get-LocalUser -Name $username
|
||||||
|
$UserAccount | Set-LocalUser -Password $securePassword
|
||||||
|
|
||||||
|
# Make it a local admin
|
||||||
|
Add-LocalGroupMember -Group "Administrators" -Member $username
|
||||||
|
|
||||||
|
# Various Ansible Settings
|
||||||
|
Set-Item -Path WSMan:\localhost\Service\Auth\Basic -Value $true
|
||||||
|
Enable-WSManCredSSP -Role Server -Force
|
||||||
@ -5,6 +5,9 @@ pipeline {
|
|||||||
parameters {
|
parameters {
|
||||||
string(name: 'host_ip', description: 'Target System Address')
|
string(name: 'host_ip', description: 'Target System Address')
|
||||||
string(name: 'api_service_port', defaultValue: "5000", description: 'API Service Port, probably don\'t change this')
|
string(name: 'api_service_port', defaultValue: "5000", description: 'API Service Port, probably don\'t change this')
|
||||||
|
booleanParam(name: 'refresh_api', defaultValue: false, description: 'When checked this will just update the API')
|
||||||
|
booleanParam(name: 'purge_venv', defaultValue: false, description: 'When checked this will remove the venv folder')
|
||||||
|
|
||||||
// reference for later
|
// reference for later
|
||||||
// choice(name: 'DEPLOY_ENV', choices: ['dev', 'staging', 'prod'], description: 'Environment to deploy to')
|
// choice(name: 'DEPLOY_ENV', choices: ['dev', 'staging', 'prod'], description: 'Environment to deploy to')
|
||||||
// booleanParam(name: 'rename_host', defaultValue: true, description: 'When checked hostname will be renamed')
|
// booleanParam(name: 'rename_host', defaultValue: true, description: 'When checked hostname will be renamed')
|
||||||
@ -54,7 +57,7 @@ pipeline {
|
|||||||
|
|
||||||
ansible-playbook -i \$inventory_file \$playbook_file \
|
ansible-playbook -i \$inventory_file \$playbook_file \
|
||||||
--ssh-common-args='-o StrictHostKeyChecking=no' \
|
--ssh-common-args='-o StrictHostKeyChecking=no' \
|
||||||
--extra-vars "api_service_port=${params.api_service_port}"
|
--extra-vars "api_service_port=${params.api_service_port} refresh_api=${params.refresh_api} purge_venv=${params.purge_venv}"
|
||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -6,9 +6,9 @@
|
|||||||
|
|
||||||
roles:
|
roles:
|
||||||
|
|
||||||
- show_user_vars
|
#- show_user_vars
|
||||||
|
|
||||||
- cosmos_init
|
#- cosmos_init
|
||||||
|
|
||||||
- storage_api
|
- storage_api
|
||||||
|
|
||||||
|
|||||||
@ -7,6 +7,8 @@ windows_base_packages:
|
|||||||
- windirstat
|
- windirstat
|
||||||
- putty
|
- putty
|
||||||
|
|
||||||
|
windows_desktop_packages:
|
||||||
|
|
||||||
windows_features:
|
windows_features:
|
||||||
- NET-Framework-Features
|
- NET-Framework-Features
|
||||||
- Telnet-Client
|
- Telnet-Client
|
||||||
|
|||||||
16
roles/cosmos_init/tasks/desktop.yaml
Normal file
16
roles/cosmos_init/tasks/desktop.yaml
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
---
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
- name: Install base packages
|
||||||
|
when: false
|
||||||
|
win_chocolatey:
|
||||||
|
name:
|
||||||
|
- "{{ windows_desktop_packages_item }}"
|
||||||
|
state: present
|
||||||
|
loop: "{{ windows_desktop_packages }}"
|
||||||
|
loop_control:
|
||||||
|
loop_var: windows_desktop_packages_item
|
||||||
|
|
||||||
|
|
||||||
|
...
|
||||||
@ -1,48 +1,62 @@
|
|||||||
---
|
---
|
||||||
|
- name: skip if refresh
|
||||||
|
when: not refresh_api | bool
|
||||||
- name: Install base packages
|
|
||||||
win_chocolatey:
|
|
||||||
name:
|
|
||||||
- "{{ windows_base_packages_item }}"
|
|
||||||
state: present
|
|
||||||
loop: "{{ windows_base_packages }}"
|
|
||||||
loop_control:
|
|
||||||
loop_var: windows_base_packages_item
|
|
||||||
|
|
||||||
- name: Install Server Services
|
|
||||||
ansible.windows.win_feature:
|
|
||||||
name:
|
|
||||||
- "{{ windows_features_item }}"
|
|
||||||
state: present
|
|
||||||
loop: "{{ windows_features }}"
|
|
||||||
loop_control:
|
|
||||||
loop_var: windows_features_item
|
|
||||||
|
|
||||||
- name: Disable ms_tcpip6 of all the Interface
|
|
||||||
community.windows.win_net_adapter_feature:
|
|
||||||
interface: '*'
|
|
||||||
state: disabled
|
|
||||||
component_id:
|
|
||||||
- ms_tcpip6
|
|
||||||
|
|
||||||
- name: disable IE Enhanced Security
|
|
||||||
block:
|
block:
|
||||||
|
- name: Get Windows version
|
||||||
|
win_shell: "systeminfo /fo csv | ConvertFrom-Csv | select OS*, System*, Hotfix* | Format-List"
|
||||||
|
register: windows_version
|
||||||
|
|
||||||
- name: disable for user
|
- name: Print Windows host information
|
||||||
win_shell: 'Set-ItemProperty -Path "HKLM:\\SOFTWARE\\Microsoft\\Active Setup\\Installed Components\\{A509B1A7-37EF-4b3f-8CFC-4F3A74704073}" -Name "IsInstalled" -Value 0'
|
debug:
|
||||||
|
msg: "{{ windows_version }}"
|
||||||
|
|
||||||
- name: disable for admin
|
- name: Install base packages
|
||||||
win_shell: 'Set-ItemProperty -Path "HKLM:\\SOFTWARE\\Microsoft\\Active Setup\\Installed Components\\{A509B1A8-37EF-4b3f-8CFC-4F3A74704073}" -Name "IsInstalled" -Value 0'
|
win_chocolatey:
|
||||||
|
name:
|
||||||
|
- "{{ windows_base_packages_item }}"
|
||||||
|
state: present
|
||||||
|
ignore_checksums: true
|
||||||
|
loop: "{{ windows_base_packages }}"
|
||||||
|
loop_control:
|
||||||
|
loop_var: windows_base_packages_item
|
||||||
|
|
||||||
- name: disable automatic updates
|
- name: Install Server Services
|
||||||
ansible.windows.win_regedit:
|
when: "'server' in windows_version"
|
||||||
path: HKLM:\Software\Policies\Microsoft\Windows\WindowsUpdate\AU
|
ansible.windows.win_feature:
|
||||||
name: NoAutoUpdate
|
name:
|
||||||
data: 1
|
- "{{ windows_features_item }}"
|
||||||
|
state: present
|
||||||
|
loop: "{{ windows_features }}"
|
||||||
|
loop_control:
|
||||||
|
loop_var: windows_features_item
|
||||||
|
|
||||||
- name: disable Firewall
|
- name: Disable ms_tcpip6 of all the Interfaces
|
||||||
win_shell: "netsh advfirewall set allprofiles state off"
|
community.windows.win_net_adapter_feature:
|
||||||
|
interface: '*'
|
||||||
|
state: disabled
|
||||||
|
component_id:
|
||||||
|
- ms_tcpip6
|
||||||
|
|
||||||
|
- name: windows server items
|
||||||
|
when: "'server' in windows_version"
|
||||||
|
block:
|
||||||
|
|
||||||
|
- name: disable IE-ESC for user
|
||||||
|
win_shell: 'Set-ItemProperty -Path "HKLM:\\SOFTWARE\\Microsoft\\Active Setup\\Installed Components\\{A509B1A7-37EF-4b3f-8CFC-4F3A74704073}" -Name "IsInstalled" -Value 0'
|
||||||
|
|
||||||
|
- name: disable IE-ESC for admin
|
||||||
|
win_shell: 'Set-ItemProperty -Path "HKLM:\\SOFTWARE\\Microsoft\\Active Setup\\Installed Components\\{A509B1A8-37EF-4b3f-8CFC-4F3A74704073}" -Name "IsInstalled" -Value 0'
|
||||||
|
|
||||||
|
- name: disable automatic updates
|
||||||
|
ansible.windows.win_regedit:
|
||||||
|
path: HKLM:\Software\Policies\Microsoft\Windows\WindowsUpdate\AU
|
||||||
|
name: NoAutoUpdate
|
||||||
|
data: 1
|
||||||
|
|
||||||
|
- name: disable Firewall
|
||||||
|
win_shell: "netsh advfirewall set allprofiles state off"
|
||||||
|
|
||||||
|
#- name: desktop items
|
||||||
|
# when: "'server' not in windows_version"
|
||||||
|
# include_tasks: desktop.yaml
|
||||||
...
|
...
|
||||||
@ -10,13 +10,13 @@
|
|||||||
msg: "Hostname: {{ hostname_output.stdout_lines[0] }}"
|
msg: "Hostname: {{ hostname_output.stdout_lines[0] }}"
|
||||||
|
|
||||||
|
|
||||||
- name: Test API
|
#- name: Test API
|
||||||
win_shell: "C:\\Windows\\system32\\curl --silent http://{{ ansible_ssh_host }}:5000/disk"
|
# win_shell: "C:\\Windows\\system32\\curl --silent http://{{ ansible_ssh_host }}:5000/disk"
|
||||||
register: api_test_output
|
# register: api_test_output
|
||||||
|
#
|
||||||
- name: Show Test Results
|
#- name: Show Test Results
|
||||||
debug:
|
# debug:
|
||||||
msg: "{{ api_test_output.stdout_lines }}"
|
# msg: "{{ api_test_output.stdout_lines }}"
|
||||||
|
|
||||||
|
|
||||||
...
|
...
|
||||||
@ -16,6 +16,8 @@ nssm_folder: "{{ cosmos_root_folder }}\\nssm"
|
|||||||
|
|
||||||
disk_service_name: "disk_api"
|
disk_service_name: "disk_api"
|
||||||
|
|
||||||
|
purge_venv: false
|
||||||
|
|
||||||
api_service_port: "5000"
|
api_service_port: "5000"
|
||||||
|
|
||||||
...
|
...
|
||||||
@ -1,2 +1,6 @@
|
|||||||
Flask
|
Flask
|
||||||
psutil
|
flask_apscheduler
|
||||||
|
psutil
|
||||||
|
requests
|
||||||
|
pywin32
|
||||||
|
wmi
|
||||||
@ -5,6 +5,7 @@
|
|||||||
###############################################
|
###############################################
|
||||||
|
|
||||||
- name: set up python venv
|
- name: set up python venv
|
||||||
|
when: not refresh_api | bool
|
||||||
include_tasks: python_venv.yaml
|
include_tasks: python_venv.yaml
|
||||||
|
|
||||||
- name: build python exe
|
- name: build python exe
|
||||||
@ -14,6 +15,7 @@
|
|||||||
include_tasks: nssm.yaml
|
include_tasks: nssm.yaml
|
||||||
|
|
||||||
- name: set up scheduled task
|
- name: set up scheduled task
|
||||||
|
when: false
|
||||||
include_tasks: update_task.yaml
|
include_tasks: update_task.yaml
|
||||||
|
|
||||||
...
|
...
|
||||||
@ -1,19 +1,21 @@
|
|||||||
---
|
---
|
||||||
|
- name: skip when refresh
|
||||||
|
when: not refresh_api | bool
|
||||||
|
block:
|
||||||
|
- name: Copy CrystalDiskInfo archive
|
||||||
|
ansible.windows.win_copy:
|
||||||
|
src: /var/jenkins_home/ansible-files/programs/CrystalDiskInfo.zip
|
||||||
|
dest: "{{ storage_api_root }}\\CrystalDiskInfo.zip"
|
||||||
|
|
||||||
- name: Copy CrystalDiskInfo archive
|
- name: Extract CrystalDiskInfo archive
|
||||||
ansible.windows.win_copy:
|
community.windows.win_unzip:
|
||||||
src: /var/jenkins_home/ansible-files/programs/CrystalDiskInfo.zip
|
src: "{{ storage_api_root }}\\CrystalDiskInfo.zip"
|
||||||
dest: "{{ storage_api_root }}\\CrystalDiskInfo.zip"
|
dest: "{{ storage_api_root }}\\dist\\"
|
||||||
|
|
||||||
- name: Extract CrystalDiskInfo archive
|
- name: Install nssm
|
||||||
community.windows.win_unzip:
|
win_chocolatey:
|
||||||
src: "{{ storage_api_root }}\\CrystalDiskInfo.zip"
|
name: nssm
|
||||||
dest: "{{ storage_api_root }}\\dist\\"
|
state: present
|
||||||
|
|
||||||
- name: Install nssm
|
|
||||||
win_chocolatey:
|
|
||||||
name: nssm
|
|
||||||
state: present
|
|
||||||
|
|
||||||
- name: Install disk_api service
|
- name: Install disk_api service
|
||||||
community.windows.win_nssm:
|
community.windows.win_nssm:
|
||||||
@ -39,7 +41,7 @@
|
|||||||
- "failed...... {{ disk_service_status.failed }}"
|
- "failed...... {{ disk_service_status.failed }}"
|
||||||
|
|
||||||
- name: Test API
|
- name: Test API
|
||||||
win_shell: "C:\\Windows\\system32\\curl http://{{ ansible_ssh_host }}:5000/disk"
|
win_shell: "C:\\Windows\\system32\\curl http://{{ ansible_ssh_host }}:5000/test"
|
||||||
register: api_test_output
|
register: api_test_output
|
||||||
|
|
||||||
- name: Show Test Results
|
- name: Show Test Results
|
||||||
|
|||||||
@ -1,6 +1,7 @@
|
|||||||
---
|
---
|
||||||
|
|
||||||
- name: Create service working folder
|
- name: Create service working folder
|
||||||
|
when: not refresh_api | bool
|
||||||
ansible.windows.win_file:
|
ansible.windows.win_file:
|
||||||
path: "{{ storage_api_root }}"
|
path: "{{ storage_api_root }}"
|
||||||
state: directory
|
state: directory
|
||||||
@ -25,6 +26,7 @@
|
|||||||
dest: "{{ storage_api_root }}\\disk_service.py"
|
dest: "{{ storage_api_root }}\\disk_service.py"
|
||||||
|
|
||||||
- name: install pyinstaller
|
- name: install pyinstaller
|
||||||
|
when: not refresh_api | bool
|
||||||
win_shell: "{{ python_venv_bin }} -m pip install pyinstaller"
|
win_shell: "{{ python_venv_bin }} -m pip install pyinstaller"
|
||||||
|
|
||||||
- name: compile binary
|
- name: compile binary
|
||||||
@ -34,6 +36,7 @@
|
|||||||
chdir: "{{ storage_api_root }}"
|
chdir: "{{ storage_api_root }}"
|
||||||
|
|
||||||
- name: Open up port 5000
|
- name: Open up port 5000
|
||||||
|
when: not refresh_api | bool
|
||||||
community.windows.win_firewall_rule:
|
community.windows.win_firewall_rule:
|
||||||
name: _ansible_python_disk_service
|
name: _ansible_python_disk_service
|
||||||
description: "Firewall rule to allow traffic for Disk info API"
|
description: "Firewall rule to allow traffic for Disk info API"
|
||||||
|
|||||||
@ -5,6 +5,20 @@
|
|||||||
name: python
|
name: python
|
||||||
state: present
|
state: present
|
||||||
|
|
||||||
|
- name: Purge venv if asked
|
||||||
|
when: purge_venv | bool and not refresh_api | bool
|
||||||
|
block:
|
||||||
|
|
||||||
|
- name: Remove service
|
||||||
|
ansible.windows.win_service:
|
||||||
|
name: "{{ disk_service_name }}"
|
||||||
|
state: absent
|
||||||
|
|
||||||
|
- name: purge folder
|
||||||
|
ansible.windows.win_file:
|
||||||
|
path: "{{ python_venv }}"
|
||||||
|
state: absent
|
||||||
|
|
||||||
- name: Create venv folder
|
- name: Create venv folder
|
||||||
ansible.windows.win_file:
|
ansible.windows.win_file:
|
||||||
path: "{{ python_venv }}"
|
path: "{{ python_venv }}"
|
||||||
|
|||||||
@ -1,11 +1,26 @@
|
|||||||
|
# this got big...
|
||||||
from flask import Flask, jsonify
|
from flask import Flask, jsonify
|
||||||
|
from flask_apscheduler import APScheduler
|
||||||
import psutil
|
import psutil
|
||||||
import os
|
import os
|
||||||
|
import requests
|
||||||
|
from requests import RequestException, Response
|
||||||
|
import json
|
||||||
|
from subprocess import check_output
|
||||||
|
import win32api
|
||||||
|
from datetime import datetime, timedelta
|
||||||
|
import sys
|
||||||
|
from pathlib import Path
|
||||||
|
import wmi
|
||||||
|
import pythoncom
|
||||||
|
import ctypes
|
||||||
|
from ctypes import wintypes
|
||||||
|
|
||||||
app = Flask(__name__)
|
app = Flask(__name__)
|
||||||
|
scheduler = APScheduler()
|
||||||
app.config['JSONIFY_PRETTYPRINT_REGULAR'] = True
|
app.config['JSONIFY_PRETTYPRINT_REGULAR'] = True
|
||||||
|
|
||||||
# Bits to Bytes etc
|
# human readable bytes
|
||||||
def bytes_to_human_readable(bytes):
|
def bytes_to_human_readable(bytes):
|
||||||
for unit in ['B', 'KB', 'MB', 'GB', 'TB']:
|
for unit in ['B', 'KB', 'MB', 'GB', 'TB']:
|
||||||
if bytes < 1024.0:
|
if bytes < 1024.0:
|
||||||
@ -24,7 +39,9 @@ def get_crystal_disk_info():
|
|||||||
|
|
||||||
# Split the file content into sections for each drive
|
# Split the file content into sections for each drive
|
||||||
drive_sections = output.split('----------------------------------------------------------------------------')
|
drive_sections = output.split('----------------------------------------------------------------------------')
|
||||||
|
disk_id = 0
|
||||||
for section in drive_sections:
|
for section in drive_sections:
|
||||||
|
|
||||||
lines = section.strip().splitlines()
|
lines = section.strip().splitlines()
|
||||||
data = {
|
data = {
|
||||||
"Hostname": None,
|
"Hostname": None,
|
||||||
@ -38,7 +55,9 @@ def get_crystal_disk_info():
|
|||||||
"Power On Count": None,
|
"Power On Count": None,
|
||||||
"Host Writes": None,
|
"Host Writes": None,
|
||||||
"Wear Level Count": None,
|
"Wear Level Count": None,
|
||||||
"Drive Letter": None
|
"Drive Letter": None,
|
||||||
|
"Interface": None,
|
||||||
|
"Transfer Mode": None
|
||||||
}
|
}
|
||||||
for line in lines:
|
for line in lines:
|
||||||
if "Model" in line:
|
if "Model" in line:
|
||||||
@ -73,11 +92,20 @@ def get_crystal_disk_info():
|
|||||||
data["Drive Letter"] = line.split(":", 1)[1].strip()
|
data["Drive Letter"] = line.split(":", 1)[1].strip()
|
||||||
elif "Disk Size" in line:
|
elif "Disk Size" in line:
|
||||||
if ":" in line:
|
if ":" in line:
|
||||||
data["Disk Size"] = line.split(":", 1)[1].strip()
|
raw = line.split(":", 1)[1].strip()
|
||||||
|
data["Disk Size"] = raw.split('GB')[0].strip() + ' GB'
|
||||||
|
elif "Interface" in line:
|
||||||
|
if ":" in line:
|
||||||
|
data["Interface"] = line.split(":", 1)[1].strip()
|
||||||
|
elif "Transfer Mode" in line:
|
||||||
|
if ":" in line:
|
||||||
|
data["Transfer Mode"] = line.split(":", 1)[1].strip()
|
||||||
|
# This makes sure something was changed,
|
||||||
if any(value is not None for value in data.values()):
|
if any(value is not None for value in data.values()):
|
||||||
|
data["Disk ID"] = disk_id
|
||||||
drives.append(data)
|
drives.append(data)
|
||||||
data["Hostname"] = "{{ hostname_output.stdout_lines[0] }}"
|
disk_id = disk_id + 1
|
||||||
|
#data["Hostname"] = "{{ hostname_output.stdout_lines[0] }}"
|
||||||
|
|
||||||
if not drives:
|
if not drives:
|
||||||
raise ValueError("No drive data found")
|
raise ValueError("No drive data found")
|
||||||
@ -92,10 +120,12 @@ def get_disk_info():
|
|||||||
partitions = psutil.disk_partitions()
|
partitions = psutil.disk_partitions()
|
||||||
for partition in partitions:
|
for partition in partitions:
|
||||||
usage = psutil.disk_usage(partition.mountpoint)
|
usage = psutil.disk_usage(partition.mountpoint)
|
||||||
|
drive_letter = partition.device.replace('\\\\', '\\').rstrip('\\')
|
||||||
disk_info.append({
|
disk_info.append({
|
||||||
'device': partition.device.replace('\\\\', '\\').rstrip('\\'),
|
'device': drive_letter,
|
||||||
|
'label': get_drive_label(drive_letter),
|
||||||
#'mountpoint': partition.mountpoint,
|
#'mountpoint': partition.mountpoint,
|
||||||
#'fstype': partition.fstype,
|
'fstype': partition.fstype,
|
||||||
'total': bytes_to_human_readable(usage.total),
|
'total': bytes_to_human_readable(usage.total),
|
||||||
'used': bytes_to_human_readable(usage.used),
|
'used': bytes_to_human_readable(usage.used),
|
||||||
'free': bytes_to_human_readable(usage.free),
|
'free': bytes_to_human_readable(usage.free),
|
||||||
@ -103,6 +133,101 @@ def get_disk_info():
|
|||||||
})
|
})
|
||||||
return disk_info
|
return disk_info
|
||||||
|
|
||||||
|
# drive label function
|
||||||
|
def get_drive_label(drive_letter: str) -> str:
|
||||||
|
result = "none"
|
||||||
|
root = drive_letter.strip()
|
||||||
|
if not root.endswith(':'):
|
||||||
|
root += ':'
|
||||||
|
if not root.endswith('\\'):
|
||||||
|
root += '\\'
|
||||||
|
|
||||||
|
# Make sure the drive actually exists
|
||||||
|
if not os.path.exists(root):
|
||||||
|
# Not a valid drive letter, return None so the caller can decide what to do.
|
||||||
|
print(f"[DEBUG] Drive '{root}' does not exist.")
|
||||||
|
result = "drive does not exist 0_o"
|
||||||
|
|
||||||
|
# Prepare buffers for the Win32 API call
|
||||||
|
volume_name_buf = ctypes.create_unicode_buffer(260) # MAX_PATH
|
||||||
|
fs_name_buf = ctypes.create_unicode_buffer(260)
|
||||||
|
|
||||||
|
serial_number = wintypes.DWORD()
|
||||||
|
max_component_len = wintypes.DWORD()
|
||||||
|
file_system_flags = wintypes.DWORD()
|
||||||
|
|
||||||
|
# Call GetVolumeInformationW
|
||||||
|
res = ctypes.windll.kernel32.GetVolumeInformationW(
|
||||||
|
ctypes.c_wchar_p(root), # lpRootPathName
|
||||||
|
volume_name_buf, # lpVolumeNameBuffer
|
||||||
|
ctypes.sizeof(volume_name_buf), # nVolumeNameSize
|
||||||
|
ctypes.byref(serial_number), # lpVolumeSerialNumber
|
||||||
|
ctypes.byref(max_component_len), # lpMaximumComponentLength
|
||||||
|
ctypes.byref(file_system_flags), # lpFileSystemFlags
|
||||||
|
fs_name_buf, # lpFileSystemNameBuffer
|
||||||
|
ctypes.sizeof(fs_name_buf) # nFileSystemNameSize
|
||||||
|
)
|
||||||
|
if res == 0: # The call failed
|
||||||
|
err = ctypes.get_last_error()
|
||||||
|
print(f"[ERROR] GetVolumeInformationW failed for '{root}'. "
|
||||||
|
f"Win32 error code: {err}")
|
||||||
|
result = "label error"
|
||||||
|
else:
|
||||||
|
result = volume_name_buf.value
|
||||||
|
if volume_name_buf.value == '':
|
||||||
|
result = "no_label"
|
||||||
|
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
|
# os info function
|
||||||
|
def get_os_info() -> str:
|
||||||
|
result = "windows"
|
||||||
|
pythoncom.CoInitialize()
|
||||||
|
#try:
|
||||||
|
wmi_data = wmi.WMI()
|
||||||
|
os_info = wmi_data.Win32_OperatingSystem()[0]
|
||||||
|
#return {
|
||||||
|
# "Name" : os_info.Name,
|
||||||
|
# "Version" : os_info.Version,
|
||||||
|
# "BuildNumber": os_info.BuildNumber,
|
||||||
|
# "InstallDate": os_info.InstallDate,
|
||||||
|
# "ProductType": int(os_info.ProductType)
|
||||||
|
#}
|
||||||
|
# 1. Major version + edition (e.g. "10 Pro")
|
||||||
|
# os_info.Caption → "Microsoft Windows 10 Pro"
|
||||||
|
parts = os_info.Caption.split()
|
||||||
|
major = parts[2] # "10"
|
||||||
|
edition = parts[3] # "Pro" (for server: "2019", etc.)
|
||||||
|
major_edition = f"{major} {edition}"
|
||||||
|
# 2. Build number
|
||||||
|
build = os_info.BuildNumber
|
||||||
|
# 3. Install date (WMI gives an ISO-8601 string)
|
||||||
|
# e.g. "20210930142300.000000+000"
|
||||||
|
install_ts = os_info.InstallDate[:14] # "20210930142300"
|
||||||
|
dt = datetime.strptime(install_ts, "%Y%m%d%H%M%S")
|
||||||
|
install_date = f"{dt.month}-{dt.day}-{dt.year}"
|
||||||
|
result = f"Windows {major_edition} - Build {build} - Installed {install_date}"
|
||||||
|
#except Exception as e:
|
||||||
|
# print(e)
|
||||||
|
# result = "wmi_error"
|
||||||
|
return result
|
||||||
|
|
||||||
|
# server reporter info
|
||||||
|
def get_server_info() -> dict:
|
||||||
|
result = {}
|
||||||
|
drives_dict = get_crystal_disk_info()
|
||||||
|
data_dict = {
|
||||||
|
"hostname": "{{ hostname_output.stdout_lines[0] }}",
|
||||||
|
"os_string": get_os_info(),
|
||||||
|
"drives": drives_dict["drives"],
|
||||||
|
"API_KEY": "deadbeef",
|
||||||
|
"storage_summary": get_disk_info()
|
||||||
|
|
||||||
|
}
|
||||||
|
result = data_dict
|
||||||
|
return result
|
||||||
|
|
||||||
# Flask endpoints
|
# Flask endpoints
|
||||||
|
|
||||||
@app.route('/disk', methods=['GET'])
|
@app.route('/disk', methods=['GET'])
|
||||||
@ -113,5 +238,69 @@ def disk():
|
|||||||
def drive_health():
|
def drive_health():
|
||||||
return jsonify(get_crystal_disk_info())
|
return jsonify(get_crystal_disk_info())
|
||||||
|
|
||||||
|
@app.route('/full', methods=['GET'])
|
||||||
|
def full_summary():
|
||||||
|
return jsonify(get_server_info())
|
||||||
|
|
||||||
|
@app.route('/test', methods=['GET'])
|
||||||
|
def test_flask():
|
||||||
|
return jsonify({
|
||||||
|
"message": "hello world",
|
||||||
|
"os_string": get_os_info()
|
||||||
|
})
|
||||||
|
|
||||||
|
|
||||||
|
def server_reporter():
|
||||||
|
#base_url="https://cosmostat.matt-cloud.com"
|
||||||
|
base_url="http://10.200.27.20:5001"
|
||||||
|
url = f"{base_url}/storage_client_update"
|
||||||
|
data_dict = get_server_info()
|
||||||
|
result = []
|
||||||
|
try:
|
||||||
|
response = requests.post(url, json=data_dict)
|
||||||
|
|
||||||
|
# Raise an exception for non-2xx status codes
|
||||||
|
response.raise_for_status()
|
||||||
|
result = response.json()
|
||||||
|
except:
|
||||||
|
result = {
|
||||||
|
"message": "error"
|
||||||
|
}
|
||||||
|
|
||||||
|
# Return the JSON payload
|
||||||
|
return result
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
|
|
||||||
|
# disk info Loop Function
|
||||||
|
def update_disk_info():
|
||||||
|
diskinfo_command = f"{{ storage_api_root }}\\dist\\DiskInfo64.exe /CopyExit"
|
||||||
|
result = check_output(diskinfo_command, shell=True)
|
||||||
|
print(result)
|
||||||
|
server_reporter()
|
||||||
|
return result
|
||||||
|
|
||||||
|
# gonna try something wild
|
||||||
|
|
||||||
|
run_date = datetime.now() + timedelta(seconds=15)
|
||||||
|
scheduler.add_job(id='init_disk_info',
|
||||||
|
func=update_disk_info,
|
||||||
|
trigger='date',
|
||||||
|
run_date=run_date)
|
||||||
|
|
||||||
|
scheduler.add_job(id='update_disk_info',
|
||||||
|
func=update_disk_info,
|
||||||
|
trigger='interval',
|
||||||
|
seconds=10000)
|
||||||
|
|
||||||
|
scheduler.add_job(id='server_reporter',
|
||||||
|
func=server_reporter,
|
||||||
|
trigger='interval',
|
||||||
|
seconds=15)
|
||||||
|
|
||||||
|
scheduler.init_app(app)
|
||||||
|
scheduler.start()
|
||||||
|
|
||||||
|
#update_disk_info()
|
||||||
|
|
||||||
app.run(host='0.0.0.0', port={{ api_service_port }})
|
app.run(host='0.0.0.0', port={{ api_service_port }})
|
||||||
|
|||||||
Reference in New Issue
Block a user