mirror of
https://github.com/keylase/nvidia-patch.git
synced 2024-08-30 18:32:50 +00:00
adds nv-update-locator utility and support utils/libs
This commit is contained in:
parent
5adcc24a94
commit
75a69c3491
107
win/tools/nv-driver-locator/.gitignore
vendored
Normal file
107
win/tools/nv-driver-locator/.gitignore
vendored
Normal file
@ -0,0 +1,107 @@
|
|||||||
|
# Do not stage vim swapfiles to commit
|
||||||
|
*.swp
|
||||||
|
|
||||||
|
# Byte-compiled / optimized / DLL files
|
||||||
|
__pycache__/
|
||||||
|
*.py[cod]
|
||||||
|
*$py.class
|
||||||
|
|
||||||
|
# C extensions
|
||||||
|
*.so
|
||||||
|
|
||||||
|
# Distribution / packaging
|
||||||
|
.Python
|
||||||
|
build/
|
||||||
|
develop-eggs/
|
||||||
|
dist/
|
||||||
|
downloads/
|
||||||
|
eggs/
|
||||||
|
.eggs/
|
||||||
|
lib/
|
||||||
|
lib64/
|
||||||
|
parts/
|
||||||
|
sdist/
|
||||||
|
var/
|
||||||
|
wheels/
|
||||||
|
*.egg-info/
|
||||||
|
.installed.cfg
|
||||||
|
*.egg
|
||||||
|
MANIFEST
|
||||||
|
|
||||||
|
# PyInstaller
|
||||||
|
# Usually these files are written by a python script from a template
|
||||||
|
# before PyInstaller builds the exe, so as to inject date/other infos into it.
|
||||||
|
*.manifest
|
||||||
|
*.spec
|
||||||
|
|
||||||
|
# Installer logs
|
||||||
|
pip-log.txt
|
||||||
|
pip-delete-this-directory.txt
|
||||||
|
|
||||||
|
# Unit test / coverage reports
|
||||||
|
htmlcov/
|
||||||
|
.tox/
|
||||||
|
.coverage
|
||||||
|
.coverage.*
|
||||||
|
.cache
|
||||||
|
nosetests.xml
|
||||||
|
coverage.xml
|
||||||
|
*.cover
|
||||||
|
.hypothesis/
|
||||||
|
.pytest_cache/
|
||||||
|
|
||||||
|
# Translations
|
||||||
|
*.mo
|
||||||
|
*.pot
|
||||||
|
|
||||||
|
# Django stuff:
|
||||||
|
*.log
|
||||||
|
local_settings.py
|
||||||
|
db.sqlite3
|
||||||
|
|
||||||
|
# Flask stuff:
|
||||||
|
instance/
|
||||||
|
.webassets-cache
|
||||||
|
|
||||||
|
# Scrapy stuff:
|
||||||
|
.scrapy
|
||||||
|
|
||||||
|
# Sphinx documentation
|
||||||
|
docs/_build/
|
||||||
|
|
||||||
|
# PyBuilder
|
||||||
|
target/
|
||||||
|
|
||||||
|
# Jupyter Notebook
|
||||||
|
.ipynb_checkpoints
|
||||||
|
|
||||||
|
# pyenv
|
||||||
|
.python-version
|
||||||
|
|
||||||
|
# celery beat schedule file
|
||||||
|
celerybeat-schedule
|
||||||
|
|
||||||
|
# SageMath parsed files
|
||||||
|
*.sage.py
|
||||||
|
|
||||||
|
# Environments
|
||||||
|
.env
|
||||||
|
.venv
|
||||||
|
env/
|
||||||
|
venv/
|
||||||
|
ENV/
|
||||||
|
env.bak/
|
||||||
|
venv.bak/
|
||||||
|
|
||||||
|
# Spyder project settings
|
||||||
|
.spyderproject
|
||||||
|
.spyproject
|
||||||
|
|
||||||
|
# Rope project settings
|
||||||
|
.ropeproject
|
||||||
|
|
||||||
|
# mkdocs documentation
|
||||||
|
/site
|
||||||
|
|
||||||
|
# mypy
|
||||||
|
.mypy_cache/
|
171
win/tools/nv-driver-locator/README.md
Normal file
171
win/tools/nv-driver-locator/README.md
Normal file
@ -0,0 +1,171 @@
|
|||||||
|
nv-driver-locator
|
||||||
|
=================
|
||||||
|
|
||||||
|
nv-driver-locator is a tool for internal usage, which purpose is to notify about new Nvidia driver releases. It's kernel supports and performs:
|
||||||
|
|
||||||
|
* Update retrieval from multiple sources (**channels** component).
|
||||||
|
* Notification through various ways (**notifiers** component).
|
||||||
|
* Driver info matching and aggregation via configurable set of attributes (**hasher** component).
|
||||||
|
* Persistence of collected data for keeping track on already seen drivers (**db** component).
|
||||||
|
|
||||||
|
## Requirements
|
||||||
|
|
||||||
|
* Python 3.4+
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
### Structure
|
||||||
|
|
||||||
|
All scripts may be used both as standalone application and importable module. For CLI synopsys invoke program with `--help` option.
|
||||||
|
|
||||||
|
* nv-driver-locator.py - main executable, intended to be run as cron job.
|
||||||
|
* mailer.py - module with email routines and minimalistic email client for test purposes.
|
||||||
|
* gfe\_get\_driver.py - GeForce Experience client library (and test util).
|
||||||
|
|
||||||
|
### Operation
|
||||||
|
|
||||||
|
1. Cron job queries all configured channels.
|
||||||
|
2. Program aggregates responses by hashing their's values covered by `key_components`. `key_components` is a list of JSON paths (represented by list too) specified in config file.
|
||||||
|
3. Program queries DB if given hash has any match in database.
|
||||||
|
4. If no match found and we have new instance all notifiers getting fired.
|
||||||
|
5. New record gets written into DB.
|
||||||
|
|
||||||
|
## Configuration example
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"db": {
|
||||||
|
"type": "file",
|
||||||
|
"params": {
|
||||||
|
"workdir": "/var/lib/nv-driver-locator"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"key_components": [
|
||||||
|
[
|
||||||
|
"DriverAttributes",
|
||||||
|
"Version"
|
||||||
|
]
|
||||||
|
],
|
||||||
|
"channels": [
|
||||||
|
{
|
||||||
|
"type": "gfe_client",
|
||||||
|
"name": "desktop defaults",
|
||||||
|
"params": {}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "gfe_client",
|
||||||
|
"name": "desktop beta",
|
||||||
|
"params": {
|
||||||
|
"beta": true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "gfe_client",
|
||||||
|
"name": "mobile",
|
||||||
|
"params": {
|
||||||
|
"notebook": true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "gfe_client",
|
||||||
|
"name": "mobile beta",
|
||||||
|
"params": {
|
||||||
|
"notebook": true,
|
||||||
|
"beta": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"notifiers": [
|
||||||
|
{
|
||||||
|
"type": "email",
|
||||||
|
"name": "my email",
|
||||||
|
"params": {
|
||||||
|
"from_addr": "notify-bot@gmail.com",
|
||||||
|
"to_addrs": [
|
||||||
|
"recepient1@domain1.tld",
|
||||||
|
"recepient2@domain2.tld"
|
||||||
|
],
|
||||||
|
"host": "smtp.google.com",
|
||||||
|
"use_starttls": true,
|
||||||
|
"login": "notify-bot",
|
||||||
|
"password": "MyGoodPass"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"name": "sample command",
|
||||||
|
"params": {
|
||||||
|
"timeout": 10.0,
|
||||||
|
"cmdline": [
|
||||||
|
"cat",
|
||||||
|
"-"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Components Reference
|
||||||
|
|
||||||
|
### DB
|
||||||
|
|
||||||
|
#### FileDB
|
||||||
|
|
||||||
|
Stores data in files.
|
||||||
|
|
||||||
|
Type: `file`
|
||||||
|
|
||||||
|
Params:
|
||||||
|
|
||||||
|
* `workdir` - files location
|
||||||
|
|
||||||
|
### Channels
|
||||||
|
|
||||||
|
#### GFEClientChannel
|
||||||
|
|
||||||
|
Queries latest driver for Windows, using GeForce Experience API.
|
||||||
|
|
||||||
|
Type: `gfe_client`
|
||||||
|
|
||||||
|
Params:
|
||||||
|
|
||||||
|
* `notebook` - seek for Mobile driver. Default: `false`
|
||||||
|
* `x86_64` - seek for 64bit driver. Default: `true`
|
||||||
|
* `os_version` - OS version. Default: `"10.0"`
|
||||||
|
* `os_build` - OS build. Default: `"17763"`
|
||||||
|
* `language` - language. Default: `1033` (English)
|
||||||
|
* `beta` - request Beta driver. Default: `false`
|
||||||
|
* `dch` - request DCH driver. Default: `false` (request Standard Driver)
|
||||||
|
|
||||||
|
### Notifiers
|
||||||
|
|
||||||
|
#### CommandNotifier
|
||||||
|
|
||||||
|
Runs external process and pipes JSON with info about new driver into it
|
||||||
|
|
||||||
|
Type: `command`
|
||||||
|
|
||||||
|
Params:
|
||||||
|
|
||||||
|
* `cmdline` - list of command line arguments (where first is executable name)
|
||||||
|
* `timeout` - allowed execution time in seconds. Default: `10.0`
|
||||||
|
|
||||||
|
#### EmailNotifier
|
||||||
|
|
||||||
|
Sends email with attached JSON file with driver info. Supports TLS, STARTTLS and authentication, so it can be used to send notification via mailbox provided by public services like gmail.
|
||||||
|
|
||||||
|
Type: `email`
|
||||||
|
|
||||||
|
Params:
|
||||||
|
|
||||||
|
* `from_addr` - originating address
|
||||||
|
* `to_addrs` - list of destination addresses
|
||||||
|
* `host` - SMTP host. Default: `localhost`
|
||||||
|
* `port` - SMTP port. Default: depends on chosen TLS/STARTTLS mode.
|
||||||
|
* `local_hostname` - hostname used in EHLO/HELO commands. Default: auto
|
||||||
|
* `use_ssl` - use SSL from beginning of connection. Default: `false`
|
||||||
|
* `use_starttls` - use STARTTLS. Default: `false`
|
||||||
|
* `login` - user login name. Default: `null` (do not use authentication)
|
||||||
|
* `password` - user password. Default: `null`
|
||||||
|
* `timeout` - allowed delay in seconds for each network operation. Default: `10.0`
|
141
win/tools/nv-driver-locator/gfe_get_driver.py
Executable file
141
win/tools/nv-driver-locator/gfe_get_driver.py
Executable file
@ -0,0 +1,141 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import urllib.request
|
||||||
|
import urllib.error
|
||||||
|
import json
|
||||||
|
import posixpath
|
||||||
|
import codecs
|
||||||
|
|
||||||
|
USER_AGENT = 'NvBackend/34.0.0.0'
|
||||||
|
TIMEOUT = 10
|
||||||
|
|
||||||
|
|
||||||
|
def serialize_req(obj):
|
||||||
|
return json.dumps(obj, separators=(',', ':'))
|
||||||
|
|
||||||
|
|
||||||
|
def getDispDrvrByDevid(query_obj):
|
||||||
|
ENDPOINT = 'https://gfwsl.geforce.com/nvidia_web_services/' \
|
||||||
|
'controller.gfeclientcontent.NG.php/' \
|
||||||
|
'com.nvidia.services.GFEClientContent_NG.getDispDrvrByDevid'
|
||||||
|
url = posixpath.join(ENDPOINT, serialize_req(query_obj))
|
||||||
|
http_req = urllib.request.Request(
|
||||||
|
url,
|
||||||
|
data=None,
|
||||||
|
headers={
|
||||||
|
'User-Agent': USER_AGENT
|
||||||
|
}
|
||||||
|
)
|
||||||
|
with urllib.request.urlopen(http_req, None, TIMEOUT) as resp:
|
||||||
|
coding = resp.headers.get_content_charset()
|
||||||
|
coding = coding if coding is not None else 'utf-8-sig'
|
||||||
|
decoder = codecs.getreader(coding)(resp)
|
||||||
|
res = json.load(decoder)
|
||||||
|
return res
|
||||||
|
|
||||||
|
|
||||||
|
def get_latest_geforce_driver(*,
|
||||||
|
notebook=False,
|
||||||
|
x86_64=True,
|
||||||
|
os_version="10.0",
|
||||||
|
os_build="17763",
|
||||||
|
language=1033,
|
||||||
|
beta=False,
|
||||||
|
dch=False):
|
||||||
|
# GeForce GTX 1080 and GP104 HD Audio
|
||||||
|
dt_id = ["1B80_10DE_119E_10DE"]
|
||||||
|
# GeForce GTX 1080 Mobile
|
||||||
|
nb_id = ["1BE0_10DE"]
|
||||||
|
|
||||||
|
dev_id = nb_id if notebook else dt_id
|
||||||
|
query_obj = {
|
||||||
|
"dIDa": dev_id, # Device PCI IDs:
|
||||||
|
# ["DEVID_VENID_DEVID_VENID"]
|
||||||
|
"osC": os_version, # OS version (Windows 10)
|
||||||
|
"osB": os_build, # OS build
|
||||||
|
"is6": "1" if x86_64 else "0", # 0 - 32bit, 1 - 64bit
|
||||||
|
"lg": str(language), # Language code
|
||||||
|
"iLp": "1" if notebook else "0", # System Is Laptop
|
||||||
|
"prvMd": "0", # Private Model?
|
||||||
|
"gcV": "3.16.0.140", # GeForce Experience client version
|
||||||
|
"gIsB": "1" if beta else "0", # Beta?
|
||||||
|
"dch": "1" if dch else "0" # 0 - Standard Driver, 1 - DCH Driver
|
||||||
|
}
|
||||||
|
try:
|
||||||
|
res = getDispDrvrByDevid(query_obj)
|
||||||
|
except urllib.error.HTTPError as e:
|
||||||
|
if e.code == 404:
|
||||||
|
res = None
|
||||||
|
else:
|
||||||
|
raise e
|
||||||
|
return res
|
||||||
|
|
||||||
|
|
||||||
|
def parse_args():
|
||||||
|
import argparse
|
||||||
|
|
||||||
|
def parse_lang(lang):
|
||||||
|
lang = int(lang)
|
||||||
|
if not (0x0 <= lang <= 0xFFFF):
|
||||||
|
raise ValueError("Bad language ID")
|
||||||
|
return lang
|
||||||
|
|
||||||
|
parser = argparse.ArgumentParser(
|
||||||
|
description="Retrieves info about latest NVIDIA drivers from GeForce "
|
||||||
|
"Experience",
|
||||||
|
formatter_class=argparse.ArgumentDefaultsHelpFormatter)
|
||||||
|
parser.add_argument("-V", "--os-version",
|
||||||
|
default="10.0",
|
||||||
|
help="OS version")
|
||||||
|
parser.add_argument("-B", "--os-build",
|
||||||
|
default="17763",
|
||||||
|
help="OS build")
|
||||||
|
parser.add_argument("-l", "--language",
|
||||||
|
default=1033,
|
||||||
|
type=parse_lang,
|
||||||
|
help="Driver language code")
|
||||||
|
parser.add_argument("-m", "--notebook",
|
||||||
|
help="Query for notebook drivers (Mobile series)",
|
||||||
|
action="store_true")
|
||||||
|
parser.add_argument("-3", "--32bit",
|
||||||
|
help="Query for 32bit drivers",
|
||||||
|
dest="_32bit",
|
||||||
|
action="store_true")
|
||||||
|
parser.add_argument("-b", "--beta",
|
||||||
|
help="Allow beta-versions in search result",
|
||||||
|
action="store_true")
|
||||||
|
parser.add_argument("-D", "--dch",
|
||||||
|
help="Query DCH driver instead of Standard driver",
|
||||||
|
action="store_true")
|
||||||
|
parser.add_argument("-R", "--raw",
|
||||||
|
help="Raw JSON output",
|
||||||
|
action="store_true")
|
||||||
|
args = parser.parse_args()
|
||||||
|
return args
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
import sys
|
||||||
|
args = parse_args()
|
||||||
|
drv = get_latest_geforce_driver(os_version=args.os_version,
|
||||||
|
os_build=args.os_build,
|
||||||
|
language=args.language,
|
||||||
|
notebook=args.notebook,
|
||||||
|
x86_64=(not args._32bit),
|
||||||
|
beta=args.beta,
|
||||||
|
dch=args.dch)
|
||||||
|
if drv is None:
|
||||||
|
print("NOT FOUND")
|
||||||
|
sys.exit(3)
|
||||||
|
if not args.raw:
|
||||||
|
print("Version: %s" % (drv['DriverAttributes']['Version'],))
|
||||||
|
print("Beta: %s" % (bool(int(drv['DriverAttributes']['IsBeta'])),))
|
||||||
|
print("WHQL: %s" % (bool(int(drv['DriverAttributes']['IsWHQL'])),))
|
||||||
|
print("URL: %s" % (drv['DriverAttributes']['DownloadURLAdmin'],))
|
||||||
|
else:
|
||||||
|
json.dump(drv, sys.stdout, indent=4)
|
||||||
|
sys.stdout.flush()
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
149
win/tools/nv-driver-locator/mailer.py
Executable file
149
win/tools/nv-driver-locator/mailer.py
Executable file
@ -0,0 +1,149 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import smtplib
|
||||||
|
import ssl
|
||||||
|
|
||||||
|
|
||||||
|
class Mailer:
|
||||||
|
def __init__(self, *,
|
||||||
|
from_addr,
|
||||||
|
host='localhost',
|
||||||
|
port=None,
|
||||||
|
local_hostname=None,
|
||||||
|
use_ssl=False,
|
||||||
|
use_starttls=False,
|
||||||
|
login=None,
|
||||||
|
password=None,
|
||||||
|
timeout=10):
|
||||||
|
if use_ssl or use_starttls:
|
||||||
|
self._ssl_context = ssl.create_default_context()
|
||||||
|
self._from_addr = from_addr
|
||||||
|
self._host = host
|
||||||
|
self._local_hostname = local_hostname
|
||||||
|
self._use_ssl = use_ssl
|
||||||
|
self._use_starttls = use_starttls
|
||||||
|
self._login = login
|
||||||
|
self._password = password
|
||||||
|
self._timeout = timeout
|
||||||
|
if port is None:
|
||||||
|
if use_ssl:
|
||||||
|
self._port = 465
|
||||||
|
elif use_starttls:
|
||||||
|
self._port = 587
|
||||||
|
else:
|
||||||
|
self._port = 25
|
||||||
|
else:
|
||||||
|
self._port = port
|
||||||
|
|
||||||
|
def send(self, to, msg, mail_options=(), rcpt_options=()):
|
||||||
|
if not self._use_ssl:
|
||||||
|
server = smtplib.SMTP(self._host, self._port, self._local_hostname,
|
||||||
|
self._timeout)
|
||||||
|
else:
|
||||||
|
server = smtplib.SMTP_SSL(self._host, self._port,
|
||||||
|
self._local_hostname,
|
||||||
|
timeout=self._timeout,
|
||||||
|
context=self._ssl_context)
|
||||||
|
|
||||||
|
with server:
|
||||||
|
if self._use_starttls and not self._use_ssl:
|
||||||
|
server.starttls(context=self._ssl_context)
|
||||||
|
if self._login is not None:
|
||||||
|
server.login(self._login, self._password)
|
||||||
|
server.sendmail(self._from_addr, to, msg,
|
||||||
|
mail_options, rcpt_options)
|
||||||
|
|
||||||
|
|
||||||
|
def parse_args():
|
||||||
|
import argparse
|
||||||
|
|
||||||
|
def check_positive_float(val):
|
||||||
|
val = float(val)
|
||||||
|
if val <= 0:
|
||||||
|
raise ValueError("Value %s is not valid positive float" %
|
||||||
|
(repr(val),))
|
||||||
|
return val
|
||||||
|
|
||||||
|
def check_port(val):
|
||||||
|
val = int(val)
|
||||||
|
if not (0 < val <= 0xFFFF):
|
||||||
|
raise ValueError("Value %s is not valid port number" %
|
||||||
|
(repr(val),))
|
||||||
|
return val
|
||||||
|
|
||||||
|
parser = argparse.ArgumentParser(
|
||||||
|
description="Simple email sender, suitable for modern email services.",
|
||||||
|
formatter_class=argparse.ArgumentDefaultsHelpFormatter)
|
||||||
|
parser.add_argument("-f", "--from",
|
||||||
|
required=True,
|
||||||
|
dest="from_address",
|
||||||
|
help="originating address")
|
||||||
|
parser.add_argument("-H", "--smtp-host",
|
||||||
|
default='localhost',
|
||||||
|
help="hostname of local MTA or external SMTP service")
|
||||||
|
parser.add_argument("-P", "--smtp-port",
|
||||||
|
type=check_port,
|
||||||
|
help="SMTP port. "
|
||||||
|
"Default value depends on SSL/TLS mode")
|
||||||
|
parser.add_argument("-L", "--local-hostname",
|
||||||
|
help="hostname to use in EHLO/HELO commands. "
|
||||||
|
"Defaults to autodiscover of local host name.")
|
||||||
|
tls_group = parser.add_mutually_exclusive_group()
|
||||||
|
tls_group.add_argument("-S", "--ssl",
|
||||||
|
help="use SSL from beginning of connection",
|
||||||
|
action="store_true")
|
||||||
|
tls_group.add_argument("-s", "--starttls",
|
||||||
|
help="use STARTTLS command for secure connection",
|
||||||
|
action="store_true")
|
||||||
|
parser.add_argument("-l", "--login",
|
||||||
|
help="user login name. "
|
||||||
|
"If omitted, no login performed.")
|
||||||
|
parser.add_argument("-p", "--password",
|
||||||
|
help="user password used for login")
|
||||||
|
parser.add_argument("-T", "--timeout",
|
||||||
|
type=check_positive_float,
|
||||||
|
default=10.,
|
||||||
|
help="timeout for network operations")
|
||||||
|
parser.add_argument("-j", "--subject",
|
||||||
|
default="",
|
||||||
|
help="email subject")
|
||||||
|
parser.add_argument("-m", "--message",
|
||||||
|
help="email message body. If not specified, message "
|
||||||
|
"will be read from stdin")
|
||||||
|
parser.add_argument("recipient",
|
||||||
|
nargs="+",
|
||||||
|
help="email destination address(es)")
|
||||||
|
|
||||||
|
args = parser.parse_args()
|
||||||
|
return args
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
import sys
|
||||||
|
from email.mime.text import MIMEText
|
||||||
|
|
||||||
|
args = parse_args()
|
||||||
|
m = Mailer(from_addr=args.from_address,
|
||||||
|
host=args.smtp_host,
|
||||||
|
port=args.smtp_port,
|
||||||
|
local_hostname=args.local_hostname,
|
||||||
|
use_ssl=args.ssl,
|
||||||
|
use_starttls=args.starttls,
|
||||||
|
login=args.login,
|
||||||
|
password=args.password,
|
||||||
|
timeout=args.timeout)
|
||||||
|
if args.message is None:
|
||||||
|
print("Reading message from standard input...", file=sys.stderr)
|
||||||
|
msg = sys.stdin.read()
|
||||||
|
else:
|
||||||
|
msg = args.message
|
||||||
|
|
||||||
|
msg = MIMEText(msg)
|
||||||
|
msg['Subject'] = args.subject
|
||||||
|
msg['From'] = args.from_address
|
||||||
|
msg['To'] = ', '.join(args.recipient)
|
||||||
|
m.send(args.recipient, msg.as_string())
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
71
win/tools/nv-driver-locator/nv-driver-locator.json.sample
Normal file
71
win/tools/nv-driver-locator/nv-driver-locator.json.sample
Normal file
@ -0,0 +1,71 @@
|
|||||||
|
{
|
||||||
|
"db": {
|
||||||
|
"type": "file",
|
||||||
|
"params": {
|
||||||
|
"workdir": "/var/lib/nv-driver-locator"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"key_components": [
|
||||||
|
[
|
||||||
|
"DriverAttributes",
|
||||||
|
"Version"
|
||||||
|
]
|
||||||
|
],
|
||||||
|
"channels": [
|
||||||
|
{
|
||||||
|
"type": "gfe_client",
|
||||||
|
"name": "desktop defaults",
|
||||||
|
"params": {}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "gfe_client",
|
||||||
|
"name": "desktop beta",
|
||||||
|
"params": {
|
||||||
|
"beta": true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "gfe_client",
|
||||||
|
"name": "mobile",
|
||||||
|
"params": {
|
||||||
|
"notebook": true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "gfe_client",
|
||||||
|
"name": "mobile beta",
|
||||||
|
"params": {
|
||||||
|
"notebook": true,
|
||||||
|
"beta": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"notifiers": [
|
||||||
|
{
|
||||||
|
"type": "email",
|
||||||
|
"name": "my email",
|
||||||
|
"params": {
|
||||||
|
"from_addr": "notify-bot@gmail.com",
|
||||||
|
"to_addrs": [
|
||||||
|
"recepient1@domain1.tld",
|
||||||
|
"recepient2@domain2.tld"
|
||||||
|
],
|
||||||
|
"host": "smtp.google.com",
|
||||||
|
"use_starttls": true,
|
||||||
|
"login": "notify-bot",
|
||||||
|
"password": "MyGoodPass"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"name": "sample command",
|
||||||
|
"params": {
|
||||||
|
"timeout": 10.0,
|
||||||
|
"cmdline": [
|
||||||
|
"cat",
|
||||||
|
"-"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
290
win/tools/nv-driver-locator/nv-driver-locator.py
Executable file
290
win/tools/nv-driver-locator/nv-driver-locator.py
Executable file
@ -0,0 +1,290 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import sys
|
||||||
|
import json
|
||||||
|
import argparse
|
||||||
|
import hashlib
|
||||||
|
import importlib
|
||||||
|
import logging
|
||||||
|
from abc import ABC, abstractmethod
|
||||||
|
|
||||||
|
|
||||||
|
HASH_DELIM = b'\x00'
|
||||||
|
HASH = hashlib.sha256
|
||||||
|
|
||||||
|
|
||||||
|
class BaseDB(ABC):
|
||||||
|
@abstractmethod
|
||||||
|
def check_key(self, key):
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def set_key(self, key, value):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class FileDB(BaseDB):
|
||||||
|
def __init__(self, workdir):
|
||||||
|
self._ospath = importlib.import_module('os.path')
|
||||||
|
self._tempfile = importlib.import_module('tempfile')
|
||||||
|
self._wd = workdir
|
||||||
|
self._test_writable()
|
||||||
|
|
||||||
|
def _test_writable(self):
|
||||||
|
TEST_STRING = b"test"
|
||||||
|
with self._tempfile.NamedTemporaryFile('w+b', 0, dir=self._wd) as f:
|
||||||
|
f.write(TEST_STRING)
|
||||||
|
f.flush()
|
||||||
|
with open(f.name, 'rb') as tf:
|
||||||
|
assert tf.read() == TEST_STRING, "Test write failed"
|
||||||
|
|
||||||
|
def _get_key_filename(self, key):
|
||||||
|
return self._ospath.join(self._wd, key + '.json')
|
||||||
|
|
||||||
|
def check_key(self, key):
|
||||||
|
filename = self._get_key_filename(key)
|
||||||
|
return self._ospath.isfile(filename)
|
||||||
|
|
||||||
|
def set_key(self, key, obj):
|
||||||
|
filename = self._get_key_filename(key)
|
||||||
|
with open(filename, 'w') as f:
|
||||||
|
json.dump(obj, f, indent=4)
|
||||||
|
f.flush()
|
||||||
|
|
||||||
|
|
||||||
|
class Hasher:
|
||||||
|
def __init__(self, key_components):
|
||||||
|
self._key_components = key_components
|
||||||
|
|
||||||
|
def _eval_key_component(self, obj, component_path):
|
||||||
|
res = obj
|
||||||
|
for path_component in component_path:
|
||||||
|
res = res[path_component]
|
||||||
|
return str(res).encode('utf-8')
|
||||||
|
|
||||||
|
def hash_object(self, obj):
|
||||||
|
return HASH(HASH_DELIM.join(
|
||||||
|
self._eval_key_component(obj, c) for c in self._key_components)
|
||||||
|
).hexdigest()
|
||||||
|
|
||||||
|
|
||||||
|
class BaseNotifier(ABC):
|
||||||
|
@abstractmethod
|
||||||
|
def notify(self, obj):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class EmailNotifier(BaseNotifier):
|
||||||
|
def __init__(self, name, *,
|
||||||
|
from_addr,
|
||||||
|
to_addrs,
|
||||||
|
host='localhost',
|
||||||
|
port=None,
|
||||||
|
local_hostname=None,
|
||||||
|
use_ssl=False,
|
||||||
|
use_starttls=False,
|
||||||
|
login=None,
|
||||||
|
password=None,
|
||||||
|
timeout=10):
|
||||||
|
self.name = name
|
||||||
|
self._from_addr = from_addr
|
||||||
|
self._Mailer = importlib.import_module('mailer').Mailer
|
||||||
|
self._MIMEText = importlib.import_module('email.mime.text').MIMEText
|
||||||
|
self._MIMEMult = importlib.import_module(
|
||||||
|
'email.mime.multipart').MIMEMultipart
|
||||||
|
self._MIMEBase = importlib.import_module('email.mime.base').MIMEBase
|
||||||
|
self._encoders = importlib.import_module('email.encoders')
|
||||||
|
self._m = self._Mailer(from_addr=from_addr,
|
||||||
|
host=host,
|
||||||
|
port=port,
|
||||||
|
local_hostname=local_hostname,
|
||||||
|
use_ssl=use_ssl,
|
||||||
|
use_starttls=use_starttls,
|
||||||
|
login=login,
|
||||||
|
password=password,
|
||||||
|
timeout=timeout)
|
||||||
|
self._to_addrs = to_addrs
|
||||||
|
|
||||||
|
def notify(self, obj):
|
||||||
|
msg = self._MIMEMult()
|
||||||
|
msg['Subject'] = "New Nvidia driver available!"
|
||||||
|
msg['From'] = self._from_addr
|
||||||
|
msg['To'] = ', '.join(self._to_addrs)
|
||||||
|
body = "See attached JSON"
|
||||||
|
msg.attach(self._MIMEText(body, 'plain'))
|
||||||
|
p = self._MIMEBase('application', 'octet-stream')
|
||||||
|
p.set_payload(json.dumps(obj, indent=4).encode('utf-8'))
|
||||||
|
self._encoders.encode_base64(p)
|
||||||
|
p.add_header('Content-Disposition', "attachment; filename=obj.json")
|
||||||
|
msg.attach(p)
|
||||||
|
self._m.send(self._to_addrs, msg.as_string())
|
||||||
|
|
||||||
|
|
||||||
|
class CommandNotifier(BaseNotifier):
|
||||||
|
def __init__(self, name, *,
|
||||||
|
cmdline,
|
||||||
|
timeout=10):
|
||||||
|
self.name = name
|
||||||
|
self._subprocess = importlib.import_module('subprocess')
|
||||||
|
self._cmdline = cmdline
|
||||||
|
self._timeout = timeout
|
||||||
|
|
||||||
|
def notify(self, obj):
|
||||||
|
proc = self._subprocess.Popen(self._cmdline,
|
||||||
|
stdin=self._subprocess.PIPE)
|
||||||
|
try:
|
||||||
|
proc.communicate(json.dumps(obj, indent=4).encode('utf-8'),
|
||||||
|
self._timeout)
|
||||||
|
except self._subprocess.TimeoutExpired:
|
||||||
|
proc.kill()
|
||||||
|
proc.communicate()
|
||||||
|
|
||||||
|
|
||||||
|
class BaseChannel(ABC):
|
||||||
|
@abstractmethod
|
||||||
|
def get_latest_driver(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class GFEClientChannel(BaseChannel):
|
||||||
|
def __init__(self, name, **kwargs):
|
||||||
|
self.name = name
|
||||||
|
self._kwargs = kwargs
|
||||||
|
gfe_get_driver = importlib.import_module('gfe_get_driver')
|
||||||
|
self._get_latest_driver = gfe_get_driver.get_latest_geforce_driver
|
||||||
|
|
||||||
|
def get_latest_driver(self):
|
||||||
|
return self._get_latest_driver(**self._kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
def parse_args():
|
||||||
|
parser = argparse.ArgumentParser(
|
||||||
|
description="Watches for GeForce experience driver updates for "
|
||||||
|
"configured systems",
|
||||||
|
formatter_class=argparse.ArgumentDefaultsHelpFormatter)
|
||||||
|
parser.add_argument("-c", "--config",
|
||||||
|
default="/etc/nv-driver-locator.json",
|
||||||
|
help="config file location")
|
||||||
|
args = parser.parse_args()
|
||||||
|
return args
|
||||||
|
|
||||||
|
|
||||||
|
class DriverLocator:
|
||||||
|
_ret_code = 0
|
||||||
|
|
||||||
|
def __init__(self, conf):
|
||||||
|
self._logger = logging.getLogger(self.__class__.__name__)
|
||||||
|
self._channels = self._construct_channels(conf['channels'])
|
||||||
|
self._db = self._construct_db(conf['db'])
|
||||||
|
self._hasher = Hasher(conf['key_components'])
|
||||||
|
self._notifiers = self._construct_notifiers(conf['notifiers'])
|
||||||
|
|
||||||
|
def _construct_channels(self, channels_config):
|
||||||
|
channel_types = {
|
||||||
|
'gfe_client': GFEClientChannel,
|
||||||
|
}
|
||||||
|
|
||||||
|
channels = []
|
||||||
|
for ch in channels_config:
|
||||||
|
try:
|
||||||
|
ctor = channel_types[ch['type']]
|
||||||
|
C = ctor(ch['name'], **ch['params'])
|
||||||
|
except Exception as e:
|
||||||
|
self._perror("Channel construction failed with exception: %s. "
|
||||||
|
"Skipping..." % (str(e),))
|
||||||
|
else:
|
||||||
|
channels.append(C)
|
||||||
|
return channels
|
||||||
|
|
||||||
|
def _construct_db(self, db_config):
|
||||||
|
db_types = {
|
||||||
|
'file': FileDB,
|
||||||
|
}
|
||||||
|
ctor = db_types[db_config['type']]
|
||||||
|
db = ctor(**db_config['params'])
|
||||||
|
return db
|
||||||
|
|
||||||
|
def _construct_notifiers(self, notifiers_config):
|
||||||
|
notifier_types = {
|
||||||
|
'email': EmailNotifier,
|
||||||
|
'command': CommandNotifier,
|
||||||
|
}
|
||||||
|
|
||||||
|
notifiers = []
|
||||||
|
for nc in notifiers_config:
|
||||||
|
try:
|
||||||
|
ctor = notifier_types[nc['type']]
|
||||||
|
N = ctor(nc['name'], **nc['params'])
|
||||||
|
except Exception as e:
|
||||||
|
self._perror("Notifier construction failed with exception: %s."
|
||||||
|
" Skipping..." % (str(e),))
|
||||||
|
else:
|
||||||
|
notifiers.append(N)
|
||||||
|
return notifiers
|
||||||
|
|
||||||
|
def _perror(self, err):
|
||||||
|
self._ret_code = 3
|
||||||
|
self._logger.error(err)
|
||||||
|
|
||||||
|
def _notify_all(self, obj):
|
||||||
|
fails = 0
|
||||||
|
for n in self._notifiers:
|
||||||
|
try:
|
||||||
|
n.notify(obj)
|
||||||
|
except Exception as e:
|
||||||
|
self._perror("Notify channel %s failed with exception: %s." %
|
||||||
|
(n.name, str(e)))
|
||||||
|
fails += 1
|
||||||
|
return fails < len(self._notifiers)
|
||||||
|
|
||||||
|
def run(self):
|
||||||
|
for ch in self._channels:
|
||||||
|
try:
|
||||||
|
drv = ch.get_latest_driver()
|
||||||
|
except Exception as e:
|
||||||
|
self._perror("get_latest_driver() invocation failed for "
|
||||||
|
"channel %s. Exception: %s. Continuing..." %
|
||||||
|
(repr(ch.name), str(e)))
|
||||||
|
continue
|
||||||
|
if drv is None:
|
||||||
|
self._perror("Driver not found for channel %s" %
|
||||||
|
(repr(ch.name),))
|
||||||
|
continue
|
||||||
|
try:
|
||||||
|
key = self._hasher.hash_object(drv)
|
||||||
|
except Exception as e:
|
||||||
|
self._perror("Key evaluation failed for channel %s. "
|
||||||
|
"Exception: %s" % (repr(name), str(e)))
|
||||||
|
continue
|
||||||
|
if not self._db.check_key(key):
|
||||||
|
if self._notify_all(drv):
|
||||||
|
self._db.set_key(key, drv)
|
||||||
|
return self._ret_code
|
||||||
|
|
||||||
|
|
||||||
|
def setup_logger(name, verbosity):
|
||||||
|
logger = logging.getLogger(name)
|
||||||
|
logger.setLevel(verbosity)
|
||||||
|
handler = logging.StreamHandler()
|
||||||
|
handler.setLevel(verbosity)
|
||||||
|
handler.setFormatter(logging.Formatter('%(asctime)s '
|
||||||
|
'%(levelname)-8s '
|
||||||
|
'%(name)s: %(message)s',
|
||||||
|
'%Y-%m-%d %H:%M:%S'))
|
||||||
|
logger.addHandler(handler)
|
||||||
|
return logger
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
args = parse_args()
|
||||||
|
setup_logger(DriverLocator.__name__, logging.ERROR)
|
||||||
|
|
||||||
|
with open(args.config, 'r') as conf_file:
|
||||||
|
conf = json.load(conf_file)
|
||||||
|
|
||||||
|
ret = DriverLocator(conf).run()
|
||||||
|
sys.exit(ret)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
Loading…
Reference in New Issue
Block a user