Skip to content

Instantly share code, notes, and snippets.

@Krailon
Forked from YoRyan/excavator-driver.py
Last active April 15, 2018 12:10
Show Gist options
  • Save Krailon/c47e9c3f32731d59ec1bbf996c9691a2 to your computer and use it in GitHub Desktop.
Save Krailon/c47e9c3f32731d59ec1bbf996c9691a2 to your computer and use it in GitHub Desktop.
Cross-platform controller for NiceHash Excavator for Nvidia (aka, NiceHash 2 for Linux). Updated for Excavator v1.5.1a (API v0.1.1).
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
"""Cross-platform controller for NiceHash Excavator for Nvidia."""
# Example usage:
# $ excavator -p 3456 &
# $ python3 excavator-driver.py
# History:
# 2017-12-03: initial version
# 2018-01-25: group devices by common algorithm; wait for excavator on startup
# 2018-04-15: [NOP-Gate] Updated for Excavator v1.5.1a (API v0.1.1)
__author__ = "Ryan Young"
__modifiedby__ = 'Krailon|NOP-Gate'
__email__ = "[email protected]"
__license__ = "public domain"
import json
import logging
import signal
import socket
import sys
import urllib.error
import urllib.request
from time import sleep
WALLET_ADDR = 'YOURWALLETADDRESS'
WORKER_NAME = 'worker0'
REGION = 'usa' # eu, usa, hk, jp, in, br
EXCAVATOR_ADDRESS = ('127.0.0.1', 3456)
# copy the numbers from excavator-benchmark (test one device at a time with -d <n>)
# convert to the base unit, H/s
# x H/s -> x
# x kH/s -> x*1e3
# x MH/s -> x*1e6
# x GH/s -> x*1e9
BENCHMARKS = {}
BENCHMARKS[0] = {
# Replace these with your GPUs benchmark results
'equihash': 0,
'pascal': 0,
'decred': 0,
'sia': 0,
'lbry': 0,
'blake2s': 0,
'lyra2rev2': 0,
'cryptonight': 0,
'daggerhashimoto': 0,
'daggerhashimoto_pascal': [0, 0],
'daggerhashimoto_decred': [0, 0],
'daggerhashimoto_sia': [0, 0],
# test manually
'neoscrypt': 0,
'nist5': 0
}]
# Duplicate benchmark results for identical second GPU
BENCHMARKS[1] = BENCHMARKS[0]
PROFIT_SWITCH_THRESHOLD = 0.1
UPDATE_INTERVAL = 60
EXCAVATOR_TIMEOUT = 10
NICEHASH_TIMEOUT = 20
### here be dragons
class ExcavatorError(Exception):
pass
class ExcavatorAPIError(ExcavatorError):
"""Exception returned by excavator."""
def __init__(self, response):
self.response = response
self.error = response['error']
def nicehash_multialgo_info():
"""Retrieves pay rates and connection ports for every algorithm from the NiceHash API."""
response = urllib.request.urlopen('https://api.nicehash.com/api?method=simplemultialgo.info',
None, NICEHASH_TIMEOUT)
query = json.loads(response.read().decode('ascii')) #json.load(response)
paying = {}
ports = {}
for algorithm in query['result']['simplemultialgo']:
name = algorithm['name']
paying[name] = float(algorithm['paying'])
ports[name] = int(algorithm['port'])
return paying, ports
def nicehash_mbtc_per_day(device, paying):
"""Calculates the BTC/day amount for every algorithm.
device -- excavator device id for benchmarks
paying -- algorithm pay information from NiceHash
"""
benchmarks = BENCHMARKS[device]
pay = lambda algo, speed: paying[algo]*speed*(24*60*60)*1e-11
def pay_benched(algo):
if '_' in algo:
return sum([pay(multi_algo, benchmarks[algo][i]) for
i, multi_algo in enumerate(algo.split('_'))])
else:
return pay(algo, benchmarks[algo])
return dict([(algo, pay_benched(algo)) for algo in benchmarks.keys()])
def do_excavator_command(method, params):
"""Sends a command to excavator, returns the JSON-encoded response.
method -- name of the command to execute
params -- list of arguments for the command
"""
BUF_SIZE = 1024
command = {
'id': 1,
'method': method,
'params': params
}
s = socket.create_connection(EXCAVATOR_ADDRESS, EXCAVATOR_TIMEOUT)
# send newline-terminated command
s.sendall((json.dumps(command).replace('\n', '\\n') + '\n').encode())
response = ''
while True:
chunk = s.recv(BUF_SIZE).decode()
# excavator responses are newline-terminated too
if '\n' in chunk:
response += chunk[:chunk.index('\n')]
break
else:
response += chunk
s.close()
response_data = json.loads(response)
if response_data['error'] is None:
return response_data
else:
raise ExcavatorAPIError(response_data)
def main():
"""Main program."""
logging.basicConfig(format='%(asctime)s %(levelname)s: %(message)s',
level=logging.INFO)
# dict of algorithm name -> (excavator id, [attached devices])
algorithm_status = {}
# dict of device id -> excavator worker id
worker_status = {}
device_algorithm = lambda device: [a for a in algorithm_status.keys() if
device in algorithm_status[a][1]][0]
def dispatch_device(device, algo, ports):
if algo in algorithm_status:
algo_id = algorithm_status[algo][0]
algorithm_status[algo][1].append(device)
else:
response = do_excavator_command('algorithm.add', [algo])
algo_id = response['id']
algorithm_status[algo] = (algo_id, [device])
response = do_excavator_command('worker.add', [algo, str(device)])
worker_status[device] = response['worker_id']
def free_device(device):
algo = device_algorithm(device)
algorithm_status[algo][1].remove(device)
worker_id = worker_status[device]
worker_status.pop(device)
do_excavator_command('worker.free', [str(worker_id)])
if len(algorithm_status[algo][1]) == 0: # no more devices attached
algo_id = algorithm_status[algo][0]
algorithm_status.pop(algo)
do_excavator_command('algorithm.remove', [algo])
def sigint_handler(signum, frame):
logging.info('cleaning up!')
active_devices = list(worker_status.keys())
for device in active_devices:
free_device(device)
sys.exit(0)
signal.signal(signal.SIGINT, sigint_handler)
def contact_excavator():
try:
do_excavator_command('message', ['%s connected' % sys.argv[0]])
except (socket.timeout, socket.error):
return False
else:
return True
logging.info('connecting to excavator at %s:%d' % EXCAVATOR_ADDRESS)
while not contact_excavator():
sleep(5)
while True:
try:
paying, ports = nicehash_multialgo_info()
except urllib.error.URLError as err:
logging.warning('failed to retrieve NiceHash stats: %s' % err.reason)
except urllib.error.HTTPError as err:
logging.warning('server error retrieving NiceHash stats: %s %s'
% (err.code, err.reason))
except socket.timeout:
logging.warning('failed to retrieve NiceHash stats: timed out')
except (json.decoder.JSONDecodeError, KeyError):
logging.warning('failed to parse NiceHash stats')
else:
for device in BENCHMARKS.keys():
payrates = nicehash_mbtc_per_day(device, paying)
best_algo = max(payrates.keys(), key=lambda algo: payrates[algo])
if device not in worker_status:
logging.info('device %s initial algorithm is %s (%.2f mBTC/day)'
% (device, best_algo, payrates[best_algo]))
dispatch_device(device, best_algo, ports)
else:
current_algo = device_algorithm(device)
if current_algo != best_algo and \
(payrates[current_algo] == 0 or \
payrates[best_algo]/payrates[current_algo] >= 1.0 + PROFIT_SWITCH_THRESHOLD):
logging.info('switching device %s to %s (%.2f mBTC/day)'
% (device, best_algo, payrates[best_algo]))
free_device(device)
dispatch_device(device, best_algo, ports)
sleep(UPDATE_INTERVAL)
if __name__ == '__main__':
main()
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment