Compare commits
18 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
55786b154d | ||
|
|
6ab9681dec | ||
|
|
89641c6316 | ||
|
|
136ff6a688 | ||
|
|
d918d93f4a | ||
|
|
8046c532a6 | ||
|
|
92820a362d | ||
|
|
9fd90031a9 | ||
|
|
2f2223a112 | ||
|
|
50e6cf9dfd | ||
|
|
1b5e3093e6 | ||
|
|
9e3578b4a2 | ||
|
|
a3087e1a96 | ||
|
|
4b16ea2ca2 | ||
|
|
5dd361c4ef | ||
|
|
098112742c | ||
|
|
cd31e0743e | ||
|
|
1a7d0bf7cc |
6
.github/workflows/python-publish.yml
vendored
6
.github/workflows/python-publish.yml
vendored
@@ -13,10 +13,10 @@ jobs:
|
||||
build:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v2
|
||||
- uses: actions/checkout@v4.2.2
|
||||
- name: Publish GH release
|
||||
uses: softprops/action-gh-release@v0.1.14
|
||||
uses: softprops/action-gh-release@v2.1.0
|
||||
- name: Build using poetry and publish to PyPi
|
||||
uses: JRubics/poetry-publish@v1.11
|
||||
uses: JRubics/poetry-publish@v2.0
|
||||
with:
|
||||
pypi_token: ${{ secrets.PYPI_API_KEY }}
|
||||
|
||||
@@ -302,6 +302,13 @@
|
||||
show_root_heading: false
|
||||
heading_level: 4
|
||||
|
||||
## S19k Pro (VNish)
|
||||
::: pyasic.miners.antminer.vnish.X19.S19.VNishS19kPro
|
||||
handler: python
|
||||
options:
|
||||
show_root_heading: false
|
||||
heading_level: 4
|
||||
|
||||
## T19 (VNish)
|
||||
::: pyasic.miners.antminer.vnish.X19.T19.VNishT19
|
||||
handler: python
|
||||
|
||||
@@ -533,6 +533,7 @@ details {
|
||||
<li><a href="../antminer/X19#s19a-vnish">S19a (VNish)</a></li>
|
||||
<li><a href="../antminer/X19#s19a-pro-vnish">S19a Pro (VNish)</a></li>
|
||||
<li><a href="../antminer/X19#s19-pro-hydro-vnish">S19 Pro Hydro (VNish)</a></li>
|
||||
<li><a href="../antminer/X19#s19k-pro-vnish">S19k Pro (VNish)</a></li>
|
||||
<li><a href="../antminer/X19#t19-vnish">T19 (VNish)</a></li>
|
||||
</ul>
|
||||
</details>
|
||||
|
||||
@@ -22,7 +22,7 @@ class MiningPreset(MinerConfigValue):
|
||||
hashrate = None
|
||||
else:
|
||||
power = hr_power_split[0].replace("watt", "").strip()
|
||||
hashrate = hr_power_split[1].replace("TH", "").strip()
|
||||
hashrate = hr_power_split[1].replace("TH", "").replace(" LC", "").strip()
|
||||
tuned = web_preset["status"] == "tuned"
|
||||
modded_psu = web_preset["modded_psu_required"]
|
||||
return cls(
|
||||
|
||||
@@ -13,19 +13,20 @@
|
||||
# See the License for the specific language governing permissions and -
|
||||
# limitations under the License. -
|
||||
# ------------------------------------------------------------------------------
|
||||
|
||||
import copy
|
||||
import time
|
||||
from datetime import datetime, timezone
|
||||
from typing import Any, List, Union
|
||||
|
||||
from pydantic import BaseModel, Field, computed_field, field_serializer
|
||||
from pydantic import BaseModel, Field, computed_field
|
||||
|
||||
from pyasic.config import MinerConfig
|
||||
from pyasic.config.mining import MiningModePowerTune
|
||||
from pyasic.data.pools import PoolMetrics, Scheme
|
||||
from pyasic.device.algorithm.hashrate import AlgoHashRateType
|
||||
from pyasic.device.algorithm.hashrate.base import GenericHashrate
|
||||
|
||||
from ..device.algorithm.hashrate.unit.base import GenericUnit
|
||||
from .boards import HashBoard
|
||||
from .device import DeviceInfo
|
||||
from .error_codes import BraiinsOSError, InnosiliconError, WhatsminerError, X19Error
|
||||
@@ -96,7 +97,7 @@ class MinerData(BaseModel):
|
||||
expected_fans: int | None = None
|
||||
|
||||
# temperature
|
||||
env_temp: int | None = None
|
||||
env_temp: float | None = None
|
||||
|
||||
# power
|
||||
wattage: int | None = None
|
||||
@@ -135,9 +136,6 @@ class MinerData(BaseModel):
|
||||
def fields(cls):
|
||||
return list(cls.model_fields.keys())
|
||||
|
||||
def __post_init__(self):
|
||||
self.raw_datetime = datetime.now(timezone.utc).astimezone()
|
||||
|
||||
def get(self, __key: str, default: Any = None):
|
||||
try:
|
||||
val = self.__getitem__(__key)
|
||||
@@ -198,35 +196,23 @@ class MinerData(BaseModel):
|
||||
|
||||
@computed_field # type: ignore[misc]
|
||||
@property
|
||||
def hashrate(self) -> AlgoHashRateType:
|
||||
def hashrate(self) -> AlgoHashRateType | None:
|
||||
if len(self.hashboards) > 0:
|
||||
hr_data = []
|
||||
for item in self.hashboards:
|
||||
if item.hashrate is not None:
|
||||
hr_data.append(item.hashrate)
|
||||
if len(hr_data) > 0:
|
||||
return sum(hr_data, start=type(hr_data[0])(rate=0))
|
||||
return sum(hr_data, start=self.hashboards[0].hashrate.__class__(rate=0))
|
||||
return self.raw_hashrate
|
||||
|
||||
@field_serializer("hashrate")
|
||||
def serialize_hashrate(self, hashrate: AlgoHashRateType | None) -> float:
|
||||
if hashrate is not None:
|
||||
return float(hashrate)
|
||||
|
||||
@field_serializer("expected_hashrate")
|
||||
def serialize_expected_hashrate(
|
||||
self, expected_hashrate: AlgoHashRateType | None, _info
|
||||
) -> float:
|
||||
if expected_hashrate is not None:
|
||||
return float(expected_hashrate)
|
||||
|
||||
@hashrate.setter
|
||||
def hashrate(self, val):
|
||||
self.raw_hashrate = val
|
||||
|
||||
@computed_field # type: ignore[misc]
|
||||
@property
|
||||
def wattage_limit(self) -> int:
|
||||
def wattage_limit(self) -> int | None:
|
||||
if self.config is not None:
|
||||
if isinstance(self.config.mining_mode, MiningModePowerTune):
|
||||
return self.config.mining_mode.power
|
||||
|
||||
@@ -51,11 +51,6 @@ class HashBoard(BaseModel):
|
||||
active: bool | None = None
|
||||
voltage: float | None = None
|
||||
|
||||
@field_serializer("hashrate")
|
||||
def serialize_hashrate(self, hashrate: AlgoHashRateType | None) -> float:
|
||||
if hashrate is not None:
|
||||
return float(hashrate)
|
||||
|
||||
def get(self, __key: str, default: Any = None):
|
||||
try:
|
||||
val = self.__getitem__(__key)
|
||||
|
||||
@@ -1,19 +1,42 @@
|
||||
from __future__ import annotations
|
||||
|
||||
from abc import ABC, abstractmethod
|
||||
|
||||
from pydantic import BaseModel
|
||||
from pydantic import BaseModel, field_serializer
|
||||
from typing_extensions import Self
|
||||
|
||||
from .unit.base import AlgoHashRateUnitType
|
||||
from .unit.base import AlgoHashRateUnitType, GenericUnit
|
||||
|
||||
|
||||
class AlgoHashRateType(BaseModel, ABC):
|
||||
unit: AlgoHashRateUnitType
|
||||
rate: float
|
||||
|
||||
@field_serializer("unit")
|
||||
def serialize_unit(self, unit: AlgoHashRateUnitType):
|
||||
return unit.model_dump()
|
||||
|
||||
@abstractmethod
|
||||
def into(self, other: "AlgoHashRateUnitType"):
|
||||
pass
|
||||
|
||||
def auto_unit(self):
|
||||
if 1 < self.rate // int(self.unit.H) < 1000:
|
||||
return self.into(self.unit.H)
|
||||
if 1 < self.rate // int(self.unit.MH) < 1000:
|
||||
return self.into(self.unit.MH)
|
||||
if 1 < self.rate // int(self.unit.GH) < 1000:
|
||||
return self.into(self.unit.GH)
|
||||
if 1 < self.rate // int(self.unit.TH) < 1000:
|
||||
return self.into(self.unit.TH)
|
||||
if 1 < self.rate // int(self.unit.PH) < 1000:
|
||||
return self.into(self.unit.PH)
|
||||
if 1 < self.rate // int(self.unit.EH) < 1000:
|
||||
return self.into(self.unit.EH)
|
||||
if 1 < self.rate // int(self.unit.ZH) < 1000:
|
||||
return self.into(self.unit.ZH)
|
||||
return self
|
||||
|
||||
def __float__(self):
|
||||
return float(self.rate)
|
||||
|
||||
@@ -27,36 +50,46 @@ class AlgoHashRateType(BaseModel, ABC):
|
||||
return round(self.rate, n)
|
||||
|
||||
def __add__(self, other: Self | int | float) -> Self:
|
||||
if isinstance(other, self.__class__):
|
||||
if isinstance(other, AlgoHashRateType):
|
||||
return self.__class__(
|
||||
rate=self.rate + other.into(self.unit).rate, unit=self.unit
|
||||
)
|
||||
return self.__class__(rate=self.rate + other, unit=self.unit)
|
||||
|
||||
def __sub__(self, other: Self | int | float) -> Self:
|
||||
if isinstance(other, self.__class__):
|
||||
if isinstance(other, AlgoHashRateType):
|
||||
return self.__class__(
|
||||
rate=self.rate - other.into(self.unit).rate, unit=self.unit
|
||||
)
|
||||
return self.__class__(rate=self.rate - other, unit=self.unit)
|
||||
|
||||
def __truediv__(self, other: Self | int | float) -> Self:
|
||||
if isinstance(other, self.__class__):
|
||||
if isinstance(other, AlgoHashRateType):
|
||||
return self.__class__(
|
||||
rate=self.rate / other.into(self.unit).rate, unit=self.unit
|
||||
)
|
||||
return self.__class__(rate=self.rate / other, unit=self.unit)
|
||||
|
||||
def __floordiv__(self, other: Self | int | float) -> Self:
|
||||
if isinstance(other, self.__class__):
|
||||
if isinstance(other, AlgoHashRateType):
|
||||
return self.__class__(
|
||||
rate=self.rate // other.into(self.unit).rate, unit=self.unit
|
||||
)
|
||||
return self.__class__(rate=self.rate // other, unit=self.unit)
|
||||
|
||||
def __mul__(self, other: Self | int | float) -> Self:
|
||||
if isinstance(other, self.__class__):
|
||||
if isinstance(other, AlgoHashRateType):
|
||||
return self.__class__(
|
||||
rate=self.rate * other.into(self.unit).rate, unit=self.unit
|
||||
)
|
||||
return self.__class__(rate=self.rate * other, unit=self.unit)
|
||||
|
||||
|
||||
class GenericHashrate(AlgoHashRateType):
|
||||
rate: float = 0
|
||||
unit: GenericUnit = GenericUnit.H
|
||||
|
||||
def into(self, other: GenericUnit):
|
||||
return self.__class__(
|
||||
rate=self.rate / (other.value / self.unit.value), unit=other
|
||||
)
|
||||
|
||||
@@ -53,3 +53,19 @@ class AlgoHashRateUnitType(IntEnum):
|
||||
|
||||
def __repr__(self):
|
||||
return str(self)
|
||||
|
||||
def model_dump(self):
|
||||
return {"value": self.value, "suffix": str(self)}
|
||||
|
||||
|
||||
class GenericUnit(AlgoHashRateUnitType):
|
||||
H = 1
|
||||
KH = int(H) * 1000
|
||||
MH = int(KH) * 1000
|
||||
GH = int(MH) * 1000
|
||||
TH = int(GH) * 1000
|
||||
PH = int(TH) * 1000
|
||||
EH = int(PH) * 1000
|
||||
ZH = int(EH) * 1000
|
||||
|
||||
default = H
|
||||
|
||||
@@ -22,6 +22,7 @@ from pyasic.miners.device.models import (
|
||||
S19aPro,
|
||||
S19j,
|
||||
S19jPro,
|
||||
S19kPro,
|
||||
S19NoPIC,
|
||||
S19Pro,
|
||||
S19ProHydro,
|
||||
@@ -62,3 +63,7 @@ class VNishS19jPro(VNish, S19jPro):
|
||||
|
||||
class VNishS19ProHydro(VNish, S19ProHydro):
|
||||
pass
|
||||
|
||||
|
||||
class VNishS19kPro(VNish, S19kPro):
|
||||
pass
|
||||
|
||||
@@ -20,6 +20,7 @@ from .S19 import (
|
||||
VNishS19aPro,
|
||||
VNishS19j,
|
||||
VNishS19jPro,
|
||||
VNishS19kPro,
|
||||
VNishS19NoPIC,
|
||||
VNishS19Pro,
|
||||
VNishS19ProHydro,
|
||||
|
||||
@@ -14,6 +14,7 @@
|
||||
# limitations under the License. -
|
||||
# ------------------------------------------------------------------------------
|
||||
|
||||
import logging
|
||||
from typing import Optional
|
||||
|
||||
from pyasic import MinerConfig
|
||||
@@ -96,6 +97,7 @@ class VNish(VNishFirmware, BMMiner):
|
||||
|
||||
supports_shutdown = True
|
||||
supports_presets = True
|
||||
supports_autotuning = True
|
||||
|
||||
data_locations = VNISH_DATA_LOC
|
||||
|
||||
@@ -272,3 +274,27 @@ class VNish(VNishFirmware, BMMiner):
|
||||
return self.config
|
||||
self.config = MinerConfig.from_vnish(web_settings, web_presets)
|
||||
return self.config
|
||||
|
||||
async def set_power_limit(self, wattage: int) -> bool:
|
||||
config = await self.get_config()
|
||||
valid_presets = [
|
||||
preset.power
|
||||
for preset in config.mining_mode.available_presets
|
||||
if preset.tuned and preset.power <= wattage
|
||||
]
|
||||
new_wattage = max(valid_presets)
|
||||
|
||||
# Set power to highest preset <= wattage
|
||||
try:
|
||||
await self.web.set_power_limit(new_wattage)
|
||||
updated_settings = await self.web.settings()
|
||||
except APIError:
|
||||
raise
|
||||
except Exception as e:
|
||||
logging.warning(f"{self} - Failed to set power limit: {e}")
|
||||
return False
|
||||
|
||||
if int(updated_settings["miner"]["overclock"]["preset"]) == new_wattage:
|
||||
return True
|
||||
else:
|
||||
return False
|
||||
|
||||
@@ -407,6 +407,7 @@ MINER_CLASSES = {
|
||||
"ANTMINER S19A": VNishS19a,
|
||||
"ANTMINER S19A PRO": VNishS19aPro,
|
||||
"ANTMINER S19 PRO HYD.": VNishS19ProHydro,
|
||||
"ANTMINER S19K PRO": VNishS19kPro,
|
||||
"ANTMINER T19": VNishT19,
|
||||
"ANTMINER S21": VNishS21,
|
||||
},
|
||||
|
||||
@@ -138,6 +138,15 @@ class VNishWebAPI(BaseWebAPI):
|
||||
async def settings(self) -> dict:
|
||||
return await self.send_command("settings")
|
||||
|
||||
async def set_power_limit(self, wattage: int) -> bool:
|
||||
# Can only set power limit to tuned preset
|
||||
settings = await self.settings()
|
||||
settings["miner"]["overclock"]["preset"] = str(wattage)
|
||||
new_settings = {"miner": {"overclock": settings["miner"]["overclock"]}}
|
||||
|
||||
# response will always be {"restart_required":false,"reboot_required":false} even if unsuccessful
|
||||
return await self.send_command("settings", privileged=True, **new_settings)
|
||||
|
||||
async def autotune_presets(self) -> dict:
|
||||
return await self.send_command("autotune/presets")
|
||||
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
[tool.poetry]
|
||||
name = "pyasic"
|
||||
version = "0.65.7"
|
||||
version = "0.67.0"
|
||||
description = "A simplified and standardized interface for Bitcoin ASICs."
|
||||
authors = ["UpstreamData <brett@upstreamdata.ca>"]
|
||||
repository = "https://github.com/UpstreamData/pyasic"
|
||||
|
||||
Reference in New Issue
Block a user