Compare commits
15 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
418c62b40d | ||
|
|
198a480c35 | ||
|
|
2d4bf4e847 | ||
|
|
774e3d1a62 | ||
|
|
ade3cd6fee | ||
|
|
28dc3ccb84 | ||
|
|
6ca8582ec3 | ||
|
|
74b4aeb44a | ||
|
|
9c7ab5ac57 | ||
|
|
65ecf1fea2 | ||
|
|
44142c658b | ||
|
|
25a205ce6c | ||
|
|
25094084cf | ||
|
|
4eac601153 | ||
|
|
f0d8d66b9b |
@@ -68,20 +68,23 @@ class BOSMiner(BaseMiner):
|
|||||||
|
|
||||||
async def send_graphql_query(self, query) -> Union[dict, None]:
|
async def send_graphql_query(self, query) -> Union[dict, None]:
|
||||||
url = f"http://{self.ip}/graphql"
|
url = f"http://{self.ip}/graphql"
|
||||||
async with httpx.AsyncClient() as client:
|
try:
|
||||||
_auth = await client.post(
|
async with httpx.AsyncClient() as client:
|
||||||
url,
|
_auth = await client.post(
|
||||||
json={
|
url,
|
||||||
"query": 'mutation{auth{login(username:"'
|
json={
|
||||||
+ self.uname
|
"query": 'mutation{auth{login(username:"'
|
||||||
+ '", password:"'
|
+ self.uname
|
||||||
+ self.pwd
|
+ '", password:"'
|
||||||
+ '"){__typename}}}'
|
+ self.pwd
|
||||||
},
|
+ '"){__typename}}}'
|
||||||
)
|
},
|
||||||
d = await client.post(url, json={"query": query})
|
)
|
||||||
if d.status_code == 200:
|
d = await client.post(url, json={"query": query})
|
||||||
return d.json()
|
if d.status_code == 200:
|
||||||
|
return d.json()
|
||||||
|
except (httpx.ReadError, httpx.ReadTimeout):
|
||||||
|
return None
|
||||||
return None
|
return None
|
||||||
|
|
||||||
async def fault_light_on(self) -> bool:
|
async def fault_light_on(self) -> bool:
|
||||||
@@ -481,7 +484,7 @@ class BOSMiner(BaseMiner):
|
|||||||
wattage_limit = tuner[0].get("PowerLimit")
|
wattage_limit = tuner[0].get("PowerLimit")
|
||||||
if wattage_limit:
|
if wattage_limit:
|
||||||
data.wattage_limit = wattage_limit
|
data.wattage_limit = wattage_limit
|
||||||
if wattage:
|
if wattage is not None:
|
||||||
data.wattage = wattage
|
data.wattage = wattage
|
||||||
|
|
||||||
chain_status = tuner[0].get("TunerChainStatus")
|
chain_status = tuner[0].get("TunerChainStatus")
|
||||||
@@ -606,7 +609,7 @@ class BOSMiner(BaseMiner):
|
|||||||
try:
|
try:
|
||||||
data.wattage = query_data["bosminer"]["info"]["workSolver"]["power"]["approxConsumptionW"]
|
data.wattage = query_data["bosminer"]["info"]["workSolver"]["power"]["approxConsumptionW"]
|
||||||
except (TypeError, KeyError, ValueError, IndexError):
|
except (TypeError, KeyError, ValueError, IndexError):
|
||||||
pass
|
data.wattage = 0
|
||||||
try:
|
try:
|
||||||
data.wattage_limit = query_data["bosminer"]["info"]["workSolver"]["power"]["limitW"]
|
data.wattage_limit = query_data["bosminer"]["info"]["workSolver"]["power"]["limitW"]
|
||||||
except (TypeError, KeyError, ValueError, IndexError):
|
except (TypeError, KeyError, ValueError, IndexError):
|
||||||
|
|||||||
@@ -52,7 +52,7 @@ class BaseMiner(ABC):
|
|||||||
return object.__new__(cls)
|
return object.__new__(cls)
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return f"{'' if not self.api_type else self.api_type} {'' if not self.model else self.model}: {str(self.ip)}"
|
return f"{'' if not self.api_type else self.api_type}{'' if not self.model else ' ' + self.model}: {str(self.ip)}"
|
||||||
|
|
||||||
def __lt__(self, other):
|
def __lt__(self, other):
|
||||||
return ipaddress.ip_address(self.ip) < ipaddress.ip_address(other.ip)
|
return ipaddress.ip_address(self.ip) < ipaddress.ip_address(other.ip)
|
||||||
|
|||||||
@@ -337,7 +337,6 @@ class MinerFactory(metaclass=Singleton):
|
|||||||
break
|
break
|
||||||
except asyncio.TimeoutError:
|
except asyncio.TimeoutError:
|
||||||
logging.warning(f"{ip}: Get Miner Timed Out")
|
logging.warning(f"{ip}: Get Miner Timed Out")
|
||||||
|
|
||||||
miner = self._select_miner_from_classes(ip, model, api, ver)
|
miner = self._select_miner_from_classes(ip, model, api, ver)
|
||||||
|
|
||||||
# save the miner to the cache at its IP if its not unknown
|
# save the miner to the cache at its IP if its not unknown
|
||||||
@@ -412,7 +411,9 @@ class MinerFactory(metaclass=Singleton):
|
|||||||
# miner refused connection on API port, we wont be able to get data this way
|
# miner refused connection on API port, we wont be able to get data this way
|
||||||
# try ssh
|
# try ssh
|
||||||
try:
|
try:
|
||||||
_model = await self.__get_model_from_ssh(ip)
|
_model = await self.__get_model_from_graphql(ip)
|
||||||
|
if not _model:
|
||||||
|
_model = await self.__get_model_from_ssh(ip)
|
||||||
if _model:
|
if _model:
|
||||||
model = _model
|
model = _model
|
||||||
api = "BOSMiner+"
|
api = "BOSMiner+"
|
||||||
@@ -433,12 +434,6 @@ class MinerFactory(metaclass=Singleton):
|
|||||||
|
|
||||||
# if we have devdetails, we can get model data from there
|
# if we have devdetails, we can get model data from there
|
||||||
if devdetails:
|
if devdetails:
|
||||||
if devdetails == {"Msg": "Disconnected"}:
|
|
||||||
model = await self.__get_model_from_graphql(ip)
|
|
||||||
if model:
|
|
||||||
api = "BOSMiner+"
|
|
||||||
return model, api, ver
|
|
||||||
|
|
||||||
for _devdetails_key in ["Model", "Driver"]:
|
for _devdetails_key in ["Model", "Driver"]:
|
||||||
try:
|
try:
|
||||||
model = devdetails["DEVDETAILS"][0][_devdetails_key].upper()
|
model = devdetails["DEVDETAILS"][0][_devdetails_key].upper()
|
||||||
@@ -446,15 +441,42 @@ class MinerFactory(metaclass=Singleton):
|
|||||||
break
|
break
|
||||||
except KeyError:
|
except KeyError:
|
||||||
continue
|
continue
|
||||||
|
try:
|
||||||
|
if devdetails[0]["STATUS"][0]["Msg"]:
|
||||||
|
model = await self.__get_model_from_graphql(ip)
|
||||||
|
if model:
|
||||||
|
api = "BOSMiner+"
|
||||||
|
except (KeyError, TypeError, ValueError, IndexError):
|
||||||
|
pass
|
||||||
|
|
||||||
if not model:
|
if not model:
|
||||||
# braiins OS bug check just in case
|
# braiins OS bug check just in case
|
||||||
if "s9" in devdetails["STATUS"][0]["Description"]:
|
if "s9" in devdetails["STATUS"][0]["Description"]:
|
||||||
model = "ANTMINER S9"
|
model = "ANTMINER S9"
|
||||||
if "s17" in version["STATUS"][0]["Description"]:
|
if "s17" in version["STATUS"][0]["Description"]:
|
||||||
model = "ANTMINER S17"
|
model = "ANTMINER S17"
|
||||||
|
if not api:
|
||||||
|
if "boser" in version["STATUS"][0]["Description"]:
|
||||||
|
api = "BOSMiner+"
|
||||||
|
else:
|
||||||
|
try:
|
||||||
|
_model = await self.__get_model_from_graphql(ip)
|
||||||
|
if _model:
|
||||||
|
model = _model
|
||||||
|
api = "BOSMiner+"
|
||||||
|
except (KeyError, TypeError, ValueError, IndexError):
|
||||||
|
pass
|
||||||
|
|
||||||
# if we have version we can get API type from here
|
# if we have version we can get API type from here
|
||||||
if version:
|
if version:
|
||||||
|
try:
|
||||||
|
if version[0]["STATUS"][0]["Msg"]:
|
||||||
|
model = await self.__get_model_from_graphql(ip)
|
||||||
|
if model:
|
||||||
|
api = "BOSMiner+"
|
||||||
|
return model, api, ver
|
||||||
|
except (KeyError, TypeError, ValueError, IndexError):
|
||||||
|
pass
|
||||||
if "VERSION" in version:
|
if "VERSION" in version:
|
||||||
api_types = ["BMMiner", "CGMiner", "BTMiner"]
|
api_types = ["BMMiner", "CGMiner", "BTMiner"]
|
||||||
# check basic API types, BOSMiner needs a special check
|
# check basic API types, BOSMiner needs a special check
|
||||||
@@ -470,6 +492,8 @@ class MinerFactory(metaclass=Singleton):
|
|||||||
api = "BOSMiner+"
|
api = "BOSMiner+"
|
||||||
if "BOSminer+" in version["VERSION"][0]:
|
if "BOSminer+" in version["VERSION"][0]:
|
||||||
api = "BOSMiner+"
|
api = "BOSMiner+"
|
||||||
|
if any("BOSer" in string for string in version["VERSION"][0]):
|
||||||
|
api = "BOSMiner+"
|
||||||
|
|
||||||
# check for avalonminers
|
# check for avalonminers
|
||||||
for _version_key in ["PROD", "MODEL"]:
|
for _version_key in ["PROD", "MODEL"]:
|
||||||
@@ -533,7 +557,6 @@ class MinerFactory(metaclass=Singleton):
|
|||||||
# don't need "Bitmain", just "ANTMINER XX" as model
|
# don't need "Bitmain", just "ANTMINER XX" as model
|
||||||
if "BITMAIN " in model:
|
if "BITMAIN " in model:
|
||||||
model = model.replace("BITMAIN ", "")
|
model = model.replace("BITMAIN ", "")
|
||||||
|
|
||||||
return model, api, ver
|
return model, api, ver
|
||||||
|
|
||||||
async def __get_devdetails_and_version(
|
async def __get_devdetails_and_version(
|
||||||
@@ -548,7 +571,7 @@ class MinerFactory(metaclass=Singleton):
|
|||||||
if not validation[0]:
|
if not validation[0]:
|
||||||
try:
|
try:
|
||||||
if data["version"][0]["STATUS"][0]["Msg"] == "Disconnected":
|
if data["version"][0]["STATUS"][0]["Msg"] == "Disconnected":
|
||||||
return {"Msg": "Disconnected"}, None
|
return data["devdetails"], data["version"]
|
||||||
except KeyError:
|
except KeyError:
|
||||||
pass
|
pass
|
||||||
raise APIError(validation[1])
|
raise APIError(validation[1])
|
||||||
@@ -605,7 +628,7 @@ class MinerFactory(metaclass=Singleton):
|
|||||||
async with httpx.AsyncClient() as client:
|
async with httpx.AsyncClient() as client:
|
||||||
d = await client.post(url, json={"query": "{bosminer {info{modelName}}}"})
|
d = await client.post(url, json={"query": "{bosminer {info{modelName}}}"})
|
||||||
if d.status_code == 200:
|
if d.status_code == 200:
|
||||||
model = d.json()["data"]["bosminer"]["info"]["modelName"].upper()
|
model = (d.json()["data"]["bosminer"]["info"]["modelName"]).upper()
|
||||||
return model
|
return model
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
[tool.poetry]
|
[tool.poetry]
|
||||||
name = "pyasic"
|
name = "pyasic"
|
||||||
version = "0.21.5"
|
version = "0.21.12"
|
||||||
description = "A set of modules for interfacing with many common types of ASIC bitcoin miners, using both their API and SSH."
|
description = "A set of modules for interfacing with many common types of ASIC bitcoin miners, using both their API and SSH."
|
||||||
authors = ["UpstreamData <brett@upstreamdata.ca>"]
|
authors = ["UpstreamData <brett@upstreamdata.ca>"]
|
||||||
repository = "https://github.com/UpstreamData/pyasic"
|
repository = "https://github.com/UpstreamData/pyasic"
|
||||||
|
|||||||
Reference in New Issue
Block a user