diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml index f6182fc..c985d53 100644 --- a/.github/workflows/release.yml +++ b/.github/workflows/release.yml @@ -57,14 +57,14 @@ jobs: - name: Publish package on PyPI if: steps.check-version.outputs.tag - uses: pypa/gh-action-pypi-publish@v1.8.5 + uses: pypa/gh-action-pypi-publish@v1.8.6 with: user: __token__ password: ${{ secrets.PYPI_TOKEN }} - name: Publish package on TestPyPI if: "! steps.check-version.outputs.tag" - uses: pypa/gh-action-pypi-publish@v1.8.5 + uses: pypa/gh-action-pypi-publish@v1.8.6 with: user: __token__ password: ${{ secrets.TEST_PYPI_TOKEN }} diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index 2a9fed4..79420eb 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -154,4 +154,4 @@ jobs: nox --session=coverage -- xml - name: Upload coverage report - uses: codecov/codecov-action@v3.1.3 + uses: codecov/codecov-action@v3.1.4 diff --git a/arm.yml b/arm.yml index 12c4968..baf6b87 100644 --- a/arm.yml +++ b/arm.yml @@ -22,7 +22,7 @@ services: - "dev" hostname: rabbit container_name: atn-dev-rabbit - image: "jessmillar/dev-rabbit-arm:chaos__e58daf6__20230308" + image: "jessmillar/dev-rabbit-arm:chaos__53ea3a0__20230622" ports: - 1885:1885 - 4369:4369 diff --git a/docs/apis/json/atn-params-brickstorageheater.json b/docs/apis/json/atn-params-brickstorageheater.json index 31e1c2d..2dd582f 100644 --- a/docs/apis/json/atn-params-brickstorageheater.json +++ b/docs/apis/json/atn-params-brickstorageheater.json @@ -25,13 +25,19 @@ }, { "const": "2127aba6", - "title": "VersantStorageHeatTariff", - "description": "" + "title": "VersantA1StorageHeatTariff", + "url": "https://github.com/thegridelectric/gridworks-ps/blob/dev/input_data/electricity_prices/isone/distp__w.isone.stetson__2022__gw.me.versant.a1.res.ets.csv", + "description": "Versant is a utility serving customers in Maine, and in particular serves much of the area behind the Keene Rd Constraint in the [GridWorks Millinocket Demo](https://gridworks.readthedocs.io/en/latest/millinocket-demo.html#background). Alternately known as the 'Home Eco Rate With Bonus Meter, Time-of-Use.' Look for rate A1 in Versant [rate schedules](https://www.versantpower.com/residential/rates/rates-schedules/); details are also available [here](https://drive.google.com/drive/u/0/folders/1mhIeNj2JWVyIJrQnSHmBDOkBpNnRRVKB). More: Service under this rate will be available to residential customers with thermal energy storage devices, electric battery storage devices, and/or vehicle chargers who agree to install a second metered point of delivery. The customer will be subject to inspections to ensure that the thermal storage device, electric battery storage device, and electric vehicle charger(s) are sized appropriately for residential use. If the thermal storage device, electric battery storage device, and electric vehicle charger(s) do not pass Company inspection, then the service will be denied. Service will be single-phase, alternating current, 60 hertz, at one standard secondary distribution voltage. Customers taking service under this rate schedule are responsible for paying both Distribution Service and Stranded Cost." }, { "const": "ea5c675a", "title": "VersantATariff", - "description": "" + "description": "Versant is a utility serving customers in Maine, and in particular serves much of the area behind the Keene Rd Constraint in the [GridWorks Millinocket Demo](https://gridworks.readthedocs.io/en/latest/millinocket-demo.html#background). The A Tariff is their standard residential tariff. Look for rate A in Versant [rate schedules](https://www.versantpower.com/residential/rates/rates-schedules/)" + }, + { + "const": "54aec3a7", + "title": "VersantA20HeatTariff", + "description": "Versant is a utility serving customers in Maine, and in particular serves much of the area behind the Keene Rd Constraint in the [GridWorks Millinocket Demo](https://gridworks.readthedocs.io/en/latest/millinocket-demo.html#background). This is an alternative tariff available for electric heat." } ] }, diff --git a/docs/apis/json/atn-params-simpleresistivehydronic.json b/docs/apis/json/atn-params-simpleresistivehydronic.json new file mode 100644 index 0000000..416e516 --- /dev/null +++ b/docs/apis/json/atn-params-simpleresistivehydronic.json @@ -0,0 +1,210 @@ +{ + "gwapi": "001", + "type_name": "atn.params.simpleresistivehydronic", + "version": "000", + "owner": "gridworks@gridworks-consulting.com", + "description": "", + "formats": { + "LeftRightDot": { + "type": "string", + "description": "Lowercase alphanumeric words separated by periods, most significant word (on the left) starting with an alphabet character.", + "example": "dw1.isone.me.freedom.apple" + } + }, + "enums": { + "DistributionTariff000": { + "type": "string", + "name": "distribution.tariff.000", + "description": "Name of distribution tariff of local network company/utility", + "oneOf": [ + { + "const": "00000000", + "title": "Unknown", + "description": "" + }, + { + "const": "2127aba6", + "title": "VersantA1StorageHeatTariff", + "url": "https://github.com/thegridelectric/gridworks-ps/blob/dev/input_data/electricity_prices/isone/distp__w.isone.stetson__2022__gw.me.versant.a1.res.ets.csv", + "description": "Versant is a utility serving customers in Maine, and in particular serves much of the area behind the Keene Rd Constraint in the [GridWorks Millinocket Demo](https://gridworks.readthedocs.io/en/latest/millinocket-demo.html#background). Alternately known as the 'Home Eco Rate With Bonus Meter, Time-of-Use.' Look for rate A1 in Versant [rate schedules](https://www.versantpower.com/residential/rates/rates-schedules/); details are also available [here](https://drive.google.com/drive/u/0/folders/1mhIeNj2JWVyIJrQnSHmBDOkBpNnRRVKB). More: Service under this rate will be available to residential customers with thermal energy storage devices, electric battery storage devices, and/or vehicle chargers who agree to install a second metered point of delivery. The customer will be subject to inspections to ensure that the thermal storage device, electric battery storage device, and electric vehicle charger(s) are sized appropriately for residential use. If the thermal storage device, electric battery storage device, and electric vehicle charger(s) do not pass Company inspection, then the service will be denied. Service will be single-phase, alternating current, 60 hertz, at one standard secondary distribution voltage. Customers taking service under this rate schedule are responsible for paying both Distribution Service and Stranded Cost." + }, + { + "const": "ea5c675a", + "title": "VersantATariff", + "description": "Versant is a utility serving customers in Maine, and in particular serves much of the area behind the Keene Rd Constraint in the [GridWorks Millinocket Demo](https://gridworks.readthedocs.io/en/latest/millinocket-demo.html#background). The A Tariff is their standard residential tariff. Look for rate A in Versant [rate schedules](https://www.versantpower.com/residential/rates/rates-schedules/)" + }, + { + "const": "54aec3a7", + "title": "VersantA20HeatTariff", + "description": "Versant is a utility serving customers in Maine, and in particular serves much of the area behind the Keene Rd Constraint in the [GridWorks Millinocket Demo](https://gridworks.readthedocs.io/en/latest/millinocket-demo.html#background). This is an alternative tariff available for electric heat." + } + ] + }, + "RecognizedCurrencyUnit000": { + "type": "string", + "name": "recognized.currency.unit.000", + "description": "Unit of currency", + "oneOf": [ + { + "const": "00000000", + "title": "Unknown", + "description": "" + }, + { + "const": "e57c5143", + "title": "USD", + "description": "US Dollar" + }, + { + "const": "f7b38fc5", + "title": "GBP", + "description": "Pounds sterling" + } + ] + }, + "EnergySupplyType000": { + "type": "string", + "name": "energy.supply.type.000", + "description": "", + "oneOf": [ + { + "const": "00000000", + "title": "Unknown", + "description": "" + }, + { + "const": "cb18f937", + "title": "StandardOffer", + "description": "" + }, + { + "const": "e9dc99a6", + "title": "RealtimeLocalLmp", + "description": "" + } + ] + } + }, + "properties": { + "GNodeAlias": { + "type": "string", + "format": "LeftRightDot", + "title": "", + "required": true + }, + "HomeCity": { + "type": "string", + "title": "", + "required": true + }, + "TimezoneString": { + "type": "string", + "title": "", + "required": true + }, + "StorageSteps": { + "type": "integer", + "title": "", + "required": true + }, + "FloSlices": { + "type": "integer", + "title": "", + "required": true + }, + "SliceDurationMinutes": { + "type": "integer", + "title": "", + "required": true + }, + "CurrencyUnit": { + "type": "string", + "format": "RecognizedCurrencyUnit000", + "title": "", + "required": true + }, + "Tariff": { + "type": "string", + "format": "DistributionTariff000", + "title": "", + "required": true + }, + "EnergyType": { + "type": "string", + "format": "EnergySupplyType000", + "title": "", + "required": true + }, + "StandardOfferPriceDollarsPerMwh": { + "type": "integer", + "title": "", + "required": true + }, + "DistributionTariffDollarsPerMwh": { + "type": "integer", + "title": "", + "required": true + }, + "StoreSizeGallons": { + "type": "integer", + "title": "", + "required": true + }, + "MaxStoreTempF": { + "type": "integer", + "title": "", + "required": true + }, + "ElementMaxPowerKw": { + "type": "number", + "title": "", + "required": true + }, + "RequiredSourceWaterTempF": { + "type": "integer", + "title": "", + "required": true + }, + "FixedPumpGpm": { + "type": "number", + "title": "", + "required": true + }, + "ReturnWaterFixedDeltaT": { + "type": "integer", + "title": "", + "required": true + }, + "AnnualHvacKwhTh": { + "type": "integer", + "title": "", + "required": true + }, + "AmbientPowerInKw": { + "type": "number", + "title": "", + "required": true + }, + "HouseWorstCaseTempF": { + "type": "integer", + "title": "", + "required": true + }, + "RoomTempF": { + "type": "integer", + "title": "", + "required": true + }, + "TypeName": { + "type": "string", + "value": "atn.params.simpleresistivehydronic.000", + "title": "The type name" + }, + "Version": { + "type": "string", + "title": "The type version", + "default": "000", + "required": true + } + } +} diff --git a/docs/apis/json/flo-params-brickstorageheater.json b/docs/apis/json/flo-params-brickstorageheater.json index c159c83..1787b3e 100644 --- a/docs/apis/json/flo-params-brickstorageheater.json +++ b/docs/apis/json/flo-params-brickstorageheater.json @@ -129,7 +129,7 @@ "C": { "type": "number", "title": "", - "required": false + "required": true }, "RealtimeElectricityPrice": { "type": "number", @@ -167,7 +167,7 @@ "type": "string", "format": "UuidCanonicalTextual", "title": "", - "required": false + "required": true }, "RegPriceUid": { "type": "string", diff --git a/docs/apis/json/flo-params-simpleresistivehydronic.json b/docs/apis/json/flo-params-simpleresistivehydronic.json new file mode 100644 index 0000000..effd61d --- /dev/null +++ b/docs/apis/json/flo-params-simpleresistivehydronic.json @@ -0,0 +1,182 @@ +{ + "gwapi": "001", + "type_name": "flo.params.simpleresistivehydronic", + "version": "000", + "owner": "gridworks@gridworks-consulting.com", + "description": "", + "formats": { + "UuidCanonicalTextual": { + "type": "string", + "description": "A string of hex words separated by hyphens of length 8-4-4-4-12.", + "example": "652ba6b0-c3bf-4f06-8a80-6b9832d60a25" + }, + "LeftRightDot": { + "type": "string", + "description": "Lowercase alphanumeric words separated by periods, most significant word (on the left) starting with an alphabet character.", + "example": "dw1.isone.me.freedom.apple" + } + }, + "enums": { + "RecognizedCurrencyUnit000": { + "type": "string", + "name": "recognized.currency.unit.000", + "description": "Unit of currency", + "oneOf": [ + { + "const": "00000000", + "title": "Unknown", + "description": "" + }, + { + "const": "e57c5143", + "title": "USD", + "description": "US Dollar" + }, + { + "const": "f7b38fc5", + "title": "GBP", + "description": "Pounds sterling" + } + ] + } + }, + "properties": { + "GNodeAlias": { + "type": "string", + "format": "LeftRightDot", + "title": "", + "required": true + }, + "FloParamsUid": { + "type": "string", + "format": "UuidCanonicalTextual", + "title": "", + "required": true + }, + "HomeCity": { + "type": "string", + "title": "", + "required": true + }, + "TimezoneString": { + "type": "string", + "title": "", + "required": true + }, + "StartYearUtc": { + "type": "integer", + "title": "", + "required": true + }, + "StartMonthUtc": { + "type": "integer", + "title": "", + "required": true + }, + "StartDayUtc": { + "type": "integer", + "title": "", + "required": true + }, + "StartHourUtc": { + "type": "integer", + "title": "", + "required": true + }, + "StartMinuteUtc": { + "type": "integer", + "title": "", + "required": true + }, + "StoreSizeGallons": { + "type": "integer", + "title": "", + "required": true + }, + "MaxStoreTempF": { + "type": "integer", + "title": "", + "required": true + }, + "ElementMaxPowerKw": { + "type": "number", + "title": "", + "required": true + }, + "RequiredSourceWaterTempF": { + "type": "integer", + "title": "", + "required": true + }, + "FixedPumpGpm": { + "type": "number", + "title": "", + "required": true + }, + "ReturnWaterFixedDeltaT": { + "type": "integer", + "title": "", + "required": true + }, + "SliceDurationMinutes": { + "type": "integer", + "title": "", + "required": true + }, + "PowerLostFromHouseKwList": { + "type": "number", + "title": "", + "required": true + }, + "OutsideTempF": { + "type": "number", + "title": "", + "required": true + }, + "DistributionPrice": { + "type": "number", + "title": "", + "required": true + }, + "RealtimeElectricityPrice": { + "type": "number", + "title": "", + "required": true + }, + "RtElecPriceUid": { + "type": "string", + "format": "UuidCanonicalTextual", + "title": "", + "required": true + }, + "WeatherUid": { + "type": "string", + "format": "UuidCanonicalTextual", + "title": "", + "required": true + }, + "DistPriceUid": { + "type": "string", + "format": "UuidCanonicalTextual", + "title": "", + "required": true + }, + "CurrencyUnit": { + "type": "string", + "format": "RecognizedCurrencyUnit000", + "title": "", + "required": true + }, + "TypeName": { + "type": "string", + "value": "flo.params.simpleresistivehydronic.000", + "title": "The type name" + }, + "Version": { + "type": "string", + "title": "The type version", + "default": "000", + "required": true + } + } +} diff --git a/docs/apis/json/simplesim-driver-report.json b/docs/apis/json/simplesim-driver-report.json index a040be3..1a19ba4 100644 --- a/docs/apis/json/simplesim-driver-report.json +++ b/docs/apis/json/simplesim-driver-report.json @@ -23,7 +23,7 @@ "title": "", "required": true }, - "FromGNodeInstanecId": { + "FromGNodeInstanceId": { "type": "string", "format": "UuidCanonicalTextual", "title": "", diff --git a/docs/apis/types.rst b/docs/apis/types.rst index e51f58d..688956f 100644 --- a/docs/apis/types.rst +++ b/docs/apis/types.rst @@ -22,6 +22,10 @@ AtnParamsReport ^^^^^^^^^^^^^^^^^^^^^^^^^^^^ .. literalinclude:: json/atn-params-report.json +AtnParamsSimpleresistivehydronic +^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +.. literalinclude:: json/atn-params-simpleresistivehydronic.json + BasegnodeScadaCreate ^^^^^^^^^^^^^^^^^^^^^^^^^^^^ .. literalinclude:: json/basegnode-scada-create.json @@ -46,6 +50,10 @@ FloParamsReport ^^^^^^^^^^^^^^^^^^^^^^^^^^^^ .. literalinclude:: json/flo-params-report.json +FloParamsSimpleresistivehydronic +^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +.. literalinclude:: json/flo-params-simpleresistivehydronic.json + InitialTadeedAlgoCreate ^^^^^^^^^^^^^^^^^^^^^^^^^^^^ .. literalinclude:: json/initial-tadeed-algo-create.json diff --git a/docs/requirements.txt b/docs/requirements.txt index e6c0441..5f54dc2 100644 --- a/docs/requirements.txt +++ b/docs/requirements.txt @@ -1,4 +1,4 @@ -furo==2023.3.27 +furo==2023.5.20 sphinx==7.0.0 sphinx-click==4.4.0 myst_parser==1.0.0 diff --git a/docs/sdk-types.rst b/docs/sdk-types.rst index 804894e..a90d86c 100644 --- a/docs/sdk-types.rst +++ b/docs/sdk-types.rst @@ -23,12 +23,14 @@ forth between type instances and Python objects. AtnParams AtnParamsBrickstorageheater AtnParamsReport + AtnParamsSimpleresistivehydronic BasegnodeScadaCreate DiscoverycertAlgoCreate DispatchContractConfirmed FloParams FloParamsBrickstorageheater FloParamsReport + FloParamsSimpleresistivehydronic InitialTadeedAlgoCreate InitialTadeedAlgoOptin InitialTadeedAlgoTransfer diff --git a/docs/types/atn-params-simpleresistivehydronic.rst b/docs/types/atn-params-simpleresistivehydronic.rst new file mode 100644 index 0000000..1bc6cf8 --- /dev/null +++ b/docs/types/atn-params-simpleresistivehydronic.rst @@ -0,0 +1,77 @@ +AtnParamsSimpleresistivehydronic +========================== +Python pydantic class corresponding to json type ```atn.params.simpleresistivehydronic```. + +.. autoclass:: gwatn.types.AtnParamsSimpleresistivehydronic + :members: + +**GNodeAlias**: + - Description: + - Format: LeftRightDot + +**HomeCity**: + - Description: + +**TimezoneString**: + - Description: + +**StorageSteps**: + - Description: + +**FloSlices**: + - Description: + +**SliceDurationMinutes**: + - Description: + +**CurrencyUnit**: + - Description: + +**Tariff**: + - Description: + +**EnergyType**: + - Description: + +**StandardOfferPriceDollarsPerMwh**: + - Description: + +**DistributionTariffDollarsPerMwh**: + - Description: + +**StoreSizeGallons**: + - Description: + +**MaxStoreTempF**: + - Description: + +**ElementMaxPowerKw**: + - Description: + +**RequiredSourceWaterTempF**: + - Description: + +**FixedPumpGpm**: + - Description: + +**ReturnWaterFixedDeltaT**: + - Description: + +**AnnualHvacKwhTh**: + - Description: + +**AmbientPowerInKw**: + - Description: + +**HouseWorstCaseTempF**: + - Description: + +**RoomTempF**: + - Description: + +.. autoclass:: gwatn.types.atn_params_simpleresistivehydronic.check_is_left_right_dot + :members: + + +.. autoclass:: gwatn.types.AtnParamsSimpleresistivehydronic_Maker + :members: diff --git a/docs/types/flo-params-simpleresistivehydronic.rst b/docs/types/flo-params-simpleresistivehydronic.rst new file mode 100644 index 0000000..2d755df --- /dev/null +++ b/docs/types/flo-params-simpleresistivehydronic.rst @@ -0,0 +1,94 @@ +FloParamsSimpleresistivehydronic +========================== +Python pydantic class corresponding to json type ```flo.params.simpleresistivehydronic```. + +.. autoclass:: gwatn.types.FloParamsSimpleresistivehydronic + :members: + +**GNodeAlias**: + - Description: + - Format: LeftRightDot + +**FloParamsUid**: + - Description: + - Format: UuidCanonicalTextual + +**HomeCity**: + - Description: + +**TimezoneString**: + - Description: + +**StartYearUtc**: + - Description: + +**StartMonthUtc**: + - Description: + +**StartDayUtc**: + - Description: + +**StartHourUtc**: + - Description: + +**StartMinuteUtc**: + - Description: + +**StoreSizeGallons**: + - Description: + +**MaxStoreTempF**: + - Description: + +**ElementMaxPowerKw**: + - Description: + +**RequiredSourceWaterTempF**: + - Description: + +**FixedPumpGpm**: + - Description: + +**ReturnWaterFixedDeltaT**: + - Description: + +**SliceDurationMinutes**: + - Description: + +**PowerLostFromHouseKwList**: + - Description: + +**OutsideTempF**: + - Description: + +**DistributionPrice**: + - Description: + +**RealtimeElectricityPrice**: + - Description: + +**RtElecPriceUid**: + - Description: + - Format: UuidCanonicalTextual + +**WeatherUid**: + - Description: + - Format: UuidCanonicalTextual + +**DistPriceUid**: + - Description: + - Format: UuidCanonicalTextual + +**CurrencyUnit**: + - Description: + +.. autoclass:: gwatn.types.flo_params_simpleresistivehydronic.check_is_uuid_canonical_textual + :members: + + +.. autoclass:: gwatn.types.flo_params_simpleresistivehydronic.check_is_left_right_dot + :members: + + +.. autoclass:: gwatn.types.FloParamsSimpleresistivehydronic_Maker + :members: diff --git a/docs/types/simplesim-driver-report.rst b/docs/types/simplesim-driver-report.rst index 6ba5f83..158ddbd 100644 --- a/docs/types/simplesim-driver-report.rst +++ b/docs/types/simplesim-driver-report.rst @@ -9,7 +9,7 @@ Python pydantic class corresponding to json type ```simplesim.driver.report```. - Description: - Format: LeftRightDot -**FromGNodeInstanecId**: +**FromGNodeInstanceId**: - Description: - Format: UuidCanonicalTextual diff --git a/poetry.lock b/poetry.lock index 4a1817e..25730fc 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,4 +1,4 @@ -# This file is automatically @generated by Poetry and should not be changed by hand. +# This file is automatically @generated by Poetry 1.4.1 and should not be changed by hand. [[package]] name = "aiocache" @@ -541,63 +541,72 @@ test = ["flake8 (==3.7.8)", "hypothesis (==3.55.3)"] [[package]] name = "coverage" -version = "7.2.5" +version = "7.2.7" description = "Code coverage measurement for Python" category = "dev" optional = false python-versions = ">=3.7" files = [ - {file = "coverage-7.2.5-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:883123d0bbe1c136f76b56276074b0c79b5817dd4238097ffa64ac67257f4b6c"}, - {file = "coverage-7.2.5-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:d2fbc2a127e857d2f8898aaabcc34c37771bf78a4d5e17d3e1f5c30cd0cbc62a"}, - {file = "coverage-7.2.5-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5f3671662dc4b422b15776cdca89c041a6349b4864a43aa2350b6b0b03bbcc7f"}, - {file = "coverage-7.2.5-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:780551e47d62095e088f251f5db428473c26db7829884323e56d9c0c3118791a"}, - {file = "coverage-7.2.5-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:066b44897c493e0dcbc9e6a6d9f8bbb6607ef82367cf6810d387c09f0cd4fe9a"}, - {file = "coverage-7.2.5-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:b9a4ee55174b04f6af539218f9f8083140f61a46eabcaa4234f3c2a452c4ed11"}, - {file = "coverage-7.2.5-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:706ec567267c96717ab9363904d846ec009a48d5f832140b6ad08aad3791b1f5"}, - {file = "coverage-7.2.5-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:ae453f655640157d76209f42c62c64c4d4f2c7f97256d3567e3b439bd5c9b06c"}, - {file = "coverage-7.2.5-cp310-cp310-win32.whl", hash = "sha256:f81c9b4bd8aa747d417407a7f6f0b1469a43b36a85748145e144ac4e8d303cb5"}, - {file = "coverage-7.2.5-cp310-cp310-win_amd64.whl", hash = "sha256:dc945064a8783b86fcce9a0a705abd7db2117d95e340df8a4333f00be5efb64c"}, - {file = "coverage-7.2.5-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:40cc0f91c6cde033da493227797be2826cbf8f388eaa36a0271a97a332bfd7ce"}, - {file = "coverage-7.2.5-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:a66e055254a26c82aead7ff420d9fa8dc2da10c82679ea850d8feebf11074d88"}, - {file = "coverage-7.2.5-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c10fbc8a64aa0f3ed136b0b086b6b577bc64d67d5581acd7cc129af52654384e"}, - {file = "coverage-7.2.5-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:9a22cbb5ede6fade0482111fa7f01115ff04039795d7092ed0db43522431b4f2"}, - {file = "coverage-7.2.5-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:292300f76440651529b8ceec283a9370532f4ecba9ad67d120617021bb5ef139"}, - {file = "coverage-7.2.5-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:7ff8f3fb38233035028dbc93715551d81eadc110199e14bbbfa01c5c4a43f8d8"}, - {file = "coverage-7.2.5-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:a08c7401d0b24e8c2982f4e307124b671c6736d40d1c39e09d7a8687bddf83ed"}, - {file = "coverage-7.2.5-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:ef9659d1cda9ce9ac9585c045aaa1e59223b143f2407db0eaee0b61a4f266fb6"}, - {file = "coverage-7.2.5-cp311-cp311-win32.whl", hash = "sha256:30dcaf05adfa69c2a7b9f7dfd9f60bc8e36b282d7ed25c308ef9e114de7fc23b"}, - {file = "coverage-7.2.5-cp311-cp311-win_amd64.whl", hash = "sha256:97072cc90f1009386c8a5b7de9d4fc1a9f91ba5ef2146c55c1f005e7b5c5e068"}, - {file = "coverage-7.2.5-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:bebea5f5ed41f618797ce3ffb4606c64a5de92e9c3f26d26c2e0aae292f015c1"}, - {file = "coverage-7.2.5-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:828189fcdda99aae0d6bf718ea766b2e715eabc1868670a0a07bf8404bf58c33"}, - {file = "coverage-7.2.5-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6e8a95f243d01ba572341c52f89f3acb98a3b6d1d5d830efba86033dd3687ade"}, - {file = "coverage-7.2.5-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e8834e5f17d89e05697c3c043d3e58a8b19682bf365048837383abfe39adaed5"}, - {file = "coverage-7.2.5-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:d1f25ee9de21a39b3a8516f2c5feb8de248f17da7eead089c2e04aa097936b47"}, - {file = "coverage-7.2.5-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:1637253b11a18f453e34013c665d8bf15904c9e3c44fbda34c643fbdc9d452cd"}, - {file = "coverage-7.2.5-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:8e575a59315a91ccd00c7757127f6b2488c2f914096077c745c2f1ba5b8c0969"}, - {file = "coverage-7.2.5-cp37-cp37m-win32.whl", hash = "sha256:509ecd8334c380000d259dc66feb191dd0a93b21f2453faa75f7f9cdcefc0718"}, - {file = "coverage-7.2.5-cp37-cp37m-win_amd64.whl", hash = "sha256:12580845917b1e59f8a1c2ffa6af6d0908cb39220f3019e36c110c943dc875b0"}, - {file = "coverage-7.2.5-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:b5016e331b75310610c2cf955d9f58a9749943ed5f7b8cfc0bb89c6134ab0a84"}, - {file = "coverage-7.2.5-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:373ea34dca98f2fdb3e5cb33d83b6d801007a8074f992b80311fc589d3e6b790"}, - {file = "coverage-7.2.5-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a063aad9f7b4c9f9da7b2550eae0a582ffc7623dca1c925e50c3fbde7a579771"}, - {file = "coverage-7.2.5-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:38c0a497a000d50491055805313ed83ddba069353d102ece8aef5d11b5faf045"}, - {file = "coverage-7.2.5-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a2b3b05e22a77bb0ae1a3125126a4e08535961c946b62f30985535ed40e26614"}, - {file = "coverage-7.2.5-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:0342a28617e63ad15d96dca0f7ae9479a37b7d8a295f749c14f3436ea59fdcb3"}, - {file = "coverage-7.2.5-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:cf97ed82ca986e5c637ea286ba2793c85325b30f869bf64d3009ccc1a31ae3fd"}, - {file = "coverage-7.2.5-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:c2c41c1b1866b670573657d584de413df701f482574bad7e28214a2362cb1fd1"}, - {file = "coverage-7.2.5-cp38-cp38-win32.whl", hash = "sha256:10b15394c13544fce02382360cab54e51a9e0fd1bd61ae9ce012c0d1e103c813"}, - {file = "coverage-7.2.5-cp38-cp38-win_amd64.whl", hash = "sha256:a0b273fe6dc655b110e8dc89b8ec7f1a778d78c9fd9b4bda7c384c8906072212"}, - {file = "coverage-7.2.5-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:5c587f52c81211d4530fa6857884d37f514bcf9453bdeee0ff93eaaf906a5c1b"}, - {file = "coverage-7.2.5-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:4436cc9ba5414c2c998eaedee5343f49c02ca93b21769c5fdfa4f9d799e84200"}, - {file = "coverage-7.2.5-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6599bf92f33ab041e36e06d25890afbdf12078aacfe1f1d08c713906e49a3fe5"}, - {file = "coverage-7.2.5-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:857abe2fa6a4973f8663e039ead8d22215d31db613ace76e4a98f52ec919068e"}, - {file = "coverage-7.2.5-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f6f5cab2d7f0c12f8187a376cc6582c477d2df91d63f75341307fcdcb5d60303"}, - {file = "coverage-7.2.5-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:aa387bd7489f3e1787ff82068b295bcaafbf6f79c3dad3cbc82ef88ce3f48ad3"}, - {file = "coverage-7.2.5-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:156192e5fd3dbbcb11cd777cc469cf010a294f4c736a2b2c891c77618cb1379a"}, - {file = "coverage-7.2.5-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:bd3b4b8175c1db502adf209d06136c000df4d245105c8839e9d0be71c94aefe1"}, - {file = "coverage-7.2.5-cp39-cp39-win32.whl", hash = "sha256:ddc5a54edb653e9e215f75de377354e2455376f416c4378e1d43b08ec50acc31"}, - {file = "coverage-7.2.5-cp39-cp39-win_amd64.whl", hash = "sha256:338aa9d9883aaaad53695cb14ccdeb36d4060485bb9388446330bef9c361c252"}, - {file = "coverage-7.2.5-pp37.pp38.pp39-none-any.whl", hash = "sha256:8877d9b437b35a85c18e3c6499b23674684bf690f5d96c1006a1ef61f9fdf0f3"}, - {file = "coverage-7.2.5.tar.gz", hash = "sha256:f99ef080288f09ffc687423b8d60978cf3a465d3f404a18d1a05474bd8575a47"}, + {file = "coverage-7.2.7-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:d39b5b4f2a66ccae8b7263ac3c8170994b65266797fb96cbbfd3fb5b23921db8"}, + {file = "coverage-7.2.7-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:6d040ef7c9859bb11dfeb056ff5b3872436e3b5e401817d87a31e1750b9ae2fb"}, + {file = "coverage-7.2.7-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ba90a9563ba44a72fda2e85302c3abc71c5589cea608ca16c22b9804262aaeb6"}, + {file = "coverage-7.2.7-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e7d9405291c6928619403db1d10bd07888888ec1abcbd9748fdaa971d7d661b2"}, + {file = "coverage-7.2.7-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:31563e97dae5598556600466ad9beea39fb04e0229e61c12eaa206e0aa202063"}, + {file = "coverage-7.2.7-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:ebba1cd308ef115925421d3e6a586e655ca5a77b5bf41e02eb0e4562a111f2d1"}, + {file = "coverage-7.2.7-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:cb017fd1b2603ef59e374ba2063f593abe0fc45f2ad9abdde5b4d83bd922a353"}, + {file = "coverage-7.2.7-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:d62a5c7dad11015c66fbb9d881bc4caa5b12f16292f857842d9d1871595f4495"}, + {file = "coverage-7.2.7-cp310-cp310-win32.whl", hash = "sha256:ee57190f24fba796e36bb6d3aa8a8783c643d8fa9760c89f7a98ab5455fbf818"}, + {file = "coverage-7.2.7-cp310-cp310-win_amd64.whl", hash = "sha256:f75f7168ab25dd93110c8a8117a22450c19976afbc44234cbf71481094c1b850"}, + {file = "coverage-7.2.7-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:06a9a2be0b5b576c3f18f1a241f0473575c4a26021b52b2a85263a00f034d51f"}, + {file = "coverage-7.2.7-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:5baa06420f837184130752b7c5ea0808762083bf3487b5038d68b012e5937dbe"}, + {file = "coverage-7.2.7-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:fdec9e8cbf13a5bf63290fc6013d216a4c7232efb51548594ca3631a7f13c3a3"}, + {file = "coverage-7.2.7-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:52edc1a60c0d34afa421c9c37078817b2e67a392cab17d97283b64c5833f427f"}, + {file = "coverage-7.2.7-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:63426706118b7f5cf6bb6c895dc215d8a418d5952544042c8a2d9fe87fcf09cb"}, + {file = "coverage-7.2.7-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:afb17f84d56068a7c29f5fa37bfd38d5aba69e3304af08ee94da8ed5b0865833"}, + {file = "coverage-7.2.7-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:48c19d2159d433ccc99e729ceae7d5293fbffa0bdb94952d3579983d1c8c9d97"}, + {file = "coverage-7.2.7-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:0e1f928eaf5469c11e886fe0885ad2bf1ec606434e79842a879277895a50942a"}, + {file = "coverage-7.2.7-cp311-cp311-win32.whl", hash = "sha256:33d6d3ea29d5b3a1a632b3c4e4f4ecae24ef170b0b9ee493883f2df10039959a"}, + {file = "coverage-7.2.7-cp311-cp311-win_amd64.whl", hash = "sha256:5b7540161790b2f28143191f5f8ec02fb132660ff175b7747b95dcb77ac26562"}, + {file = "coverage-7.2.7-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:f2f67fe12b22cd130d34d0ef79206061bfb5eda52feb6ce0dba0644e20a03cf4"}, + {file = "coverage-7.2.7-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a342242fe22407f3c17f4b499276a02b01e80f861f1682ad1d95b04018e0c0d4"}, + {file = "coverage-7.2.7-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:171717c7cb6b453aebac9a2ef603699da237f341b38eebfee9be75d27dc38e01"}, + {file = "coverage-7.2.7-cp312-cp312-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:49969a9f7ffa086d973d91cec8d2e31080436ef0fb4a359cae927e742abfaaa6"}, + {file = "coverage-7.2.7-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:b46517c02ccd08092f4fa99f24c3b83d8f92f739b4657b0f146246a0ca6a831d"}, + {file = "coverage-7.2.7-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:a3d33a6b3eae87ceaefa91ffdc130b5e8536182cd6dfdbfc1aa56b46ff8c86de"}, + {file = "coverage-7.2.7-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:976b9c42fb2a43ebf304fa7d4a310e5f16cc99992f33eced91ef6f908bd8f33d"}, + {file = "coverage-7.2.7-cp312-cp312-win32.whl", hash = "sha256:8de8bb0e5ad103888d65abef8bca41ab93721647590a3f740100cd65c3b00511"}, + {file = "coverage-7.2.7-cp312-cp312-win_amd64.whl", hash = "sha256:9e31cb64d7de6b6f09702bb27c02d1904b3aebfca610c12772452c4e6c21a0d3"}, + {file = "coverage-7.2.7-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:58c2ccc2f00ecb51253cbe5d8d7122a34590fac9646a960d1430d5b15321d95f"}, + {file = "coverage-7.2.7-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d22656368f0e6189e24722214ed8d66b8022db19d182927b9a248a2a8a2f67eb"}, + {file = "coverage-7.2.7-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a895fcc7b15c3fc72beb43cdcbdf0ddb7d2ebc959edac9cef390b0d14f39f8a9"}, + {file = "coverage-7.2.7-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e84606b74eb7de6ff581a7915e2dab7a28a0517fbe1c9239eb227e1354064dcd"}, + {file = "coverage-7.2.7-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:0a5f9e1dbd7fbe30196578ca36f3fba75376fb99888c395c5880b355e2875f8a"}, + {file = "coverage-7.2.7-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:419bfd2caae268623dd469eff96d510a920c90928b60f2073d79f8fe2bbc5959"}, + {file = "coverage-7.2.7-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:2aee274c46590717f38ae5e4650988d1af340fe06167546cc32fe2f58ed05b02"}, + {file = "coverage-7.2.7-cp37-cp37m-win32.whl", hash = "sha256:61b9a528fb348373c433e8966535074b802c7a5d7f23c4f421e6c6e2f1697a6f"}, + {file = "coverage-7.2.7-cp37-cp37m-win_amd64.whl", hash = "sha256:b1c546aca0ca4d028901d825015dc8e4d56aac4b541877690eb76490f1dc8ed0"}, + {file = "coverage-7.2.7-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:54b896376ab563bd38453cecb813c295cf347cf5906e8b41d340b0321a5433e5"}, + {file = "coverage-7.2.7-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:3d376df58cc111dc8e21e3b6e24606b5bb5dee6024f46a5abca99124b2229ef5"}, + {file = "coverage-7.2.7-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5e330fc79bd7207e46c7d7fd2bb4af2963f5f635703925543a70b99574b0fea9"}, + {file = "coverage-7.2.7-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1e9d683426464e4a252bf70c3498756055016f99ddaec3774bf368e76bbe02b6"}, + {file = "coverage-7.2.7-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8d13c64ee2d33eccf7437961b6ea7ad8673e2be040b4f7fd4fd4d4d28d9ccb1e"}, + {file = "coverage-7.2.7-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:b7aa5f8a41217360e600da646004f878250a0d6738bcdc11a0a39928d7dc2050"}, + {file = "coverage-7.2.7-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:8fa03bce9bfbeeef9f3b160a8bed39a221d82308b4152b27d82d8daa7041fee5"}, + {file = "coverage-7.2.7-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:245167dd26180ab4c91d5e1496a30be4cd721a5cf2abf52974f965f10f11419f"}, + {file = "coverage-7.2.7-cp38-cp38-win32.whl", hash = "sha256:d2c2db7fd82e9b72937969bceac4d6ca89660db0a0967614ce2481e81a0b771e"}, + {file = "coverage-7.2.7-cp38-cp38-win_amd64.whl", hash = "sha256:2e07b54284e381531c87f785f613b833569c14ecacdcb85d56b25c4622c16c3c"}, + {file = "coverage-7.2.7-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:537891ae8ce59ef63d0123f7ac9e2ae0fc8b72c7ccbe5296fec45fd68967b6c9"}, + {file = "coverage-7.2.7-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:06fb182e69f33f6cd1d39a6c597294cff3143554b64b9825d1dc69d18cc2fff2"}, + {file = "coverage-7.2.7-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:201e7389591af40950a6480bd9edfa8ed04346ff80002cec1a66cac4549c1ad7"}, + {file = "coverage-7.2.7-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f6951407391b639504e3b3be51b7ba5f3528adbf1a8ac3302b687ecababf929e"}, + {file = "coverage-7.2.7-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6f48351d66575f535669306aa7d6d6f71bc43372473b54a832222803eb956fd1"}, + {file = "coverage-7.2.7-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:b29019c76039dc3c0fd815c41392a044ce555d9bcdd38b0fb60fb4cd8e475ba9"}, + {file = "coverage-7.2.7-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:81c13a1fc7468c40f13420732805a4c38a105d89848b7c10af65a90beff25250"}, + {file = "coverage-7.2.7-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:975d70ab7e3c80a3fe86001d8751f6778905ec723f5b110aed1e450da9d4b7f2"}, + {file = "coverage-7.2.7-cp39-cp39-win32.whl", hash = "sha256:7ee7d9d4822c8acc74a5e26c50604dff824710bc8de424904c0982e25c39c6cb"}, + {file = "coverage-7.2.7-cp39-cp39-win_amd64.whl", hash = "sha256:eb393e5ebc85245347950143969b241d08b52b88a3dc39479822e073a1a8eb27"}, + {file = "coverage-7.2.7-pp37.pp38.pp39-none-any.whl", hash = "sha256:b7b4c971f05e6ae490fef852c218b0e79d4e52f79ef0c8475566584a8fb3e01d"}, + {file = "coverage-7.2.7.tar.gz", hash = "sha256:924d94291ca674905fe9481f12294eb11f2d3d3fd1adb20314ba89e94f44ed59"}, ] [package.dependencies] @@ -1011,21 +1020,20 @@ test = ["objgraph", "psutil"] [[package]] name = "gridworks" -version = "0.2.4" +version = "0.2.7" description = "Gridworks" category = "main" optional = false python-versions = ">=3.10,<4.0" files = [ - {file = "gridworks-0.2.4-py3-none-any.whl", hash = "sha256:a838977a8f237f21cc92efd31a75f94ea8de99e45c73236e7cbd72600a6697e1"}, - {file = "gridworks-0.2.4.tar.gz", hash = "sha256:1d67a4781d724641f240c734c3c6b2cb16ea8d5ace2082958b384265ee4c028a"}, + {file = "gridworks-0.2.7-py3-none-any.whl", hash = "sha256:8ce5b35597934e00fe7bcf9a0652661c4cb48b618f9d588416590ac4cca556b6"}, + {file = "gridworks-0.2.7.tar.gz", hash = "sha256:cad8cc45199a3ad87256466b9b601956e332dee6d83e89b5169f6dcb7bbe0258"}, ] [package.dependencies] aiocache = {version = ">=0.11.1,<0.12.0", extras = ["memcached", "redis"]} aioredis = "1.3.1" beaker-pyteal = ">=0.4.0,<0.5.0" -click = ">=8.0.1" fastapi = ">=0.88.0,<0.89.0" fastapi-utils = ">=0.2.1,<0.3.0" pendulum = ">=2.1.2,<3.0.0" @@ -1041,24 +1049,63 @@ sphinx-rtd-theme = ">=1.1.1,<2.0.0" types-requests = ">=2.28.11.2,<3.0.0.0" uvicorn = {version = ">=0.19.0,<0.20.0", extras = ["standard"]} +[[package]] +name = "gridworks-proactor" +version = "0.2.2" +description = "Gridworks Proactor" +category = "main" +optional = false +python-versions = ">=3.10,<4.0" +files = [ + {file = "gridworks_proactor-0.2.2-py3-none-any.whl", hash = "sha256:4ae3c9415671e44470e545e0479934c97d6721561648dc1f516ded5139630c39"}, + {file = "gridworks_proactor-0.2.2.tar.gz", hash = "sha256:791435101e7d0a410ff1ca8eb1e5a2308c01c8bdaf3193744bef9908d6778e0d"}, +] + +[package.dependencies] +gridworks-protocol = ">=0.5.5" +paho-mqtt = ">=1.6.1,<2.0.0" +pendulum = ">=2.1.2,<3.0.0" +pydantic = ">=1.10.6,<2.0.0" +python-dotenv = ">=1.0.0,<2.0.0" +result = ">=0.9.0,<0.10.0" +xdg = ">=6.0.0,<7.0.0" + +[package.extras] +tests = ["pytest (>=7.2.0,<8.0.0)", "pytest-asyncio (>=0.20.3,<0.21.0)"] + [[package]] name = "gridworks-protocol" -version = "0.5.2" +version = "0.5.5" description = "Gridworks Protocol" category = "main" optional = false python-versions = ">=3.10,<4.0" files = [ - {file = "gridworks_protocol-0.5.2-py3-none-any.whl", hash = "sha256:f874d2dac65ec4f049d74840aaac44ab138acd6da0ee73ee80222a44902908bb"}, - {file = "gridworks_protocol-0.5.2.tar.gz", hash = "sha256:dc97179abd5e5c508bad504f9288e676daddd1ae9c2f07c60fd2c36dedaa3bb0"}, + {file = "gridworks_protocol-0.5.5-py3-none-any.whl", hash = "sha256:be7c2a695c9b0f77747728d106ab72f9827f3a360d41f44067ee1c5918ba33c0"}, + {file = "gridworks_protocol-0.5.5.tar.gz", hash = "sha256:da5215c32aca2c2f4ce78d94c4fed0d60602ed560bafc8432373c658b55a862f"}, ] [package.dependencies] fastapi-utils = ">=0.2.1,<0.3.0" -gridworks = "0.2.4" +gridworks = ">=0.2.7,<0.3.0" pendulum = ">=2.1.2,<3.0.0" pydantic = ">=1.10.2,<2.0.0" +[[package]] +name = "gridworks-ps" +version = "0.0.1" +description = "GridWorks Price Service" +category = "main" +optional = false +python-versions = ">=3.10,<4.0" +files = [ + {file = "gridworks_ps-0.0.1-py3-none-any.whl", hash = "sha256:4df6cbbd8495d1eb80dc9b06798fb69a200c3024d2c4bb6f4fb76004bfa91a36"}, + {file = "gridworks_ps-0.0.1.tar.gz", hash = "sha256:d00b6afa2ebee5db6a8f81eeb72e68ea265a8a068f6ae36fd772906540d2c6d8"}, +] + +[package.dependencies] +gridworks = ">=0.2.7,<0.3.0" + [[package]] name = "h11" version = "0.8.1" @@ -1370,14 +1417,14 @@ test = ["flaky", "ipyparallel", "pre-commit", "pytest (>=7.0)", "pytest-asyncio" [[package]] name = "ipython" -version = "8.13.1" +version = "8.13.2" description = "IPython: Productive Interactive Computing" category = "dev" optional = false python-versions = ">=3.9" files = [ - {file = "ipython-8.13.1-py3-none-any.whl", hash = "sha256:1c80d08f04144a1994cda25569eab07fbdc4989bd8d8793e3a4ff643065ccb51"}, - {file = "ipython-8.13.1.tar.gz", hash = "sha256:9c8487ac18f330c8a683fc50ab6d7bc0fcf9ef1d7a9f6ce7926938261067b81f"}, + {file = "ipython-8.13.2-py3-none-any.whl", hash = "sha256:ffca270240fbd21b06b2974e14a86494d6d29290184e788275f55e0b55914926"}, + {file = "ipython-8.13.2.tar.gz", hash = "sha256:7dff3fad32b97f6488e02f87b970f309d082f758d7b7fc252e3b19ee0e432dbb"}, ] [package.dependencies] @@ -2595,89 +2642,89 @@ files = [ [[package]] name = "pyzmq" -version = "25.0.2" +version = "25.1.0" description = "Python bindings for 0MQ" category = "dev" optional = false python-versions = ">=3.6" files = [ - {file = "pyzmq-25.0.2-cp310-cp310-macosx_10_15_universal2.whl", hash = "sha256:ac178e666c097c8d3deb5097b58cd1316092fc43e8ef5b5fdb259b51da7e7315"}, - {file = "pyzmq-25.0.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:659e62e1cbb063151c52f5b01a38e1df6b54feccfa3e2509d44c35ca6d7962ee"}, - {file = "pyzmq-25.0.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8280ada89010735a12b968ec3ea9a468ac2e04fddcc1cede59cb7f5178783b9c"}, - {file = "pyzmq-25.0.2-cp310-cp310-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a9b5eeb5278a8a636bb0abdd9ff5076bcbb836cd2302565df53ff1fa7d106d54"}, - {file = "pyzmq-25.0.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9a2e5fe42dfe6b73ca120b97ac9f34bfa8414feb15e00e37415dbd51cf227ef6"}, - {file = "pyzmq-25.0.2-cp310-cp310-manylinux_2_28_x86_64.whl", hash = "sha256:827bf60e749e78acb408a6c5af6688efbc9993e44ecc792b036ec2f4b4acf485"}, - {file = "pyzmq-25.0.2-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:7b504ae43d37e282301da586529e2ded8b36d4ee2cd5e6db4386724ddeaa6bbc"}, - {file = "pyzmq-25.0.2-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:cb1f69a0a2a2b1aae8412979dd6293cc6bcddd4439bf07e4758d864ddb112354"}, - {file = "pyzmq-25.0.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:2b9c9cc965cdf28381e36da525dcb89fc1571d9c54800fdcd73e3f73a2fc29bd"}, - {file = "pyzmq-25.0.2-cp310-cp310-win32.whl", hash = "sha256:24abbfdbb75ac5039205e72d6c75f10fc39d925f2df8ff21ebc74179488ebfca"}, - {file = "pyzmq-25.0.2-cp310-cp310-win_amd64.whl", hash = "sha256:6a821a506822fac55d2df2085a52530f68ab15ceed12d63539adc32bd4410f6e"}, - {file = "pyzmq-25.0.2-cp311-cp311-macosx_10_15_universal2.whl", hash = "sha256:9af0bb0277e92f41af35e991c242c9c71920169d6aa53ade7e444f338f4c8128"}, - {file = "pyzmq-25.0.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:54a96cf77684a3a537b76acfa7237b1e79a8f8d14e7f00e0171a94b346c5293e"}, - {file = "pyzmq-25.0.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:88649b19ede1cab03b96b66c364cbbf17c953615cdbc844f7f6e5f14c5e5261c"}, - {file = "pyzmq-25.0.2-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:715cff7644a80a7795953c11b067a75f16eb9fc695a5a53316891ebee7f3c9d5"}, - {file = "pyzmq-25.0.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:312b3f0f066b4f1d17383aae509bacf833ccaf591184a1f3c7a1661c085063ae"}, - {file = "pyzmq-25.0.2-cp311-cp311-manylinux_2_28_x86_64.whl", hash = "sha256:d488c5c8630f7e782e800869f82744c3aca4aca62c63232e5d8c490d3d66956a"}, - {file = "pyzmq-25.0.2-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:38d9f78d69bcdeec0c11e0feb3bc70f36f9b8c44fc06e5d06d91dc0a21b453c7"}, - {file = "pyzmq-25.0.2-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:3059a6a534c910e1d5d068df42f60d434f79e6cc6285aa469b384fa921f78cf8"}, - {file = "pyzmq-25.0.2-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:6526d097b75192f228c09d48420854d53dfbc7abbb41b0e26f363ccb26fbc177"}, - {file = "pyzmq-25.0.2-cp311-cp311-win32.whl", hash = "sha256:5c5fbb229e40a89a2fe73d0c1181916f31e30f253cb2d6d91bea7927c2e18413"}, - {file = "pyzmq-25.0.2-cp311-cp311-win_amd64.whl", hash = "sha256:ed15e3a2c3c2398e6ae5ce86d6a31b452dfd6ad4cd5d312596b30929c4b6e182"}, - {file = "pyzmq-25.0.2-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:032f5c8483c85bf9c9ca0593a11c7c749d734ce68d435e38c3f72e759b98b3c9"}, - {file = "pyzmq-25.0.2-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:374b55516393bfd4d7a7daa6c3b36d6dd6a31ff9d2adad0838cd6a203125e714"}, - {file = "pyzmq-25.0.2-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:08bfcc21b5997a9be4fefa405341320d8e7f19b4d684fb9c0580255c5bd6d695"}, - {file = "pyzmq-25.0.2-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:1a843d26a8da1b752c74bc019c7b20e6791ee813cd6877449e6a1415589d22ff"}, - {file = "pyzmq-25.0.2-cp36-cp36m-musllinux_1_1_aarch64.whl", hash = "sha256:b48616a09d7df9dbae2f45a0256eee7b794b903ddc6d8657a9948669b345f220"}, - {file = "pyzmq-25.0.2-cp36-cp36m-musllinux_1_1_i686.whl", hash = "sha256:d4427b4a136e3b7f85516c76dd2e0756c22eec4026afb76ca1397152b0ca8145"}, - {file = "pyzmq-25.0.2-cp36-cp36m-musllinux_1_1_x86_64.whl", hash = "sha256:26b0358e8933990502f4513c991c9935b6c06af01787a36d133b7c39b1df37fa"}, - {file = "pyzmq-25.0.2-cp36-cp36m-win32.whl", hash = "sha256:c8fedc3ccd62c6b77dfe6f43802057a803a411ee96f14e946f4a76ec4ed0e117"}, - {file = "pyzmq-25.0.2-cp36-cp36m-win_amd64.whl", hash = "sha256:2da6813b7995b6b1d1307329c73d3e3be2fd2d78e19acfc4eff2e27262732388"}, - {file = "pyzmq-25.0.2-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:a35960c8b2f63e4ef67fd6731851030df68e4b617a6715dd11b4b10312d19fef"}, - {file = "pyzmq-25.0.2-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:eef2a0b880ab40aca5a878933376cb6c1ec483fba72f7f34e015c0f675c90b20"}, - {file = "pyzmq-25.0.2-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:85762712b74c7bd18e340c3639d1bf2f23735a998d63f46bb6584d904b5e401d"}, - {file = "pyzmq-25.0.2-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:64812f29d6eee565e129ca14b0c785744bfff679a4727137484101b34602d1a7"}, - {file = "pyzmq-25.0.2-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:510d8e55b3a7cd13f8d3e9121edf0a8730b87d925d25298bace29a7e7bc82810"}, - {file = "pyzmq-25.0.2-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:b164cc3c8acb3d102e311f2eb6f3c305865ecb377e56adc015cb51f721f1dda6"}, - {file = "pyzmq-25.0.2-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:28fdb9224a258134784a9cf009b59265a9dde79582fb750d4e88a6bcbc6fa3dc"}, - {file = "pyzmq-25.0.2-cp37-cp37m-win32.whl", hash = "sha256:dd771a440effa1c36d3523bc6ba4e54ff5d2e54b4adcc1e060d8f3ca3721d228"}, - {file = "pyzmq-25.0.2-cp37-cp37m-win_amd64.whl", hash = "sha256:9bdc40efb679b9dcc39c06d25629e55581e4c4f7870a5e88db4f1c51ce25e20d"}, - {file = "pyzmq-25.0.2-cp38-cp38-macosx_10_15_universal2.whl", hash = "sha256:1f82906a2d8e4ee310f30487b165e7cc8ed09c009e4502da67178b03083c4ce0"}, - {file = "pyzmq-25.0.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:21ec0bf4831988af43c8d66ba3ccd81af2c5e793e1bf6790eb2d50e27b3c570a"}, - {file = "pyzmq-25.0.2-cp38-cp38-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:abbce982a17c88d2312ec2cf7673985d444f1beaac6e8189424e0a0e0448dbb3"}, - {file = "pyzmq-25.0.2-cp38-cp38-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:9e1d2f2d86fc75ed7f8845a992c5f6f1ab5db99747fb0d78b5e4046d041164d2"}, - {file = "pyzmq-25.0.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a2e92ff20ad5d13266bc999a29ed29a3b5b101c21fdf4b2cf420c09db9fb690e"}, - {file = "pyzmq-25.0.2-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:edbbf06cc2719889470a8d2bf5072bb00f423e12de0eb9ffec946c2c9748e149"}, - {file = "pyzmq-25.0.2-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:77942243ff4d14d90c11b2afd8ee6c039b45a0be4e53fb6fa7f5e4fd0b59da39"}, - {file = "pyzmq-25.0.2-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:ab046e9cb902d1f62c9cc0eca055b1d11108bdc271caf7c2171487298f229b56"}, - {file = "pyzmq-25.0.2-cp38-cp38-win32.whl", hash = "sha256:ad761cfbe477236802a7ab2c080d268c95e784fe30cafa7e055aacd1ca877eb0"}, - {file = "pyzmq-25.0.2-cp38-cp38-win_amd64.whl", hash = "sha256:8560756318ec7c4c49d2c341012167e704b5a46d9034905853c3d1ade4f55bee"}, - {file = "pyzmq-25.0.2-cp39-cp39-macosx_10_15_universal2.whl", hash = "sha256:ab2c056ac503f25a63f6c8c6771373e2a711b98b304614151dfb552d3d6c81f6"}, - {file = "pyzmq-25.0.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:cca8524b61c0eaaa3505382dc9b9a3bc8165f1d6c010fdd1452c224225a26689"}, - {file = "pyzmq-25.0.2-cp39-cp39-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:cfb9f7eae02d3ac42fbedad30006b7407c984a0eb4189a1322241a20944d61e5"}, - {file = "pyzmq-25.0.2-cp39-cp39-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:5eaeae038c68748082137d6896d5c4db7927e9349237ded08ee1bbd94f7361c9"}, - {file = "pyzmq-25.0.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4a31992a8f8d51663ebf79df0df6a04ffb905063083d682d4380ab8d2c67257c"}, - {file = "pyzmq-25.0.2-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:6a979e59d2184a0c8f2ede4b0810cbdd86b64d99d9cc8a023929e40dce7c86cc"}, - {file = "pyzmq-25.0.2-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:1f124cb73f1aa6654d31b183810febc8505fd0c597afa127c4f40076be4574e0"}, - {file = "pyzmq-25.0.2-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:65c19a63b4a83ae45d62178b70223adeee5f12f3032726b897431b6553aa25af"}, - {file = "pyzmq-25.0.2-cp39-cp39-win32.whl", hash = "sha256:83d822e8687621bed87404afc1c03d83fa2ce39733d54c2fd52d8829edb8a7ff"}, - {file = "pyzmq-25.0.2-cp39-cp39-win_amd64.whl", hash = "sha256:24683285cc6b7bf18ad37d75b9db0e0fefe58404e7001f1d82bf9e721806daa7"}, - {file = "pyzmq-25.0.2-pp37-pypy37_pp73-macosx_10_9_x86_64.whl", hash = "sha256:4a4b4261eb8f9ed71f63b9eb0198dd7c934aa3b3972dac586d0ef502ba9ab08b"}, - {file = "pyzmq-25.0.2-pp37-pypy37_pp73-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:62ec8d979f56c0053a92b2b6a10ff54b9ec8a4f187db2b6ec31ee3dd6d3ca6e2"}, - {file = "pyzmq-25.0.2-pp37-pypy37_pp73-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:affec1470351178e892121b3414c8ef7803269f207bf9bef85f9a6dd11cde264"}, - {file = "pyzmq-25.0.2-pp37-pypy37_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ffc71111433bd6ec8607a37b9211f4ef42e3d3b271c6d76c813669834764b248"}, - {file = "pyzmq-25.0.2-pp37-pypy37_pp73-win_amd64.whl", hash = "sha256:6fadc60970714d86eff27821f8fb01f8328dd36bebd496b0564a500fe4a9e354"}, - {file = "pyzmq-25.0.2-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:269968f2a76c0513490aeb3ba0dc3c77b7c7a11daa894f9d1da88d4a0db09835"}, - {file = "pyzmq-25.0.2-pp38-pypy38_pp73-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:f7c8b8368e84381ae7c57f1f5283b029c888504aaf4949c32e6e6fb256ec9bf0"}, - {file = "pyzmq-25.0.2-pp38-pypy38_pp73-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:25e6873a70ad5aa31e4a7c41e5e8c709296edef4a92313e1cd5fc87bbd1874e2"}, - {file = "pyzmq-25.0.2-pp38-pypy38_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b733076ff46e7db5504c5e7284f04a9852c63214c74688bdb6135808531755a3"}, - {file = "pyzmq-25.0.2-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:a6f6ae12478fdc26a6d5fdb21f806b08fa5403cd02fd312e4cb5f72df078f96f"}, - {file = "pyzmq-25.0.2-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:67da1c213fbd208906ab3470cfff1ee0048838365135a9bddc7b40b11e6d6c89"}, - {file = "pyzmq-25.0.2-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:531e36d9fcd66f18de27434a25b51d137eb546931033f392e85674c7a7cea853"}, - {file = "pyzmq-25.0.2-pp39-pypy39_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:34a6fddd159ff38aa9497b2e342a559f142ab365576284bc8f77cb3ead1f79c5"}, - {file = "pyzmq-25.0.2-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b491998ef886662c1f3d49ea2198055a9a536ddf7430b051b21054f2a5831800"}, - {file = "pyzmq-25.0.2-pp39-pypy39_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:5d496815074e3e3d183fe2c7fcea2109ad67b74084c254481f87b64e04e9a471"}, - {file = "pyzmq-25.0.2-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:56a94ab1d12af982b55ca96c6853db6ac85505e820d9458ac76364c1998972f4"}, - {file = "pyzmq-25.0.2.tar.gz", hash = "sha256:6b8c1bbb70e868dc88801aa532cae6bd4e3b5233784692b786f17ad2962e5149"}, + {file = "pyzmq-25.1.0-cp310-cp310-macosx_10_15_universal2.whl", hash = "sha256:1a6169e69034eaa06823da6a93a7739ff38716142b3596c180363dee729d713d"}, + {file = "pyzmq-25.1.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:19d0383b1f18411d137d891cab567de9afa609b214de68b86e20173dc624c101"}, + {file = "pyzmq-25.1.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f1e931d9a92f628858a50f5bdffdfcf839aebe388b82f9d2ccd5d22a38a789dc"}, + {file = "pyzmq-25.1.0-cp310-cp310-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:97d984b1b2f574bc1bb58296d3c0b64b10e95e7026f8716ed6c0b86d4679843f"}, + {file = "pyzmq-25.1.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:154bddda2a351161474b36dba03bf1463377ec226a13458725183e508840df89"}, + {file = "pyzmq-25.1.0-cp310-cp310-manylinux_2_28_x86_64.whl", hash = "sha256:cb6d161ae94fb35bb518b74bb06b7293299c15ba3bc099dccd6a5b7ae589aee3"}, + {file = "pyzmq-25.1.0-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:90146ab578931e0e2826ee39d0c948d0ea72734378f1898939d18bc9c823fcf9"}, + {file = "pyzmq-25.1.0-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:831ba20b660b39e39e5ac8603e8193f8fce1ee03a42c84ade89c36a251449d80"}, + {file = "pyzmq-25.1.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:3a522510e3434e12aff80187144c6df556bb06fe6b9d01b2ecfbd2b5bfa5c60c"}, + {file = "pyzmq-25.1.0-cp310-cp310-win32.whl", hash = "sha256:be24a5867b8e3b9dd5c241de359a9a5217698ff616ac2daa47713ba2ebe30ad1"}, + {file = "pyzmq-25.1.0-cp310-cp310-win_amd64.whl", hash = "sha256:5693dcc4f163481cf79e98cf2d7995c60e43809e325b77a7748d8024b1b7bcba"}, + {file = "pyzmq-25.1.0-cp311-cp311-macosx_10_15_universal2.whl", hash = "sha256:13bbe36da3f8aaf2b7ec12696253c0bf6ffe05f4507985a8844a1081db6ec22d"}, + {file = "pyzmq-25.1.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:69511d604368f3dc58d4be1b0bad99b61ee92b44afe1cd9b7bd8c5e34ea8248a"}, + {file = "pyzmq-25.1.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4a983c8694667fd76d793ada77fd36c8317e76aa66eec75be2653cef2ea72883"}, + {file = "pyzmq-25.1.0-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:332616f95eb400492103ab9d542b69d5f0ff628b23129a4bc0a2fd48da6e4e0b"}, + {file = "pyzmq-25.1.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:58416db767787aedbfd57116714aad6c9ce57215ffa1c3758a52403f7c68cff5"}, + {file = "pyzmq-25.1.0-cp311-cp311-manylinux_2_28_x86_64.whl", hash = "sha256:cad9545f5801a125f162d09ec9b724b7ad9b6440151b89645241d0120e119dcc"}, + {file = "pyzmq-25.1.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:d6128d431b8dfa888bf51c22a04d48bcb3d64431caf02b3cb943269f17fd2994"}, + {file = "pyzmq-25.1.0-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:2b15247c49d8cbea695b321ae5478d47cffd496a2ec5ef47131a9e79ddd7e46c"}, + {file = "pyzmq-25.1.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:442d3efc77ca4d35bee3547a8e08e8d4bb88dadb54a8377014938ba98d2e074a"}, + {file = "pyzmq-25.1.0-cp311-cp311-win32.whl", hash = "sha256:65346f507a815a731092421d0d7d60ed551a80d9b75e8b684307d435a5597425"}, + {file = "pyzmq-25.1.0-cp311-cp311-win_amd64.whl", hash = "sha256:8b45d722046fea5a5694cba5d86f21f78f0052b40a4bbbbf60128ac55bfcc7b6"}, + {file = "pyzmq-25.1.0-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:f45808eda8b1d71308c5416ef3abe958f033fdbb356984fabbfc7887bed76b3f"}, + {file = "pyzmq-25.1.0-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8b697774ea8273e3c0460cf0bba16cd85ca6c46dfe8b303211816d68c492e132"}, + {file = "pyzmq-25.1.0-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:b324fa769577fc2c8f5efcd429cef5acbc17d63fe15ed16d6dcbac2c5eb00849"}, + {file = "pyzmq-25.1.0-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:5873d6a60b778848ce23b6c0ac26c39e48969823882f607516b91fb323ce80e5"}, + {file = "pyzmq-25.1.0-cp36-cp36m-musllinux_1_1_aarch64.whl", hash = "sha256:f0d9e7ba6a815a12c8575ba7887da4b72483e4cfc57179af10c9b937f3f9308f"}, + {file = "pyzmq-25.1.0-cp36-cp36m-musllinux_1_1_i686.whl", hash = "sha256:414b8beec76521358b49170db7b9967d6974bdfc3297f47f7d23edec37329b00"}, + {file = "pyzmq-25.1.0-cp36-cp36m-musllinux_1_1_x86_64.whl", hash = "sha256:01f06f33e12497dca86353c354461f75275a5ad9eaea181ac0dc1662da8074fa"}, + {file = "pyzmq-25.1.0-cp36-cp36m-win32.whl", hash = "sha256:b5a07c4f29bf7cb0164664ef87e4aa25435dcc1f818d29842118b0ac1eb8e2b5"}, + {file = "pyzmq-25.1.0-cp36-cp36m-win_amd64.whl", hash = "sha256:968b0c737797c1809ec602e082cb63e9824ff2329275336bb88bd71591e94a90"}, + {file = "pyzmq-25.1.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:47b915ba666c51391836d7ed9a745926b22c434efa76c119f77bcffa64d2c50c"}, + {file = "pyzmq-25.1.0-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5af31493663cf76dd36b00dafbc839e83bbca8a0662931e11816d75f36155897"}, + {file = "pyzmq-25.1.0-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:5489738a692bc7ee9a0a7765979c8a572520d616d12d949eaffc6e061b82b4d1"}, + {file = "pyzmq-25.1.0-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:1fc56a0221bdf67cfa94ef2d6ce5513a3d209c3dfd21fed4d4e87eca1822e3a3"}, + {file = "pyzmq-25.1.0-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:75217e83faea9edbc29516fc90c817bc40c6b21a5771ecb53e868e45594826b0"}, + {file = "pyzmq-25.1.0-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:3830be8826639d801de9053cf86350ed6742c4321ba4236e4b5568528d7bfed7"}, + {file = "pyzmq-25.1.0-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:3575699d7fd7c9b2108bc1c6128641a9a825a58577775ada26c02eb29e09c517"}, + {file = "pyzmq-25.1.0-cp37-cp37m-win32.whl", hash = "sha256:95bd3a998d8c68b76679f6b18f520904af5204f089beebb7b0301d97704634dd"}, + {file = "pyzmq-25.1.0-cp37-cp37m-win_amd64.whl", hash = "sha256:dbc466744a2db4b7ca05589f21ae1a35066afada2f803f92369f5877c100ef62"}, + {file = "pyzmq-25.1.0-cp38-cp38-macosx_10_15_universal2.whl", hash = "sha256:3bed53f7218490c68f0e82a29c92335daa9606216e51c64f37b48eb78f1281f4"}, + {file = "pyzmq-25.1.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:eb52e826d16c09ef87132c6e360e1879c984f19a4f62d8a935345deac43f3c12"}, + {file = "pyzmq-25.1.0-cp38-cp38-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:ddbef8b53cd16467fdbfa92a712eae46dd066aa19780681a2ce266e88fbc7165"}, + {file = "pyzmq-25.1.0-cp38-cp38-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:9301cf1d7fc1ddf668d0abbe3e227fc9ab15bc036a31c247276012abb921b5ff"}, + {file = "pyzmq-25.1.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7e23a8c3b6c06de40bdb9e06288180d630b562db8ac199e8cc535af81f90e64b"}, + {file = "pyzmq-25.1.0-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:4a82faae00d1eed4809c2f18b37f15ce39a10a1c58fe48b60ad02875d6e13d80"}, + {file = "pyzmq-25.1.0-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:c8398a1b1951aaa330269c35335ae69744be166e67e0ebd9869bdc09426f3871"}, + {file = "pyzmq-25.1.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:d40682ac60b2a613d36d8d3a0cd14fbdf8e7e0618fbb40aa9fa7b796c9081584"}, + {file = "pyzmq-25.1.0-cp38-cp38-win32.whl", hash = "sha256:33d5c8391a34d56224bccf74f458d82fc6e24b3213fc68165c98b708c7a69325"}, + {file = "pyzmq-25.1.0-cp38-cp38-win_amd64.whl", hash = "sha256:c66b7ff2527e18554030319b1376d81560ca0742c6e0b17ff1ee96624a5f1afd"}, + {file = "pyzmq-25.1.0-cp39-cp39-macosx_10_15_universal2.whl", hash = "sha256:af56229ea6527a849ac9fb154a059d7e32e77a8cba27e3e62a1e38d8808cb1a5"}, + {file = "pyzmq-25.1.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:bdca18b94c404af6ae5533cd1bc310c4931f7ac97c148bbfd2cd4bdd62b96253"}, + {file = "pyzmq-25.1.0-cp39-cp39-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:0b6b42f7055bbc562f63f3df3b63e3dd1ebe9727ff0f124c3aa7bcea7b3a00f9"}, + {file = "pyzmq-25.1.0-cp39-cp39-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:4c2fc7aad520a97d64ffc98190fce6b64152bde57a10c704b337082679e74f67"}, + {file = "pyzmq-25.1.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:be86a26415a8b6af02cd8d782e3a9ae3872140a057f1cadf0133de685185c02b"}, + {file = "pyzmq-25.1.0-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:851fb2fe14036cfc1960d806628b80276af5424db09fe5c91c726890c8e6d943"}, + {file = "pyzmq-25.1.0-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:2a21fec5c3cea45421a19ccbe6250c82f97af4175bc09de4d6dd78fb0cb4c200"}, + {file = "pyzmq-25.1.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:bad172aba822444b32eae54c2d5ab18cd7dee9814fd5c7ed026603b8cae2d05f"}, + {file = "pyzmq-25.1.0-cp39-cp39-win32.whl", hash = "sha256:4d67609b37204acad3d566bb7391e0ecc25ef8bae22ff72ebe2ad7ffb7847158"}, + {file = "pyzmq-25.1.0-cp39-cp39-win_amd64.whl", hash = "sha256:71c7b5896e40720d30cd77a81e62b433b981005bbff0cb2f739e0f8d059b5d99"}, + {file = "pyzmq-25.1.0-pp37-pypy37_pp73-macosx_10_9_x86_64.whl", hash = "sha256:4cb27ef9d3bdc0c195b2dc54fcb8720e18b741624686a81942e14c8b67cc61a6"}, + {file = "pyzmq-25.1.0-pp37-pypy37_pp73-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:0c4fc2741e0513b5d5a12fe200d6785bbcc621f6f2278893a9ca7bed7f2efb7d"}, + {file = "pyzmq-25.1.0-pp37-pypy37_pp73-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:fc34fdd458ff77a2a00e3c86f899911f6f269d393ca5675842a6e92eea565bae"}, + {file = "pyzmq-25.1.0-pp37-pypy37_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8751f9c1442624da391bbd92bd4b072def6d7702a9390e4479f45c182392ff78"}, + {file = "pyzmq-25.1.0-pp37-pypy37_pp73-win_amd64.whl", hash = "sha256:6581e886aec3135964a302a0f5eb68f964869b9efd1dbafdebceaaf2934f8a68"}, + {file = "pyzmq-25.1.0-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:5482f08d2c3c42b920e8771ae8932fbaa0a67dff925fc476996ddd8155a170f3"}, + {file = "pyzmq-25.1.0-pp38-pypy38_pp73-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:5e7fbcafa3ea16d1de1f213c226005fea21ee16ed56134b75b2dede5a2129e62"}, + {file = "pyzmq-25.1.0-pp38-pypy38_pp73-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:adecf6d02b1beab8d7c04bc36f22bb0e4c65a35eb0b4750b91693631d4081c70"}, + {file = "pyzmq-25.1.0-pp38-pypy38_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f6d39e42a0aa888122d1beb8ec0d4ddfb6c6b45aecb5ba4013c27e2f28657765"}, + {file = "pyzmq-25.1.0-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:7018289b402ebf2b2c06992813523de61d4ce17bd514c4339d8f27a6f6809492"}, + {file = "pyzmq-25.1.0-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:9e68ae9864d260b18f311b68d29134d8776d82e7f5d75ce898b40a88df9db30f"}, + {file = "pyzmq-25.1.0-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e21cc00e4debe8f54c3ed7b9fcca540f46eee12762a9fa56feb8512fd9057161"}, + {file = "pyzmq-25.1.0-pp39-pypy39_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:2f666ae327a6899ff560d741681fdcdf4506f990595201ed39b44278c471ad98"}, + {file = "pyzmq-25.1.0-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2f5efcc29056dfe95e9c9db0dfbb12b62db9c4ad302f812931b6d21dd04a9119"}, + {file = "pyzmq-25.1.0-pp39-pypy39_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:48e5e59e77c1a83162ab3c163fc01cd2eebc5b34560341a67421b09be0891287"}, + {file = "pyzmq-25.1.0-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:108c96ebbd573d929740d66e4c3d1bdf31d5cde003b8dc7811a3c8c5b0fc173b"}, + {file = "pyzmq-25.1.0.tar.gz", hash = "sha256:80c41023465d36280e801564a69cbfce8ae85ff79b080e1913f6e90481fb8957"}, ] [package.dependencies] @@ -2734,6 +2781,18 @@ files = [ [package.dependencies] docutils = ">=0.11,<1.0" +[[package]] +name = "result" +version = "0.9.0" +description = "A Rust-like result type for Python" +category = "main" +optional = false +python-versions = ">=3.7" +files = [ + {file = "result-0.9.0-py3-none-any.whl", hash = "sha256:2dd342c13fbf28d207c3f466f01a7915c6ee0d1fe3235fd9ba2041d36de498c1"}, + {file = "result-0.9.0.tar.gz", hash = "sha256:cc4147c824221e8326d143a55065ec17b6337a9c0dda7700f7f947ca33ac63fd"}, +] + [[package]] name = "rfc3986" version = "1.5.0" @@ -2802,7 +2861,8 @@ files = [ {file = "ruamel.yaml.clib-0.2.7-cp310-cp310-win32.whl", hash = "sha256:763d65baa3b952479c4e972669f679fe490eee058d5aa85da483ebae2009d231"}, {file = "ruamel.yaml.clib-0.2.7-cp310-cp310-win_amd64.whl", hash = "sha256:d000f258cf42fec2b1bbf2863c61d7b8918d31ffee905da62dede869254d3b8a"}, {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:045e0626baf1c52e5527bd5db361bc83180faaba2ff586e763d3d5982a876a9e"}, - {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-macosx_12_6_arm64.whl", hash = "sha256:721bc4ba4525f53f6a611ec0967bdcee61b31df5a56801281027a3a6d1c2daf5"}, + {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-macosx_13_0_arm64.whl", hash = "sha256:1a6391a7cabb7641c32517539ca42cf84b87b667bad38b78d4d42dd23e957c81"}, + {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-manylinux2014_aarch64.whl", hash = "sha256:9c7617df90c1365638916b98cdd9be833d31d337dbcd722485597b43c4a215bf"}, {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:41d0f1fa4c6830176eef5b276af04c89320ea616655d01327d5ce65e50575c94"}, {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-win32.whl", hash = "sha256:f6d3d39611ac2e4f62c3128a9eed45f19a6608670c5a2f4f07f24e8de3441d38"}, {file = "ruamel.yaml.clib-0.2.7-cp311-cp311-win_amd64.whl", hash = "sha256:da538167284de58a52109a9b89b8f6a53ff8437dd6dc26d33b57bf6699153122"}, @@ -3222,7 +3282,7 @@ files = [ ] [package.dependencies] -greenlet = {version = "!=0.4.17", markers = "python_version >= \"3\" and (platform_machine == \"aarch64\" or platform_machine == \"ppc64le\" or platform_machine == \"x86_64\" or platform_machine == \"amd64\" or platform_machine == \"AMD64\" or platform_machine == \"win32\" or platform_machine == \"WIN32\")"} +greenlet = {version = "!=0.4.17", markers = "python_version >= \"3\" and platform_machine == \"aarch64\" or python_version >= \"3\" and platform_machine == \"ppc64le\" or python_version >= \"3\" and platform_machine == \"x86_64\" or python_version >= \"3\" and platform_machine == \"amd64\" or python_version >= \"3\" and platform_machine == \"AMD64\" or python_version >= \"3\" and platform_machine == \"win32\" or python_version >= \"3\" and platform_machine == \"WIN32\""} [package.extras] aiomysql = ["aiomysql", "greenlet (!=0.4.17)"] @@ -3407,14 +3467,14 @@ types-urllib3 = "<1.27" [[package]] name = "types-urllib3" -version = "1.26.25.12" +version = "1.26.25.13" description = "Typing stubs for urllib3" category = "main" optional = false python-versions = "*" files = [ - {file = "types-urllib3-1.26.25.12.tar.gz", hash = "sha256:a1557355ce8d350a555d142589f3001903757d2d36c18a66f588d9659bbc917d"}, - {file = "types_urllib3-1.26.25.12-py3-none-any.whl", hash = "sha256:3ba3d3a8ee46e0d5512c6bd0594da4f10b2584b47a470f8422044a2ab462f1df"}, + {file = "types-urllib3-1.26.25.13.tar.gz", hash = "sha256:3300538c9dc11dad32eae4827ac313f5d986b8b21494801f1bf97a1ac6c03ae5"}, + {file = "types_urllib3-1.26.25.13-py3-none-any.whl", hash = "sha256:5dbd1d2bef14efee43f5318b5d36d805a489f6600252bb53626d4bfafd95e27c"}, ] [[package]] @@ -3587,82 +3647,94 @@ files = [ [[package]] name = "websockets" -version = "11.0.2" +version = "11.0.3" description = "An implementation of the WebSocket Protocol (RFC 6455 & 7692)" category = "main" optional = false python-versions = ">=3.7" files = [ - {file = "websockets-11.0.2-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:580cc95c58118f8c39106be71e24d0b7e1ad11a155f40a2ee687f99b3e5e432e"}, - {file = "websockets-11.0.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:143782041e95b63083b02107f31cda999f392903ae331de1307441f3a4557d51"}, - {file = "websockets-11.0.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:8df63dcd955eb6b2e371d95aacf8b7c535e482192cff1b6ce927d8f43fb4f552"}, - {file = "websockets-11.0.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ca9b2dced5cbbc5094678cc1ec62160f7b0fe4defd601cd28a36fde7ee71bbb5"}, - {file = "websockets-11.0.2-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e0eeeea3b01c97fd3b5049a46c908823f68b59bf0e18d79b231d8d6764bc81ee"}, - {file = "websockets-11.0.2-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:502683c5dedfc94b9f0f6790efb26aa0591526e8403ad443dce922cd6c0ec83b"}, - {file = "websockets-11.0.2-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:d3cc3e48b6c9f7df8c3798004b9c4b92abca09eeea5e1b0a39698f05b7a33b9d"}, - {file = "websockets-11.0.2-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:808b8a33c961bbd6d33c55908f7c137569b09ea7dd024bce969969aa04ecf07c"}, - {file = "websockets-11.0.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:34a6f8996964ccaa40da42ee36aa1572adcb1e213665e24aa2f1037da6080909"}, - {file = "websockets-11.0.2-cp310-cp310-win32.whl", hash = "sha256:8f24cd758cbe1607a91b720537685b64e4d39415649cac9177cd1257317cf30c"}, - {file = "websockets-11.0.2-cp310-cp310-win_amd64.whl", hash = "sha256:3b87cd302f08ea9e74fdc080470eddbed1e165113c1823fb3ee6328bc40ca1d3"}, - {file = "websockets-11.0.2-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:3565a8f8c7bdde7c29ebe46146bd191290413ee6f8e94cf350609720c075b0a1"}, - {file = "websockets-11.0.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:f97e03d4d5a4f0dca739ea274be9092822f7430b77d25aa02da6775e490f6846"}, - {file = "websockets-11.0.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:8f392587eb2767afa8a34e909f2fec779f90b630622adc95d8b5e26ea8823cb8"}, - {file = "websockets-11.0.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7742cd4524622cc7aa71734b51294644492a961243c4fe67874971c4d3045982"}, - {file = "websockets-11.0.2-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:46dda4bc2030c335abe192b94e98686615f9274f6b56f32f2dd661fb303d9d12"}, - {file = "websockets-11.0.2-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d6b2bfa1d884c254b841b0ff79373b6b80779088df6704f034858e4d705a4802"}, - {file = "websockets-11.0.2-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:1df2413266bf48430ef2a752c49b93086c6bf192d708e4a9920544c74cd2baa6"}, - {file = "websockets-11.0.2-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:cf45d273202b0c1cec0f03a7972c655b93611f2e996669667414557230a87b88"}, - {file = "websockets-11.0.2-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:3a09cce3dacb6ad638fdfa3154d9e54a98efe7c8f68f000e55ca9c716496ca67"}, - {file = "websockets-11.0.2-cp311-cp311-win32.whl", hash = "sha256:2174a75d579d811279855df5824676d851a69f52852edb0e7551e0eeac6f59a4"}, - {file = "websockets-11.0.2-cp311-cp311-win_amd64.whl", hash = "sha256:c78ca3037a954a4209b9f900e0eabbc471fb4ebe96914016281df2c974a93e3e"}, - {file = "websockets-11.0.2-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:3a2100b02d1aaf66dc48ff1b2a72f34f6ebc575a02bc0350cc8e9fbb35940166"}, - {file = "websockets-11.0.2-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:dca9708eea9f9ed300394d4775beb2667288e998eb6f542cdb6c02027430c599"}, - {file = "websockets-11.0.2-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:320ddceefd2364d4afe6576195201a3632a6f2e6d207b0c01333e965b22dbc84"}, - {file = "websockets-11.0.2-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b2a573c8d71b7af937852b61e7ccb37151d719974146b5dc734aad350ef55a02"}, - {file = "websockets-11.0.2-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:13bd5bebcd16a4b5e403061b8b9dcc5c77e7a71e3c57e072d8dff23e33f70fba"}, - {file = "websockets-11.0.2-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:95c09427c1c57206fe04277bf871b396476d5a8857fa1b99703283ee497c7a5d"}, - {file = "websockets-11.0.2-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:2eb042734e710d39e9bc58deab23a65bd2750e161436101488f8af92f183c239"}, - {file = "websockets-11.0.2-cp37-cp37m-win32.whl", hash = "sha256:5875f623a10b9ba154cb61967f940ab469039f0b5e61c80dd153a65f024d9fb7"}, - {file = "websockets-11.0.2-cp37-cp37m-win_amd64.whl", hash = "sha256:634239bc844131863762865b75211a913c536817c0da27f691400d49d256df1d"}, - {file = "websockets-11.0.2-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:3178d965ec204773ab67985a09f5696ca6c3869afeed0bb51703ea404a24e975"}, - {file = "websockets-11.0.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:955fcdb304833df2e172ce2492b7b47b4aab5dcc035a10e093d911a1916f2c87"}, - {file = "websockets-11.0.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:cb46d2c7631b2e6f10f7c8bac7854f7c5e5288f024f1c137d4633c79ead1e3c0"}, - {file = "websockets-11.0.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:25aae96c1060e85836552a113495db6d857400288161299d77b7b20f2ac569f2"}, - {file = "websockets-11.0.2-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:2abeeae63154b7f63d9f764685b2d299e9141171b8b896688bd8baec6b3e2303"}, - {file = "websockets-11.0.2-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:daa1e8ea47507555ed7a34f8b49398d33dff5b8548eae3de1dc0ef0607273a33"}, - {file = "websockets-11.0.2-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:954eb789c960fa5daaed3cfe336abc066941a5d456ff6be8f0e03dd89886bb4c"}, - {file = "websockets-11.0.2-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:3ffe251a31f37e65b9b9aca5d2d67fd091c234e530f13d9dce4a67959d5a3fba"}, - {file = "websockets-11.0.2-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:adf6385f677ed2e0b021845b36f55c43f171dab3a9ee0ace94da67302f1bc364"}, - {file = "websockets-11.0.2-cp38-cp38-win32.whl", hash = "sha256:aa7b33c1fb2f7b7b9820f93a5d61ffd47f5a91711bc5fa4583bbe0c0601ec0b2"}, - {file = "websockets-11.0.2-cp38-cp38-win_amd64.whl", hash = "sha256:220d5b93764dd70d7617f1663da64256df7e7ea31fc66bc52c0e3750ee134ae3"}, - {file = "websockets-11.0.2-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:0fb4480556825e4e6bf2eebdbeb130d9474c62705100c90e59f2f56459ddab42"}, - {file = "websockets-11.0.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:ec00401846569aaf018700249996143f567d50050c5b7b650148989f956547af"}, - {file = "websockets-11.0.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:87c69f50281126dcdaccd64d951fb57fbce272578d24efc59bce72cf264725d0"}, - {file = "websockets-11.0.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:232b6ba974f5d09b1b747ac232f3a3d8f86de401d7b565e837cc86988edf37ac"}, - {file = "websockets-11.0.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:392d409178db1e46d1055e51cc850136d302434e12d412a555e5291ab810f622"}, - {file = "websockets-11.0.2-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a4fe2442091ff71dee0769a10449420fd5d3b606c590f78dd2b97d94b7455640"}, - {file = "websockets-11.0.2-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:ede13a6998ba2568b21825809d96e69a38dc43184bdeebbde3699c8baa21d015"}, - {file = "websockets-11.0.2-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:4c54086b2d2aec3c3cb887ad97e9c02c6be9f1d48381c7419a4aa932d31661e4"}, - {file = "websockets-11.0.2-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:e37a76ccd483a6457580077d43bc3dfe1fd784ecb2151fcb9d1c73f424deaeba"}, - {file = "websockets-11.0.2-cp39-cp39-win32.whl", hash = "sha256:d1881518b488a920434a271a6e8a5c9481a67c4f6352ebbdd249b789c0467ddc"}, - {file = "websockets-11.0.2-cp39-cp39-win_amd64.whl", hash = "sha256:25e265686ea385f22a00cc2b719b880797cd1bb53b46dbde969e554fb458bfde"}, - {file = "websockets-11.0.2-pp37-pypy37_pp73-macosx_10_9_x86_64.whl", hash = "sha256:ce69f5c742eefd039dce8622e99d811ef2135b69d10f9aa79fbf2fdcc1e56cd7"}, - {file = "websockets-11.0.2-pp37-pypy37_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b985ba2b9e972cf99ddffc07df1a314b893095f62c75bc7c5354a9c4647c6503"}, - {file = "websockets-11.0.2-pp37-pypy37_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1b52def56d2a26e0e9c464f90cadb7e628e04f67b0ff3a76a4d9a18dfc35e3dd"}, - {file = "websockets-11.0.2-pp37-pypy37_pp73-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d70a438ef2a22a581d65ad7648e949d4ccd20e3c8ed7a90bbc46df4e60320891"}, - {file = "websockets-11.0.2-pp37-pypy37_pp73-win_amd64.whl", hash = "sha256:752fbf420c71416fb1472fec1b4cb8631c1aa2be7149e0a5ba7e5771d75d2bb9"}, - {file = "websockets-11.0.2-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:dd906b0cdc417ea7a5f13bb3c6ca3b5fd563338dc596996cb0fdd7872d691c0a"}, - {file = "websockets-11.0.2-pp38-pypy38_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3e79065ff6549dd3c765e7916067e12a9c91df2affea0ac51bcd302aaf7ad207"}, - {file = "websockets-11.0.2-pp38-pypy38_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:46388a050d9e40316e58a3f0838c63caacb72f94129eb621a659a6e49bad27ce"}, - {file = "websockets-11.0.2-pp38-pypy38_pp73-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5c7de298371d913824f71b30f7685bb07ad13969c79679cca5b1f7f94fec012f"}, - {file = "websockets-11.0.2-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:6d872c972c87c393e6a49c1afbdc596432df8c06d0ff7cd05aa18e885e7cfb7c"}, - {file = "websockets-11.0.2-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:b444366b605d2885f0034dd889faf91b4b47668dd125591e2c64bfde611ac7e1"}, - {file = "websockets-11.0.2-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e8b967a4849db6b567dec3f7dd5d97b15ce653e3497b8ce0814e470d5e074750"}, - {file = "websockets-11.0.2-pp39-pypy39_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:2acdc82099999e44fa7bd8c886f03c70a22b1d53ae74252f389be30d64fd6004"}, - {file = "websockets-11.0.2-pp39-pypy39_pp73-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:518ed6782d9916c5721ebd61bb7651d244178b74399028302c8617d0620af291"}, - {file = "websockets-11.0.2-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:58477b041099bb504e1a5ddd8aa86302ed1d5c6995bdd3db2b3084ef0135d277"}, - {file = "websockets-11.0.2-py3-none-any.whl", hash = "sha256:5004c087d17251938a52cce21b3dbdabeecbbe432ce3f5bbbf15d8692c36eac9"}, - {file = "websockets-11.0.2.tar.gz", hash = "sha256:b1a69701eb98ed83dd099de4a686dc892c413d974fa31602bc00aca7cb988ac9"}, + {file = "websockets-11.0.3-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:3ccc8a0c387629aec40f2fc9fdcb4b9d5431954f934da3eaf16cdc94f67dbfac"}, + {file = "websockets-11.0.3-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:d67ac60a307f760c6e65dad586f556dde58e683fab03323221a4e530ead6f74d"}, + {file = "websockets-11.0.3-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:84d27a4832cc1a0ee07cdcf2b0629a8a72db73f4cf6de6f0904f6661227f256f"}, + {file = "websockets-11.0.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ffd7dcaf744f25f82190856bc26ed81721508fc5cbf2a330751e135ff1283564"}, + {file = "websockets-11.0.3-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7622a89d696fc87af8e8d280d9b421db5133ef5b29d3f7a1ce9f1a7bf7fcfa11"}, + {file = "websockets-11.0.3-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bceab846bac555aff6427d060f2fcfff71042dba6f5fca7dc4f75cac815e57ca"}, + {file = "websockets-11.0.3-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:54c6e5b3d3a8936a4ab6870d46bdd6ec500ad62bde9e44462c32d18f1e9a8e54"}, + {file = "websockets-11.0.3-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:41f696ba95cd92dc047e46b41b26dd24518384749ed0d99bea0a941ca87404c4"}, + {file = "websockets-11.0.3-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:86d2a77fd490ae3ff6fae1c6ceaecad063d3cc2320b44377efdde79880e11526"}, + {file = "websockets-11.0.3-cp310-cp310-win32.whl", hash = "sha256:2d903ad4419f5b472de90cd2d40384573b25da71e33519a67797de17ef849b69"}, + {file = "websockets-11.0.3-cp310-cp310-win_amd64.whl", hash = "sha256:1d2256283fa4b7f4c7d7d3e84dc2ece74d341bce57d5b9bf385df109c2a1a82f"}, + {file = "websockets-11.0.3-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:e848f46a58b9fcf3d06061d17be388caf70ea5b8cc3466251963c8345e13f7eb"}, + {file = "websockets-11.0.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:aa5003845cdd21ac0dc6c9bf661c5beddd01116f6eb9eb3c8e272353d45b3288"}, + {file = "websockets-11.0.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:b58cbf0697721120866820b89f93659abc31c1e876bf20d0b3d03cef14faf84d"}, + {file = "websockets-11.0.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:660e2d9068d2bedc0912af508f30bbeb505bbbf9774d98def45f68278cea20d3"}, + {file = "websockets-11.0.3-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c1f0524f203e3bd35149f12157438f406eff2e4fb30f71221c8a5eceb3617b6b"}, + {file = "websockets-11.0.3-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:def07915168ac8f7853812cc593c71185a16216e9e4fa886358a17ed0fd9fcf6"}, + {file = "websockets-11.0.3-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:b30c6590146e53149f04e85a6e4fcae068df4289e31e4aee1fdf56a0dead8f97"}, + {file = "websockets-11.0.3-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:619d9f06372b3a42bc29d0cd0354c9bb9fb39c2cbc1a9c5025b4538738dbffaf"}, + {file = "websockets-11.0.3-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:01f5567d9cf6f502d655151645d4e8b72b453413d3819d2b6f1185abc23e82dd"}, + {file = "websockets-11.0.3-cp311-cp311-win32.whl", hash = "sha256:e1459677e5d12be8bbc7584c35b992eea142911a6236a3278b9b5ce3326f282c"}, + {file = "websockets-11.0.3-cp311-cp311-win_amd64.whl", hash = "sha256:e7837cb169eca3b3ae94cc5787c4fed99eef74c0ab9506756eea335e0d6f3ed8"}, + {file = "websockets-11.0.3-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:9f59a3c656fef341a99e3d63189852be7084c0e54b75734cde571182c087b152"}, + {file = "websockets-11.0.3-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2529338a6ff0eb0b50c7be33dc3d0e456381157a31eefc561771ee431134a97f"}, + {file = "websockets-11.0.3-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:34fd59a4ac42dff6d4681d8843217137f6bc85ed29722f2f7222bd619d15e95b"}, + {file = "websockets-11.0.3-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:332d126167ddddec94597c2365537baf9ff62dfcc9db4266f263d455f2f031cb"}, + {file = "websockets-11.0.3-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:6505c1b31274723ccaf5f515c1824a4ad2f0d191cec942666b3d0f3aa4cb4007"}, + {file = "websockets-11.0.3-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:f467ba0050b7de85016b43f5a22b46383ef004c4f672148a8abf32bc999a87f0"}, + {file = "websockets-11.0.3-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:9d9acd80072abcc98bd2c86c3c9cd4ac2347b5a5a0cae7ed5c0ee5675f86d9af"}, + {file = "websockets-11.0.3-cp37-cp37m-win32.whl", hash = "sha256:e590228200fcfc7e9109509e4d9125eace2042fd52b595dd22bbc34bb282307f"}, + {file = "websockets-11.0.3-cp37-cp37m-win_amd64.whl", hash = "sha256:b16fff62b45eccb9c7abb18e60e7e446998093cdcb50fed33134b9b6878836de"}, + {file = "websockets-11.0.3-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:fb06eea71a00a7af0ae6aefbb932fb8a7df3cb390cc217d51a9ad7343de1b8d0"}, + {file = "websockets-11.0.3-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:8a34e13a62a59c871064dfd8ffb150867e54291e46d4a7cf11d02c94a5275bae"}, + {file = "websockets-11.0.3-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:4841ed00f1026dfbced6fca7d963c4e7043aa832648671b5138008dc5a8f6d99"}, + {file = "websockets-11.0.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1a073fc9ab1c8aff37c99f11f1641e16da517770e31a37265d2755282a5d28aa"}, + {file = "websockets-11.0.3-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:68b977f21ce443d6d378dbd5ca38621755f2063d6fdb3335bda981d552cfff86"}, + {file = "websockets-11.0.3-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e1a99a7a71631f0efe727c10edfba09ea6bee4166a6f9c19aafb6c0b5917d09c"}, + {file = "websockets-11.0.3-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:bee9fcb41db2a23bed96c6b6ead6489702c12334ea20a297aa095ce6d31370d0"}, + {file = "websockets-11.0.3-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:4b253869ea05a5a073ebfdcb5cb3b0266a57c3764cf6fe114e4cd90f4bfa5f5e"}, + {file = "websockets-11.0.3-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:1553cb82942b2a74dd9b15a018dce645d4e68674de2ca31ff13ebc2d9f283788"}, + {file = "websockets-11.0.3-cp38-cp38-win32.whl", hash = "sha256:f61bdb1df43dc9c131791fbc2355535f9024b9a04398d3bd0684fc16ab07df74"}, + {file = "websockets-11.0.3-cp38-cp38-win_amd64.whl", hash = "sha256:03aae4edc0b1c68498f41a6772d80ac7c1e33c06c6ffa2ac1c27a07653e79d6f"}, + {file = "websockets-11.0.3-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:777354ee16f02f643a4c7f2b3eff8027a33c9861edc691a2003531f5da4f6bc8"}, + {file = "websockets-11.0.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:8c82f11964f010053e13daafdc7154ce7385ecc538989a354ccc7067fd7028fd"}, + {file = "websockets-11.0.3-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:3580dd9c1ad0701169e4d6fc41e878ffe05e6bdcaf3c412f9d559389d0c9e016"}, + {file = "websockets-11.0.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6f1a3f10f836fab6ca6efa97bb952300b20ae56b409414ca85bff2ad241d2a61"}, + {file = "websockets-11.0.3-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:df41b9bc27c2c25b486bae7cf42fccdc52ff181c8c387bfd026624a491c2671b"}, + {file = "websockets-11.0.3-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:279e5de4671e79a9ac877427f4ac4ce93751b8823f276b681d04b2156713b9dd"}, + {file = "websockets-11.0.3-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:1fdf26fa8a6a592f8f9235285b8affa72748dc12e964a5518c6c5e8f916716f7"}, + {file = "websockets-11.0.3-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:69269f3a0b472e91125b503d3c0b3566bda26da0a3261c49f0027eb6075086d1"}, + {file = "websockets-11.0.3-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:97b52894d948d2f6ea480171a27122d77af14ced35f62e5c892ca2fae9344311"}, + {file = "websockets-11.0.3-cp39-cp39-win32.whl", hash = "sha256:c7f3cb904cce8e1be667c7e6fef4516b98d1a6a0635a58a57528d577ac18a128"}, + {file = "websockets-11.0.3-cp39-cp39-win_amd64.whl", hash = "sha256:c792ea4eabc0159535608fc5658a74d1a81020eb35195dd63214dcf07556f67e"}, + {file = "websockets-11.0.3-pp37-pypy37_pp73-macosx_10_9_x86_64.whl", hash = "sha256:f2e58f2c36cc52d41f2659e4c0cbf7353e28c8c9e63e30d8c6d3494dc9fdedcf"}, + {file = "websockets-11.0.3-pp37-pypy37_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:de36fe9c02995c7e6ae6efe2e205816f5f00c22fd1fbf343d4d18c3d5ceac2f5"}, + {file = "websockets-11.0.3-pp37-pypy37_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:0ac56b661e60edd453585f4bd68eb6a29ae25b5184fd5ba51e97652580458998"}, + {file = "websockets-11.0.3-pp37-pypy37_pp73-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e052b8467dd07d4943936009f46ae5ce7b908ddcac3fda581656b1b19c083d9b"}, + {file = "websockets-11.0.3-pp37-pypy37_pp73-win_amd64.whl", hash = "sha256:42cc5452a54a8e46a032521d7365da775823e21bfba2895fb7b77633cce031bb"}, + {file = "websockets-11.0.3-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:e6316827e3e79b7b8e7d8e3b08f4e331af91a48e794d5d8b099928b6f0b85f20"}, + {file = "websockets-11.0.3-pp38-pypy38_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8531fdcad636d82c517b26a448dcfe62f720e1922b33c81ce695d0edb91eb931"}, + {file = "websockets-11.0.3-pp38-pypy38_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c114e8da9b475739dde229fd3bc6b05a6537a88a578358bc8eb29b4030fac9c9"}, + {file = "websockets-11.0.3-pp38-pypy38_pp73-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e063b1865974611313a3849d43f2c3f5368093691349cf3c7c8f8f75ad7cb280"}, + {file = "websockets-11.0.3-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:92b2065d642bf8c0a82d59e59053dd2fdde64d4ed44efe4870fa816c1232647b"}, + {file = "websockets-11.0.3-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:0ee68fe502f9031f19d495dae2c268830df2760c0524cbac5d759921ba8c8e82"}, + {file = "websockets-11.0.3-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:dcacf2c7a6c3a84e720d1bb2b543c675bf6c40e460300b628bab1b1efc7c034c"}, + {file = "websockets-11.0.3-pp39-pypy39_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:b67c6f5e5a401fc56394f191f00f9b3811fe843ee93f4a70df3c389d1adf857d"}, + {file = "websockets-11.0.3-pp39-pypy39_pp73-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1d5023a4b6a5b183dc838808087033ec5df77580485fc533e7dab2567851b0a4"}, + {file = "websockets-11.0.3-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:ed058398f55163a79bb9f06a90ef9ccc063b204bb346c4de78efc5d15abfe602"}, + {file = "websockets-11.0.3-py3-none-any.whl", hash = "sha256:6681ba9e7f8f3b19440921e99efbb40fc89f26cd71bf539e45d8c8a25c976dc6"}, + {file = "websockets-11.0.3.tar.gz", hash = "sha256:88fc51d9a26b10fc331be344f1781224a375b78488fc343620184e95a4b27016"}, +] + +[[package]] +name = "xdg" +version = "6.0.0" +description = "Variables defined by the XDG Base Directory Specification" +category = "main" +optional = false +python-versions = ">=3.7,<4.0" +files = [ + {file = "xdg-6.0.0-py3-none-any.whl", hash = "sha256:df3510755b4395157fc04fc3b02467c777f3b3ca383257397f09ab0d4c16f936"}, + {file = "xdg-6.0.0.tar.gz", hash = "sha256:24278094f2d45e846d1eb28a2ebb92d7b67fc0cab5249ee3ce88c95f649a1c92"}, ] [[package]] @@ -3710,4 +3782,4 @@ files = [ [metadata] lock-version = "2.0" python-versions = ">=3.10,<4.0" -content-hash = "0101bd065af0eb8ce9f77ba30cac0f714a1cc2f717871b63ef9d40ad7b82bad9" +content-hash = "334140502553252824737c85921344d30595badf3fab0b73776558cdfc9cd6d2" diff --git a/pyproject.toml b/pyproject.toml index 56164df..a366e10 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,8 +1,8 @@ [tool.poetry] name = "gridworks-atn" -version = "0.3.7" +version = "0.3.8" description = "Gridworks Atn Spaceheat" -authors = ["Jessica Millar "] +authors = ["GridWorks "] license = "None" readme = "README.md" homepage = "https://github.com/thegridelectric/gridworks-atn" @@ -20,13 +20,12 @@ Changelog = "https://github.com/thegridelectric/gridworks-atn/releases" [tool.poetry.dependencies] python = ">=3.10,<4.0" -gridworks = "^0.2.4" -gridworks-protocol = "^0.5.2" # Note gridworks-protocol includes gridworks; these must stay in sync -# gridworks-proactor = "^0.1.8" paho-mqtt = "^1.6.1" - - numpy = "^1.23.4" +#gridworks-proactor = "0.2.2" +#gridworks = { path = "../gridworks"} +gridworks-proactor = "^0.2.2" +gridworks-ps = "^0.0.1" diff --git a/src/gwatn/api_types.py b/src/gwatn/api_types.py index 7f24920..5fc25d1 100644 --- a/src/gwatn/api_types.py +++ b/src/gwatn/api_types.py @@ -8,6 +8,7 @@ from gwatn.types import AtnParams_Maker from gwatn.types import AtnParamsBrickstorageheater_Maker from gwatn.types import AtnParamsReport_Maker +from gwatn.types import AtnParamsSimpleresistivehydronic_Maker from gwatn.types import BaseGNodeGt_Maker from gwatn.types import BasegnodeScadaCreate_Maker from gwatn.types import ComponentAttributeClassGt_Maker @@ -22,6 +23,7 @@ from gwatn.types import FloParams_Maker from gwatn.types import FloParamsBrickstorageheater_Maker from gwatn.types import FloParamsReport_Maker +from gwatn.types import FloParamsSimpleresistivehydronic_Maker from gwatn.types import GNodeGt_Maker from gwatn.types import GNodeInstanceGt_Maker from gwatn.types import GtDispatchBoolean_Maker @@ -91,6 +93,7 @@ def type_makers() -> List[HeartbeatA_Maker]: AtnParams_Maker, AtnParamsBrickstorageheater_Maker, AtnParamsReport_Maker, + AtnParamsSimpleresistivehydronic_Maker, BaseGNodeGt_Maker, BasegnodeScadaCreate_Maker, ComponentAttributeClassGt_Maker, @@ -105,6 +108,7 @@ def type_makers() -> List[HeartbeatA_Maker]: FloParams_Maker, FloParamsBrickstorageheater_Maker, FloParamsReport_Maker, + FloParamsSimpleresistivehydronic_Maker, GNodeGt_Maker, GNodeInstanceGt_Maker, GtDispatchBoolean_Maker, @@ -180,6 +184,7 @@ def version_by_type_name() -> Dict[str, str]: "atn.params": "000", "atn.params.brickstorageheater": "000", "atn.params.report": "000", + "atn.params.simpleresistivehydronic": "000", "base.g.node.gt": "002", "basegnode.scada.create": "000", "component.attribute.class.gt": "000", @@ -194,6 +199,7 @@ def version_by_type_name() -> Dict[str, str]: "flo.params": "000", "flo.params.brickstorageheater": "000", "flo.params.report": "000", + "flo.params.simpleresistivehydronic": "000", "g.node.gt": "002", "g.node.instance.gt": "000", "gt.dispatch.boolean": "110", @@ -262,11 +268,12 @@ def status_by_versioned_type_name() -> Dict[str, str]: """ v: Dict[str, str] = { - "accepted.bid.000": "Pending", - "atn.bid.001": "Pending", + "accepted.bid.000": "Active", + "atn.bid.001": "Active", "atn.params.000": "Active", - "atn.params.brickstorageheater.000": "Pending", + "atn.params.brickstorageheater.000": "Active", "atn.params.report.000": "Active", + "atn.params.simpleresistivehydronic.000": "Pending", "base.g.node.gt.002": "Active", "basegnode.scada.create.000": "Active", "component.attribute.class.gt.000": "Active", @@ -280,7 +287,8 @@ def status_by_versioned_type_name() -> Dict[str, str]: "electric.meter.component.gt.000": "Active", "flo.params.000": "Active", "flo.params.brickstorageheater.000": "Active", - "flo.params.report.000": "Pending", + "flo.params.report.000": "Active", + "flo.params.simpleresistivehydronic.000": "Pending", "g.node.gt.002": "Active", "g.node.instance.gt.000": "Active", "gt.dispatch.boolean.110": "Active", @@ -300,9 +308,9 @@ def status_by_versioned_type_name() -> Dict[str, str]: "initial.tadeed.algo.optin.002": "Active", "initial.tadeed.algo.transfer.000": "Active", "join.dispatch.contract.000": "Active", - "latest.price.000": "Pending", - "market.slot.000": "Pending", - "market.type.gt.000": "Pending", + "latest.price.000": "Active", + "market.slot.000": "Active", + "market.type.gt.000": "Active", "multipurpose.sensor.cac.gt.000": "Active", "multipurpose.sensor.component.gt.000": "Active", "new.tadeed.algo.optin.000": "Active", @@ -311,32 +319,32 @@ def status_by_versioned_type_name() -> Dict[str, str]: "pipe.flow.sensor.cac.gt.000": "Active", "pipe.flow.sensor.component.gt.000": "Active", "power.watts.000": "Active", - "price.quantity.000": "Pending", - "price.quantity.unitless.000": "Pending", + "price.quantity.000": "Active", + "price.quantity.unitless.000": "Active", "ready.001": "Active", "relay.cac.gt.000": "Active", "relay.component.gt.000": "Active", "resistive.heater.cac.gt.000": "Active", "resistive.heater.component.gt.000": "Active", - "scada.cert.transfer.000": "Pending", + "scada.cert.transfer.000": "Active", "sim.timestep.000": "Active", "simple.temp.sensor.cac.gt.000": "Active", "simple.temp.sensor.component.gt.000": "Active", - "simplesim.driver.data.000": "Pending", - "simplesim.driver.data.bsh.000": "Pending", - "simplesim.driver.report.000": "Pending", - "simplesim.snapshot.brickstorageheater.000": "Pending", - "sla.enter.000": "Pending", + "simplesim.driver.data.000": "Active", + "simplesim.driver.data.bsh.000": "Active", + "simplesim.driver.report.000": "Active", + "simplesim.snapshot.brickstorageheater.000": "Active", + "sla.enter.000": "Active", "snapshot.spaceheat.000": "Active", "spaceheat.node.gt.100": "Active", "super.starter.000": "Active", "supervisor.container.gt.000": "Active", - "tadeed.specs.hack.000": "Pending", + "tadeed.specs.hack.000": "Active", "tavalidatorcert.algo.create.000": "Active", "tavalidatorcert.algo.transfer.000": "Active", "telemetry.reporting.config.000": "Active", "telemetry.snapshot.spaceheat.000": "Active", - "terminalasset.certify.hack.000": "Pending", + "terminalasset.certify.hack.000": "Active", } return v diff --git a/src/gwatn/atn_actor_base.py b/src/gwatn/atn_actor_base.py index f23ebb3..1beebdc 100644 --- a/src/gwatn/atn_actor_base.py +++ b/src/gwatn/atn_actor_base.py @@ -49,7 +49,6 @@ from gwatn.types import LatestPrice_Maker from gwatn.types import PowerWatts from gwatn.types import SimTimestep -from gwatn.types import SimTimestep_Maker from gwatn.types import SnapshotSpaceheat @@ -85,10 +84,13 @@ def dummy_atn_params() -> AtnParams: class AtnActorBase(TwoChannelActorBase): def __init__(self, settings: AtnSettings, use_algo: bool = False): super().__init__(settings=settings) - self.settings: AtnSettings = settings self.scada_gni_id = settings.scada_gni_id self._time: float = self.get_initial_time_s() - + self.atn_params: AtnParams = dummy_atn_params() + self.dc_app_id: Optional[int] = None + self.dc_client: Optional[ApplicationClient] = None + self.trading_rights_id: Optional[GwCertId] = None + self.hb_status = HbStatus(LastHeartbeatReceivedMs=int(time.time() * 1000)) if use_algo is True: self.acct: BasicAccount = BasicAccount(settings.sk.get_secret_value()) self.client: AlgodClient = AlgodClient( @@ -100,24 +102,20 @@ def __init__(self, settings: AtnSettings, use_algo: bool = False): f"Insufficiently funded. Make sure atn has at least 5 algos" ) # TODO: move this into spaceheat along with join_dispatch_contract_received - self.atn_params: AtnParams = dummy_atn_params() + self.sp = self.client.suggested_params() self.sp.flat_fee = True self.sp.fee = 2000 # this is initialized with the AppId provided by the SCADA - self.dc_app_id: Optional[int] = None - self.dc_client: Optional[ApplicationClient] = None self.check_for_dispatch_contract() self.universe_type = as_enum( self.settings.universe_type_value, UniverseType, UniverseType.default() ) - self.trading_rights_id: Optional[GwCertId] = None self.update_trading_rights() - self.hb_status = HbStatus(LastHeartbeatReceivedMs=int(time.time() * 1000)) def local_rabbit_startup(self) -> None: rjb = MessageCategorySymbol.rjb.value - tc_alias_lrh = self.settings.time_coordinator_alias.replace(".", "-") + tc_alias_lrh = self.settings.my_time_coordinator_alias.replace(".", "-") binding = f"{rjb}.{tc_alias_lrh}.timecoordinator.sim-timestep" cb = functools.partial(self.on_timecoordinator_bindok, binding=binding) @@ -259,16 +257,7 @@ def route_message( self, from_alias: str, from_role: GNodeRole, payload: HeartbeatA ) -> None: self.payload = payload - if payload.TypeName == HeartbeatA_Maker.type_name: - if from_role != GNodeRole.Supervisor: - LOGGER.info( - f"Ignoring HeartbeatA from GNode with role {from_role}; expects Supervisor" - ) - try: - self.heartbeat_from_super(from_alias, payload) - except: - LOGGER.exception("Error in heartbeat_received") - elif payload.TypeName == HeartbeatB_Maker.type_name: + if payload.TypeName == HeartbeatB_Maker.type_name: if from_role != GNodeRole.Scada: LOGGER.info( f"Ignoring HeartbeatB from GNode with role {from_role}; expects Scada" @@ -292,26 +281,9 @@ def route_message( self.latest_price_from_market_maker(payload) except: LOGGER.exception("Error in latest_price_from_market_maker") - elif payload.TypeName == SimTimestep_Maker.type_name: - try: - self.timestep_from_timecoordinator(payload) - except: - LOGGER.exception("Error in timestep_from_timecoordinator") - - def heartbeat_from_super(self, from_alias: str, ping: HeartbeatA) -> None: - pong = HeartbeatA_Maker( - my_hex=str(random.choice("0123456789abcdef")), your_last_hex=ping.MyHex - ).tuple - - self.send_message( - payload=pong, - to_role=GNodeRole.Supervisor, - to_g_node_alias=self.settings.my_super_alias, - ) - - LOGGER.debug( - f"[{self.alias}] Sent HB: SuHex {pong.YourLastHex}, AtnHex {pong.MyHex}" - ) + else: + # If the message is not recognized, kick up to base class + super().route_message(from_alias, from_role, payload) def heartbeat_from_scada(self, ping: HeartbeatB) -> None: """ @@ -390,7 +362,7 @@ def turn_off(self, relay_node_name: str) -> None: [SpaceheatNode](https://gridworks-protocol.readthedocs.io/en/latest/spaceheat-node.html) [BooleanActuator Role](https://gridworks-protocol.readthedocs.io/en/latest/enums.html#gwproto.enums.Role) Args: - relay_node_name: the name of the relay, as string in LeftRightDot format. This must be the + relay_node_name (str): the name of the relay, as string in LeftRightDot format. This must be the name of a SpaceheatNode in the hardware layout with role "BooleanActuator" Returns: diff --git a/src/gwatn/atn_utils.py b/src/gwatn/atn_utils.py index 656536c..7759119 100644 --- a/src/gwatn/atn_utils.py +++ b/src/gwatn/atn_utils.py @@ -3,10 +3,13 @@ from gwatn import property_format from gwatn.data_classes import MarketType from gwatn.enums import MarketTypeName +from gwatn.types import AtnParams +from gwatn.types import FloParams from gwatn.types import MarketSlot from gwatn.types import MarketTypeGt_Maker +DUMMY_TERMINALASSET_ALIAS = "d1.isone.dummy.ta" DUMMY_ALGO_TXN = "gqNzaWfEQNPXbrAiWd+cNgsIaM3N0PSu3repauvmjuHmoKjh6sd3L5U4/YpovcXN7/ATH1LgcI4cgV+SU3VQ6bsm/gfAOQyjdHhuiaNhbXTNB9CjZmVlzQPoomZ2HaNnZW6qc2FuZG5ldC12MaJnaMQgaJXPYTdWaeTNSs8FMMzPNfV7SrHXqJgFsJLxRbSPjzCibHbNBAWjcmN2xCBWkH3PValty0Rb0cyZo69Alhp4IbNKFnhXtgJ++A9EzKNzbmTEIOJPEbccL6IqmeBeaLzbLav25U9jBMjloaIyF1eY9HFxpHR5cGWjcGF5" @@ -20,6 +23,18 @@ class CostAndQuantityBought(BaseModel): Cost: float +def is_dummy_atn_params(atn_params: AtnParams) -> bool: + if atn_params.GNodeAlias == DUMMY_TERMINALASSET_ALIAS: + return True + return False + + +def is_dummy_flo_params(flo_params: FloParams) -> bool: + if flo_params.GNodeAlias == DUMMY_TERMINALASSET_ALIAS: + return True + return False + + def name_from_market_slot(slot: MarketSlot) -> str: return f"{slot.Type.Name.value}.{slot.MarketMakerAlias}.{slot.StartUnixS}" diff --git a/src/gwatn/config.py b/src/gwatn/config.py index 9f1ca19..5bd32ca 100644 --- a/src/gwatn/config.py +++ b/src/gwatn/config.py @@ -11,26 +11,24 @@ class AtnSettings(GNodeSettings): + # Changing the default values for these GNodeSettings g_node_alias: str = "d1.isone.ver.keene.holly" g_node_role_value: str = "AtomicTNode" my_super_alias: str = "d1.isone.ver.keene.super1" g_node_id: str = "6bb37cc5-740d-40f5-a535-43987a5d07b4" - g_node_instance_id: str = "00000000-0000-0000-0000-000000000000" - scada_gni_id: Optional[str] = None sk: SecretStr = SecretStr( "K6iB3AHmzSQ8wDE91QdUfaheDMEtf2WJUMYeeRptKxHiTxG3HC+iKpngXmi82y2r9uVPYwTI5aGiMhdXmPRxcQ==" ) - # Public address 4JHRDNY4F6RCVGPALZULZWZNVP3OKT3DATEOLINCGILVPGHUOFY7KCHVIQ + # Secret key for public Algorand address 4JHRDNY4F6RCVGPALZULZWZNVP3OKT3DATEOLINCGILVPGHUOFY7KCHVIQ + # Additional settings specific to AtomicTNodes + scada_gni_id: Optional[str] = None # Next 4 settings are consistent with dev env settings in gridworks-marketmaker repo market_maker_alias = "d1.isone.ver.keene" market_maker_algo_address = ( "CYWMWYHJ7ON4IR5XQDJBBPDU472QU4KJQ6XQVZBIIRTCHT6SHTFNHEAVC4" ) mm_api_root = "http://localhost:7997" - initial_time_unix_s = pendulum.datetime( - year=2020, month=1, day=1, hour=4, minute=20 - ).int_timestamp class Config: env_prefix = "ATN_" diff --git a/src/gwatn/enums/__init__.py b/src/gwatn/enums/__init__.py index 108de4b..8943488 100644 --- a/src/gwatn/enums/__init__.py +++ b/src/gwatn/enums/__init__.py @@ -27,10 +27,6 @@ # From gwatn from gwatn.enums.distribution_tariff import DistributionTariff from gwatn.enums.energy_supply_type import EnergySupplyType -from gwatn.enums.hack_price_method import PriceMethod -from gwatn.enums.hack_recognized_p_node_alias import RecognizedPNodeAlias -from gwatn.enums.hack_weather_method import WeatherMethod -from gwatn.enums.hack_weather_source import WeatherSource from gwatn.enums.recognized_irradiance_type import RecognizedIrradianceType from gwatn.enums.recognized_temperature_unit import RecognizedTemperatureUnit @@ -51,10 +47,8 @@ "MarketTypeName", "MessageCategory", "MessageCategorySymbol", - "PriceMethod", "RecognizedCurrencyUnit", "RecognizedIrradianceType", - "RecognizedPNodeAlias", "RecognizedTemperatureUnit", "Role", "StrategyName", @@ -62,6 +56,4 @@ "TelemetryName", "Unit", "UniverseType", - "WeatherMethod", - "WeatherSource", ] diff --git a/src/gwatn/enums/distribution_tariff.py b/src/gwatn/enums/distribution_tariff.py index 83d7f2d..9046fb6 100644 --- a/src/gwatn/enums/distribution_tariff.py +++ b/src/gwatn/enums/distribution_tariff.py @@ -11,13 +11,15 @@ class DistributionTariff(StrEnum): Choices and descriptions: * Unknown: - * VersantStorageHeatTariff: - * VersantATariff: + * VersantA1StorageHeatTariff: Versant is a utility serving customers in Maine, and in particular serves much of the area behind the Keene Rd Constraint in the [GridWorks Millinocket Demo](https://gridworks.readthedocs.io/en/latest/millinocket-demo.html#background). Alternately known as the "Home Eco Rate With Bonus Meter, Time-of-Use". Look for rate A1 in Versant [rate schedules](https://www.versantpower.com/residential/rates/rates-schedules/); details are also available [here](https://drive.google.com/drive/u/0/folders/1mhIeNj2JWVyIJrQnSHmBDOkBpNnRRVKB). More: Service under this rate will be available to residential customers with thermal energy storage devices, electric battery storage devices, and/or vehicle chargers who agree to install a second metered point of delivery. The customer will be subject to inspections to ensure that the thermal storage device, electric battery storage device, and electric vehicle charger(s) are sized appropriately for residential use. If the thermal storage device, electric battery storage device, and electric vehicle charger(s) do not pass Company inspection, then the service will be denied. Service will be single-phase, alternating current, 60 hertz, at one standard secondary distribution voltage. Customers taking service under this rate schedule are responsible for paying both Distribution Service and Stranded Cost.. [More Info](https://github.com/thegridelectric/gridworks-ps/blob/dev/input_data/electricity_prices/isone/distp__w.isone.stetson__2020__gw.me.versant.a1.res.ets.csv). + * VersantATariff: Versant is a utility serving customers in Maine, and in particular serves much of the area behind the Keene Rd Constraint in the [GridWorks Millinocket Demo](https://gridworks.readthedocs.io/en/latest/millinocket-demo.html#background). The A Tariff is their standard residential tariff. Look for rate A in Versant [rate schedules](https://www.versantpower.com/residential/rates/rates-schedules/) + * VersantA20HeatTariff: """ Unknown = auto() - VersantStorageHeatTariff = auto() + VersantA1StorageHeatTariff = auto() VersantATariff = auto() + VersantA20HeatTariff = auto() @classmethod def default(cls) -> "DistributionTariff": diff --git a/src/gwatn/simple_atn_actor.py b/src/gwatn/simple_atn_actor.py index df2b684..a5d14bf 100644 --- a/src/gwatn/simple_atn_actor.py +++ b/src/gwatn/simple_atn_actor.py @@ -6,9 +6,11 @@ from gwproto.messages import PeerActiveEvent from gwproto.messages import Ping as GridworksPing +import gwatn.atn_utils as atn_utils import gwatn.config as config from gwatn.atn_actor_base import AtnActorBase from gwatn.enums import TelemetryName +from gwatn.types import AtnParams from gwatn.types import GtShStatus from gwatn.types import LatestPrice from gwatn.types import PowerWatts @@ -33,6 +35,7 @@ def __init__( ), ): super().__init__(settings=settings) + self.atn_params = AtnParams(GNodeAlias=atn_utils.DUMMY_TERMINALASSET_ALIAS) self._power_watts: int = 0 LOGGER.info("Simple Atn Initialized") diff --git a/src/gwatn/strategies/__init__.py b/src/gwatn/strategies/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/src/gwatn/brick_storage_heater/atn.py b/src/gwatn/strategies/brick_storage_heater/atn.py similarity index 97% rename from src/gwatn/brick_storage_heater/atn.py rename to src/gwatn/strategies/brick_storage_heater/atn.py index e342237..d0b197a 100644 --- a/src/gwatn/brick_storage_heater/atn.py +++ b/src/gwatn/strategies/brick_storage_heater/atn.py @@ -15,6 +15,7 @@ import time import uuid from typing import Optional +from typing import cast from typing import no_type_check import dotenv @@ -41,7 +42,8 @@ from gwatn.enums import MessageCategorySymbol from gwatn.enums import UniverseType from gwatn.types import AcceptedBid_Maker -from gwatn.types import AtnParamsBrickstorageheater as AtnParams +from gwatn.types import AtnParams +from gwatn.types import AtnParamsBrickstorageheater from gwatn.types import AtnParamsReport_Maker from gwatn.types import HeartbeatA from gwatn.types import HeartbeatA_Maker @@ -82,7 +84,9 @@ def __init__( settings.public.algod_address, ) if self.universe_type == UniverseType.Dev: - self.atn_params: AtnParams = strategy_utils.dummy_atn_params() + self.atn_params: AtnParamsBrickstorageheater = ( + strategy_utils.dummy_atn_params() + ) else: self.get_initial_params() self.market_type: MarketTypeGt = MarketTypeGt_Maker.dc_to_tuple(Rt60Gate30B) @@ -148,7 +152,7 @@ def new_timestep(self, payload: SimTimestep) -> None: # LOGGER.info("----------------------------------------------------") # This gets called on the first timestep - if strategy_utils.is_dummy_atn_params(self.atn_params): + if atn_utils.is_dummy_atn_params(self.atn_params): self.get_initial_params() LOGGER.info("Correcting runs on initial timestep") self.active_run = strategy_utils.dummy_slot_stuff( @@ -422,13 +426,13 @@ def submit_bid(self, slot_stuff: SlotStuff) -> RestfulResponse: ######################## def dev_get_initial_params(self) -> None: - if not strategy_utils.is_dummy_atn_params(self.atn_params): + if not atn_utils.is_dummy_atn_params(cast(AtnParams, self.atn_params)): LOGGER.warning("Tried to get initial params but already have them") now_ms = int(self.time()) * 1000 self.atn_params = dev_io.atn_params_from_alias(alias=self.alias, now_ms=now_ms) self.atn_params.GNodeInstanceId = self.g_node_instance_id - if strategy_utils.is_dummy_atn_params(self.atn_params): + if atn_utils.is_dummy_atn_params(cast(AtnParams, self.atn_params)): LOGGER.warning(f"No atn_params for {self.alias} before {self.time_str()}") return payload = AtnParamsReport_Maker( @@ -436,7 +440,7 @@ def dev_get_initial_params(self) -> None: g_node_instance_id=self.g_node_instance_id, atn_params_type_name=self.atn_params.TypeName, time_unix_s=int(self.time()), - params=self.atn_params, + params=cast(AtnParams, self.atn_params), irl_time_unix_s=int(time.time()), ).tuple diff --git a/src/gwatn/brick_storage_heater/dev_atn_params_data.csv b/src/gwatn/strategies/brick_storage_heater/dev_atn_params_data.csv similarity index 100% rename from src/gwatn/brick_storage_heater/dev_atn_params_data.csv rename to src/gwatn/strategies/brick_storage_heater/dev_atn_params_data.csv diff --git a/src/gwatn/brick_storage_heater/dev_io.py b/src/gwatn/strategies/brick_storage_heater/dev_io.py similarity index 100% rename from src/gwatn/brick_storage_heater/dev_io.py rename to src/gwatn/strategies/brick_storage_heater/dev_io.py diff --git a/src/gwatn/brick_storage_heater/edge.py b/src/gwatn/strategies/brick_storage_heater/edge.py similarity index 100% rename from src/gwatn/brick_storage_heater/edge.py rename to src/gwatn/strategies/brick_storage_heater/edge.py diff --git a/src/gwatn/brick_storage_heater/flo.py b/src/gwatn/strategies/brick_storage_heater/flo.py similarity index 100% rename from src/gwatn/brick_storage_heater/flo.py rename to src/gwatn/strategies/brick_storage_heater/flo.py diff --git a/src/gwatn/brick_storage_heater/flo_output.py b/src/gwatn/strategies/brick_storage_heater/flo_output.py similarity index 100% rename from src/gwatn/brick_storage_heater/flo_output.py rename to src/gwatn/strategies/brick_storage_heater/flo_output.py diff --git a/src/gwatn/brick_storage_heater/make_dev_input_data.py b/src/gwatn/strategies/brick_storage_heater/make_dev_input_data.py similarity index 100% rename from src/gwatn/brick_storage_heater/make_dev_input_data.py rename to src/gwatn/strategies/brick_storage_heater/make_dev_input_data.py diff --git a/src/gwatn/brick_storage_heater/node.py b/src/gwatn/strategies/brick_storage_heater/node.py similarity index 100% rename from src/gwatn/brick_storage_heater/node.py rename to src/gwatn/strategies/brick_storage_heater/node.py diff --git a/src/gwatn/brick_storage_heater/strategy_utils.py b/src/gwatn/strategies/brick_storage_heater/strategy_utils.py similarity index 70% rename from src/gwatn/brick_storage_heater/strategy_utils.py rename to src/gwatn/strategies/brick_storage_heater/strategy_utils.py index bf35222..1c06042 100644 --- a/src/gwatn/brick_storage_heater/strategy_utils.py +++ b/src/gwatn/strategies/brick_storage_heater/strategy_utils.py @@ -1,18 +1,23 @@ from typing import List from typing import Optional +from typing import cast from pydantic import BaseModel +from gwatn.atn_utils import DUMMY_TERMINALASSET_ALIAS +from gwatn.atn_utils import is_dummy_flo_params from gwatn.brick_storage_heater.flo import Flo__BrickStorageHeater as Flo from gwatn.types import AtnBid from gwatn.types import AtnParamsBrickstorageheater as AtnParams -from gwatn.types import FloParamsBrickstorageheater as FloParams +from gwatn.types import AtnParamsBrickstorageheater_Maker as AtnParams_Maker +from gwatn.types import FloParams +from gwatn.types import FloParamsBrickstorageheater from gwatn.types import MarketSlot class SlotStuff(BaseModel): Slot: MarketSlot - BidParams: Optional[FloParams] = None + BidParams: Optional[FloParamsBrickstorageheater] = None Flo: Optional[Flo] = None Bid: Optional[AtnBid] = None Price: Optional[float] = None @@ -31,36 +36,19 @@ def dummy_slot_stuff(slot: MarketSlot) -> SlotStuff: def is_dummy_slot_stuff(bid_stuff: SlotStuff) -> bool: - if is_dummy_flo_params(bid_stuff.BidParams): - return True - return False - - -def is_dummy_atn_params(atn_params: AtnParams) -> bool: - if atn_params.GNodeAlias == "d1.isone.dummy.ta": + if is_dummy_flo_params(cast(FloParams, bid_stuff.BidParams)): return True return False def dummy_atn_params() -> AtnParams: - return AtnParams( - SliceDurationMinutes=60, - FloSlices=48, - GNodeAlias="d1.isone.dummy.ta", - GNodeInstanceId="00000000-0000-0000-0000-000000000000", - TypeName="atn.params.heatpumpwithbooststore", - Version="000", - ) - - -def is_dummy_flo_params(flo_params: FloParams) -> bool: - if flo_params.RtElecPriceUid == "00000000-0000-0000-0000-000000000000": - return True - return False + return AtnParams(GNodeAlias=DUMMY_TERMINALASSET_ALIAS) -def dummy_flo_params() -> FloParams: - return FloParams( +def dummy_flo_params() -> FloParamsBrickstorageheater: + return FloParamsBrickstorageheater( + GNodeAlias=DUMMY_TERMINALASSET_ALIAS, + FloParamsUid="00000000-0000-0000-0000-000000000000", RtElecPriceUid="00000000-0000-0000-0000-000000000000", WeatherUid="00000000-0000-0000-0000-000000000000", ) diff --git a/src/gwatn/strategies/simple_resistive_hydronic/atn.py b/src/gwatn/strategies/simple_resistive_hydronic/atn.py new file mode 100644 index 0000000..d0b197a --- /dev/null +++ b/src/gwatn/strategies/simple_resistive_hydronic/atn.py @@ -0,0 +1,512 @@ +""" BrickStorageHeater AtomicTNode Strategy. + + This is a heating and bidding strategy for a thermal storage heater that is + designed to heat part or all of a single room. The storage medium is ceramic + bricks and the heating source is resistive elements embedded in those bricks. + + This kind of heater is often called a Night Storage Heater in the UK, and + is often also referred to (ambigiously, since there are other kinds) as + an Electric Thermal Storage (ETS) heater. + """ +import functools +import logging +import math +import random +import time +import uuid +from typing import Optional +from typing import cast +from typing import no_type_check + +import dotenv +import gridworks.algo_utils as algo_utils +import pendulum +import requests +from algosdk import encoding +from algosdk.future import transaction +from algosdk.v2client.algod import AlgodClient +from gridworks.algo_utils import BasicAccount +from gridworks.data_classes.market_type import Rt60Gate30B +from gridworks.utils import RestfulResponse + +import gwatn.atn_utils as atn_utils +import gwatn.brick_storage_heater.dev_io as dev_io +import gwatn.brick_storage_heater.strategy_utils as strategy_utils +import gwatn.config as config +from gwatn.atn_actor_base import AtnActorBase +from gwatn.brick_storage_heater.edge import Edge__BrickStorageHeater as Edge +from gwatn.brick_storage_heater.flo import Flo__BrickStorageHeater as Flo +from gwatn.brick_storage_heater.strategy_utils import SlotStuff +from gwatn.enums import GNodeRole +from gwatn.enums import MessageCategory +from gwatn.enums import MessageCategorySymbol +from gwatn.enums import UniverseType +from gwatn.types import AcceptedBid_Maker +from gwatn.types import AtnParams +from gwatn.types import AtnParamsBrickstorageheater +from gwatn.types import AtnParamsReport_Maker +from gwatn.types import HeartbeatA +from gwatn.types import HeartbeatA_Maker +from gwatn.types import LatestPrice +from gwatn.types import MarketSlot +from gwatn.types import MarketTypeGt +from gwatn.types import MarketTypeGt_Maker +from gwatn.types import PriceQuantityUnitless +from gwatn.types import Ready_Maker +from gwatn.types import SimplesimSnapshotBrickstorageheater_Maker as Snapshot_Maker +from gwatn.types import SimTimestep + + +LOG_FORMAT = ( + "%(levelname) -10s %(asctime)s %(name) -30s %(funcName) " + "-35s %(lineno) -5d: %(message)s" +) +LOGGER = logging.getLogger(__name__) + +LOGGER.setLevel(logging.WARNING) + + +class Atn__BrickStorageHeater(AtnActorBase): + """AtomicTNode HeatPumpWithBoostStore strategy for thermal storage heat pump + space heating system""" + + def __init__( + self, + settings: config.AtnSettings = config.AtnSettings( + _env_file=dotenv.find_dotenv() + ), + ): + super().__init__(settings) + LOGGER.info("Initializing HeatPumpWithBoostStore Atn") + self.algo_acct: BasicAccount = BasicAccount(settings.sk.get_secret_value()) + self.algo_client: AlgodClient = AlgodClient( + settings.algo_api_secrets.algod_token.get_secret_value(), + settings.public.algod_address, + ) + if self.universe_type == UniverseType.Dev: + self.atn_params: AtnParamsBrickstorageheater = ( + strategy_utils.dummy_atn_params() + ) + else: + self.get_initial_params() + self.market_type: MarketTypeGt = MarketTypeGt_Maker.dc_to_tuple(Rt60Gate30B) + self.active_run: SlotStuff = strategy_utils.dummy_slot_stuff( + slot=self.active_slot(self.market_type) + ) + self.next_run: SlotStuff = strategy_utils.dummy_slot_stuff( + slot=self.next_slot(self.market_type) + ) + self.store_kwh: float = 0 + self._power_watts: Optional[int] = None + self.store_update_time: float = self.time() + self.latest_price_received_time: float = 0 + self.latest_price_dollars_per_mwh: float = 10**6 + self._first_start_finished: bool = False + + @no_type_check + def strategy_rabbit_startup(self) -> None: + mm_alias_lrh = self.settings.market_maker_alias.replace(".", "-") + rjb = MessageCategorySymbol.rjb.value + binding = f"{rjb}.{mm_alias_lrh}.marketmaker.latest-price.{self.market_type.Name.value}" + cb = functools.partial(self.on_marketprice_bindok, binding=binding) + self._consume_channel.queue_bind( + self.queue_name, "marketmakermic_tx", routing_key=binding, callback=cb + ) + + pong = HeartbeatA_Maker( + my_hex=str(random.choice("0123456789abcdef")), your_last_hex="0" + ).tuple + self.send_message( + payload=pong, + to_role=GNodeRole.Supervisor, + to_g_node_alias=self.settings.my_super_alias, + ) + d = pendulum.from_timestamp(time.time()) + LOGGER.warning( + f"[{self.short_alias}] Sent first heartbeat to super: {d.minute}:{d.second}.{d.microsecond}" + ) + self._first_start_finished = True + + @no_type_check + def on_marketprice_bindok(self, _unused_frame, binding) -> None: + LOGGER.info(f"Queue {self.queue_name} bound with {binding}") + + def heartbeat_received(self, from_alias: str, ping: HeartbeatA): + pong = HeartbeatA_Maker( + my_hex=str(random.choice("0123456789abcdef")), your_last_hex=ping.MyHex + ).tuple + + self.send_message( + payload=pong, + to_role=GNodeRole.Supervisor, + to_g_node_alias=self.settings.my_super_alias, + ) + + LOGGER.debug( + f"[{self.short_alias}] Sent HB: SuHex {pong.YourLastHex}, AtnHex {pong.MyHex}" + ) + + def new_timestep(self, payload: SimTimestep) -> None: + # LOGGER.info("----------------------------------------------------") + # LOGGER.info(f"[{self.time_str()}: {self.short_alias}] NEW TIMESTEP") + # LOGGER.info("----------------------------------------------------") + + # This gets called on the first timestep + if atn_utils.is_dummy_atn_params(self.atn_params): + self.get_initial_params() + LOGGER.info("Correcting runs on initial timestep") + self.active_run = strategy_utils.dummy_slot_stuff( + slot=self.last_slot(self.market_type), + ) + try: + self.next_run = dev_io.initialize_slot_stuff( + slot=self.active_slot(self.market_type), + atn_params=self.atn_params, + atn_gni_id=self.g_node_instance_id, + ) + except Exception as e: + LOGGER.warning( + f"Failed to initialize FloParams! Not creating a next_run {e}" + ) + return + + try: + self.update_store_level() + except Exception: + LOGGER.exception("Error in update_store_level") + + # Note: compressing the bid and the market start into the same time, + # since time steps are only once per slot right now + if self.time() >= self.next_run.Slot.StartUnixS: + self.active_run = self.next_run + self.active_run.BidParams.StartingStoreIdx = self.store_idx() + try: + self.get_bid() + except Exception: + LOGGER.exception("Error in get_bid") + self.next_run = dev_io.initialize_slot_stuff( + slot=self.next_slot(self.market_type), + atn_params=self.atn_params, + atn_gni_id=self.g_node_instance_id, + ) + try: + self.submit_bid(self.active_run) + except Exception: + LOGGER.exception("Error in submit_bid") + if self.active_run.Price is not None: + self.respond_to_price() + + def repeat_timestep(self, payload: SimTimestep) -> None: + LOGGER.info(f"[{self.time_str()}: {self.short_alias}] Timestep received again") + + def latest_price_from_market_maker(self, payload: LatestPrice) -> None: + self.mm_payload = payload + + if payload.FromGNodeAlias != self.settings.market_maker_alias: + LOGGER.warning( + f"Received price from {payload.FromGNodeAlias}." + f" Expected {self.settings.market_maker_alias} " + ) + return + slot = atn_utils.market_slot_from_name(payload.MarketSlotName) + + # time_str = pendulum.from_timestamp(slot.StartUnixS).strftime("%m/%d/%Y, %H:%M") + # LOGGER.info("----------------------------------------------------") + # LOGGER.info( + # f"[{self.time_str()}: {self.short_alias}] LATEST PRICE From MarketMaker: " + # f"${payload.PriceTimes1000 / 1000}/Mwh for slot starting {time_str}" + # ) + # LOGGER.info("----------------------------------------------------") + if slot.Type != self.market_type: + LOGGER.warning( + f"Received price for {slot.Type}, only participate in {self.market_type}" + ) + return + + if slot == self.active_run.Slot: + # LOGGER.info("Received price from MarketMaker. Adding price to active run") + self.active_run.Price = payload.PriceTimes1000 / 1000 + elif slot == self.next_run.Slot: + # this happens when the price arrives before the timestep + # LOGGER.info("Received price from MarketMaker. Adding price to next run") + self.next_run.Price = payload.PriceTimes1000 / 1000 + else: + raise Exception("HUH. did not add price to a run") + + if self.active_run.Price is not None: + self.respond_to_price() + + def respond_to_price(self) -> None: + try: + self.update_power_levels() + except Exception: + LOGGER.exception("Error in update_power_levels") + payload = Snapshot_Maker( + from_g_node_alias=self.alias, + from_g_node_instance_id=self.g_node_instance_id, + power_watts=self._power_watts, + store_kwh=int(self.store_kwh), + max_store_kwh=int( + strategy_utils.get_max_store_kwh_th( + max_brick_temp_c=self.atn_params.MaxBrickTempC, + c=self.atn_params.C, + room_temp_f=self.atn_params.RoomTempF, + ) + ), + about_terminal_asset_alias=self.alias + ".ta", + ).tuple + # pprint(payload) + self.send_message( + payload=payload, message_category=MessageCategory.RabbitJsonBroadcast + ) + + payload = Ready_Maker( + from_g_node_alias=self.alias, + from_g_node_instance_id=self.g_node_instance_id, + time_unix_s=int(self.time()), + ).tuple + self.send_message( + payload=payload, + to_role=GNodeRole.TimeCoordinator, + to_g_node_alias=self.settings.time_coordinator_alias, + ) + + ######################### + # Terminal Asset State + ######################### + + def update_power_levels(self) -> None: + # use price and Flo results to figure out heatpump and boost levels + if self.active_run.Flo is None: + return + node = self.active_run.Flo.node[0][self.active_run.Flo.starting_store_idx] + edge: Edge = list( + filter( + lambda x: x.end_idx == node.best_next_idx, + self.active_run.Flo.edges[node], + ) + )[0] + self._power_watts = edge.avg_kw * 1000 + + # LOGGER.info( + # f"[{self.time_str()}: {self.short_alias}] Updating state: heatpump {round(self.heatpump_kw,2)}" + # ) + + def update_store_level(self) -> None: + if strategy_utils.is_dummy_slot_stuff(self.active_run): + LOGGER.info(f"Not updating storage - active run is a dummy") + self.store_update_time = self.time() + return + duration_minutes = int((self.time() - self.store_update_time) / 60) + if duration_minutes != self.market_type.DurationMinutes: + raise Exception( + f"In update_store_level: Expected {self.market_type.DurationMinutes} minutes, got {duration_minutes} minutes." + ) + flo = self.active_run.Flo + if flo is None: + raise Exception( + f"In update_store_level: Expected Flo for last_run {self.active_run}!" + ) + # TODO: find next step in graph based on the Flo, the Bid, and the actual price + if flo.starting_store_idx != self.store_idx(): + raise Exception( + f"flo.starting_idx is {flo.starting_store_idx} but self.store_idx is {self.store_idx()}!" + ) + try: + store_idx = flo.node[0][flo.starting_store_idx].best_next_idx + except: + raise Exception(f"Flo node did not have best_node_idx!") + # self.store_kwh = ... + self.store_kwh = ( + store_idx + * strategy_utils.get_max_store_kwh_th( + max_brick_temp_c=self.atn_params.MaxBrickTempC, + c=self.atn_params.C, + room_temp_f=self.atn_params.RoomTempF, + ) + / self.atn_params.StorageSteps + ) + # LOGGER.info( + # f"[{self.time_str()}: {self.short_alias}] Storage level {round(self.store_kwh,2)} kWh, " + # f"StoreIdx {self.active_run.BidParams.StartingStoreIdx}" + # ) + self.store_update_time = self.time() + + ######################## + # FLO + ######################## + + def get_bid(self): + self.active_run.Flo = Flo( + params=self.active_run.BidParams, d_graph_id=str(uuid.uuid4()) + ) + slot_start = self.active_run.Slot.StartUnixS + time_str = pendulum.from_timestamp(slot_start).strftime("%m/%d/%Y, %H:%M") + # LOGGER.info(f"[{self.time_str()}: {self.short_alias}] Solving Flo ") + self.active_run.Flo.solve_dijkstra() + # export_xlsx(alias=self.alias, flo=flo, atn_params=self.atn_params) + node = self.active_run.Flo.node[0][self.active_run.Flo.starting_store_idx] + edge: Edge = list( + filter( + lambda x: x.end_idx == node.best_next_idx, + self.active_run.Flo.edges[node], + ) + )[0] + q = edge.avg_kw + pq = PriceQuantityUnitless( + PriceTimes1000=int( + self.active_run.Flo.params.RealtimeElectricityPrice[0] * 1000 + ), + QuantityTimes1000=int(q * 1000), + ) + bid_list = [pq] + self.active_run.Bid.BidList = bid_list + + ######################## + # Market transactions + ###################### + + def pay_market_fee(self) -> str: + market_fee_micro_algos = 2 * 10**3 + txn = transaction.PaymentTxn( + sender=self.algo_acct.addr, + receiver=self.settings.market_maker_algo_address, + amt=market_fee_micro_algos, + sp=self.algo_client.suggested_params(), + ) + signed_txn = txn.sign(self.algo_acct.sk) + try: + self.algo_client.send_transaction(signed_txn) + except Exception as e: + note = f"Algorand Failure sending market payment transaction: {e}" + LOGGER.info(note) + return atn_utils.DUMMY_ALGO_TXN + try: + algo_utils.wait_for_transaction(self.algo_client, signed_txn.get_txid()) + except: + return atn_utils.DUMMY_ALGO_TXN + return encoding.msgpack_encode(signed_txn) + + def submit_bid(self, slot_stuff: SlotStuff) -> RestfulResponse: + slot_stuff.Bid.SignedMarketFeeTxn = self.pay_market_fee() + slot_start = slot_stuff.Slot.StartUnixS + time_str = pendulum.from_timestamp(slot_start).strftime("%m/%d/%Y, %H:%M") + # LOGGER.info( + # f"[{self.time_str()}: {self.short_alias}] Submitting bid to MarketMaker RestAPI for Slot starting {time_str}" + # ) + + api_endpoint = f"{self.settings.mm_api_root}/atn-bid/" + r = requests.post(url=api_endpoint, json=slot_stuff.Bid.as_dict()) + + # Duplicate on rabbit broker for ear + self.send_message( + payload=slot_stuff.Bid, + to_role=GNodeRole.MarketMaker, + to_g_node_alias=self.settings.market_maker_alias, + ) + + if r.status_code > 200: + if r.status_code == 422: + note = f"Error entering SLA: " + r.json()["detail"] + else: + note = r.reason + return RestfulResponse(Note=note, HttpStatusCode=422) + else: + rr = RestfulResponse(**r.json()) + if rr.PayloadTypeName == AcceptedBid_Maker.type_name: + ab = AcceptedBid_Maker.dict_to_tuple(rr.PayloadAsDict) + self.send_message( + payload=ab, message_category=MessageCategory.RabbitJsonBroadcast + ) + # pprint(rr) + return rr + + ######################## + # Initialization + ######################## + + def dev_get_initial_params(self) -> None: + if not atn_utils.is_dummy_atn_params(cast(AtnParams, self.atn_params)): + LOGGER.warning("Tried to get initial params but already have them") + + now_ms = int(self.time()) * 1000 + self.atn_params = dev_io.atn_params_from_alias(alias=self.alias, now_ms=now_ms) + self.atn_params.GNodeInstanceId = self.g_node_instance_id + if atn_utils.is_dummy_atn_params(cast(AtnParams, self.atn_params)): + LOGGER.warning(f"No atn_params for {self.alias} before {self.time_str()}") + return + payload = AtnParamsReport_Maker( + g_node_alias=self.alias, + g_node_instance_id=self.g_node_instance_id, + atn_params_type_name=self.atn_params.TypeName, + time_unix_s=int(self.time()), + params=cast(AtnParams, self.atn_params), + irl_time_unix_s=int(time.time()), + ).tuple + + self.send_message( + payload=payload, message_category=MessageCategory.RabbitJsonBroadcast + ) + # LOGGER.info(f"[{self.time_str()}: {self.short_alias}] Initial params loaded") + # pprint(self.atn_params) + + def get_initial_params(self) -> None: + if self.universe_type == UniverseType.Dev: + self.dev_get_initial_params() + else: + raise NotImplementedError + + @property + def short_alias(self) -> str: + return self.alias.split(".")[-1] + + ################## + # Market slot starts + ################## + + def last_slot(self, market_type: MarketTypeGt) -> MarketSlot: + start_s = self.active_slot( + market_type + ).StartUnixS - self.market_slot_duration_s(market_type) + return MarketSlot( + Type=market_type, + MarketMakerAlias=self.active_slot(market_type).MarketMakerAlias, + StartUnixS=start_s, + ) + + def active_slot(self, market_type: MarketTypeGt) -> MarketSlot: + market_delta = market_type.DurationMinutes * 60 + start = math.floor(self.time() / market_delta) * market_delta + return MarketSlot( + Type=market_type, + MarketMakerAlias=self.settings.market_maker_alias, + StartUnixS=start, + ) + + def next_slot(self, market_type: MarketTypeGt) -> MarketSlot: + start_s = self.active_slot( + market_type + ).StartUnixS + self.market_slot_duration_s(market_type) + return MarketSlot( + Type=market_type, + MarketMakerAlias=self.active_slot(market_type).MarketMakerAlias, + StartUnixS=start_s, + ) + + def market_slot_duration_s(self, market_type: MarketTypeGt) -> int: + return market_type.DurationMinutes * 60 + + ############# + # TerminalAsset properties + ################# + + def store_idx(self) -> int: + return round( + self.atn_params.StorageSteps + * self.store_kwh + / strategy_utils.get_max_store_kwh_th( + max_brick_temp_c=self.atn_params.MaxBrickTempC, + c=self.atn_params.C, + room_temp_f=self.atn_params.RoomTempF, + ) + ) diff --git a/src/gwatn/strategies/simple_resistive_hydronic/dev_io.py b/src/gwatn/strategies/simple_resistive_hydronic/dev_io.py new file mode 100644 index 0000000..8027b88 --- /dev/null +++ b/src/gwatn/strategies/simple_resistive_hydronic/dev_io.py @@ -0,0 +1,242 @@ +import csv +import json +import logging +import os +import typing +import uuid +from typing import List +from typing import Optional + +import pendulum +import satn.strategies.heatpumpwithbooststore.strategy_utils as strategy_utils +from pydantic import BaseModel +from satn.strategies.heatpumpwithbooststore.strategy_utils import SlotStuff +from satn.types import AtnParamsHeatpumpwithbooststore as AtnParams +from satn.types import AtnParamsHeatpumpwithbooststore_Maker as AtnParams_Maker +from satn.types import FloParamsHeatpumpwithbooststore as FloParams + +import gwatn.atn_utils as atn_utils +from gwatn.types import AtnBid +from gwatn.types import AtnParamsReport_Maker +from gwatn.types import MarketSlot +from gwatn.types.ps_distprices_gnode.csv_distp_sync.csv_distp_sync_1_0_0 import ( + Csv_Distp_Sync_1_0_0, +) +from gwatn.types.ps_electricityprices_gnode.csv_eprt_sync.csv_eprt_sync_1_0_0 import ( + Csv_Eprt_Sync_1_0_0, +) +from gwatn.types.ws_forecast_gnode.csv_weather_forecast_sync.csv_weather_forecast_sync_1_0_0 import ( + Csv_Weather_Forecast_Sync_1_0_0, +) + + +LOG_FORMAT = ( + "%(levelname) -10s %(asctime)s %(name) -30s %(funcName) " + "-35s %(lineno) -5d: %(message)s" +) +LOGGER = logging.getLogger(__name__) + +LOGGER.setLevel(logging.WARNING) +DATA_DIR = "input_data" +EVENTSTORE_DIR = f"{DATA_DIR}/eventstore" + +ELEC_PRICE_FILE = "input_data/electricity_prices/isone/eprt__w.isone.stetson__2020.csv" +DIST_PRICE_FILE = "input_data/electricity_prices/isone/distp__w.isone.stetson__2020__gw.me.versant.a1.res.ets.csv" +WEATHER_PRICE_FILE = ( + "input_data/weather/us/me/temp__ws.us.me.millinocketairport__2020.csv" +) +HEAT_PROFILE_FILE = "input_data/dev_heat_profile_data.csv" + + +class FileNameMeta(BaseModel): + FromGNodeAlias: str + TypeName: str + UnixTimeMs: int + FileName: str + + +def fn_from_file_name(file: str) -> Optional[FileNameMeta]: + try: + fn = FileNameMeta( + FromGNodeAlias=file.split("-")[0].strip(), + TypeName=file.split("-")[1].strip(), + UnixTimeMs=int(file.split("-")[2].split(".")[0]), + FileName=file, + ) + except Exception: + LOGGER.info(f"Ignoring incorectly formatted file {file}") + return None + return fn + + +def atn_params_from_alias(alias: str, now_ms: int) -> AtnParams: + files = os.listdir(EVENTSTORE_DIR) + fns: List[FileNameMeta] = [] + for file in files: + fn = fn_from_file_name(file) + if fn: + fns.append(fn) + mine: List[FileNameMeta] = list(filter(lambda x: x.FromGNodeAlias == alias, fns)) + mine = list(filter(lambda x: x.TypeName == AtnParamsReport_Maker.type_name, mine)) + mine = list(filter(lambda x: x.UnixTimeMs <= now_ms, mine)) + mine.sort(key=lambda x: x.UnixTimeMs, reverse=True) + if len(mine) == 0: + return strategy_utils.dummy_atn_params() + file_name = mine[0].FileName + with open(f"{EVENTSTORE_DIR}/{file_name}") as f: + data = json.load(f) + if "AtnParamsTypeName" not in data.keys(): + raise Exception(f"{file_name} is not an AtnParamsReport") + if data["AtnParamsTypeName"] != "atn.params.heatpumpwithbooststore": + raise Exception( + f"{file_name} doesn't have AtnParams for Heatpumpwithbooststore" + ) + params = AtnParams_Maker.dict_to_tuple(data["Params"]) + params_report = AtnParamsReport_Maker.dict_to_tuple(data) + params_report.Params = params + if params_report.GNodeAlias != alias: + raise Exception( + f"file {file_name} was for {params_report.GNodeAlias}", + f"instead of {alias}", + ) + if params_report.AtnParamsTypeName != "atn.params.heatpumpwithbooststore": + raise Exception( + f"Expects atn.params.heatpumpwithbooststore, got {params_report.AtnParamsTypeName}" + ) + + return typing.cast(AtnParams, params_report.Params) + + +def get_flo_params( + atn_params: AtnParams, + slot: MarketSlot, + storage_idx=50, +) -> FloParams: + alias = atn_params.GNodeAlias + start_s = slot.StartUnixS + slices = atn_params.FloSlices + slice_duration_minutes = atn_params.SliceDurationMinutes + + defaults = [] + with open(f"{HEAT_PROFILE_FILE}") as f: + reader = csv.reader(f) + for row in reader: + defaults.append(row) + aliases: List[str] = defaults[10] + try: + idx: int = aliases.index(alias) + except ValueError: + LOGGER.warning( + f"No heat profile data found for {alias}. Using heat profile for rose" + ) + idx = 10 + + dt = pendulum.datetime( + year=int(defaults[4][1]), + month=int(defaults[5][1]), + day=int(defaults[6][1]), + hour=int(defaults[7][1]), + minute=int(defaults[8][1]), + ) + file_start = dt.int_timestamp + if start_s < file_start: + raise Exception( + f"Tried to get flo params at {pendulum.from_timestamp(start_s)}. " + f"Only have data starting at {pendulum.from_timestamp(file_start)}" + ) + FIRST_HEAT_RATIO_IDX = 11 + start_idx = int((start_s - file_start) / 3600) + FIRST_HEAT_RATIO_IDX + + power_required_list: List[float] = [] + for i in range(slices): + try: + power_required = ( + float(defaults[start_idx + i][idx]) * atn_params.AnnualHvacKwhTh + ) + except: + fail_t = start_s + i * 3600 + raise Exception(f"No heat use data for {pendulum.from_timestamp(fail_t)}!") + power_required_list.append(power_required) + + ep = Csv_Eprt_Sync_1_0_0(ELEC_PRICE_FILE).payload + ep_start = pendulum.datetime( + year=ep.StartYearUtc, + month=ep.StartMonthUtc, + day=ep.StartDayUtc, + hour=ep.StartHourUtc, + minute=ep.StartMinuteUtc, + ).int_timestamp + start_idx = int((start_s - ep_start) / 3600) + elec_price_list: List[float] = ep.Prices[start_idx : start_idx + slices] + + dp = Csv_Distp_Sync_1_0_0(DIST_PRICE_FILE).payload + dp_start = pendulum.datetime( + year=dp.StartYearUtc, + month=dp.StartMonthUtc, + day=dp.StartDayUtc, + hour=dp.StartHourUtc, + minute=dp.StartMinuteUtc, + ).int_timestamp + start_idx = int((start_s - dp_start) / 3600) + dist_price_list: List[float] = dp.Prices[start_idx : start_idx + slices] + + tp = Csv_Weather_Forecast_Sync_1_0_0(WEATHER_PRICE_FILE).payload + tp_start = pendulum.datetime( + year=tp.StartYearUtc, + month=tp.StartMonthUtc, + day=tp.StartDayUtc, + hour=tp.StartHourUtc, + ).int_timestamp + start_idx = int((start_s - tp_start) / 3600) + temp_list: List[float] = tp.Temperatures[start_idx : start_idx + slices] + + start = pendulum.from_timestamp(start_s) + d = dict(atn_params.dict(), TypeName="flo.params.heatpumpwithbooststore") + d["SystemMaxHeatOutputKwAvg"] = strategy_utils.get_system_max_heat_output_kw_avg( + system_max_heat_output_gpm=atn_params.SystemMaxHeatOutputGpm, + system_max_heat_output_delta_temp_f=atn_params.SystemMaxHeatOutputDeltaTempF, + ) + d["K"] = strategy_utils.get_k( + system_max_heat_output_delta_temp_f=atn_params.SystemMaxHeatOutputDeltaTempF, + system_max_heat_output_gpm=atn_params.SystemMaxHeatOutputGpm, + system_max_heat_output_swt_f=atn_params.SystemMaxHeatOutputSwtF, + room_temp_f=atn_params.RoomTempF, + ) + d["IsRegulating"] = False + d["SliceDurationMinutes"] = [slice_duration_minutes] * slices + d["PowerRequiredByHouseFromSystemAvgKwList"] = power_required_list + d["OutsideTempF"] = temp_list + d["RealtimeElectricityPrice"] = elec_price_list + d["DistributionPrice"] = dist_price_list + d["RegulationPrice"] = [] + d["RtElecPriceUid"] = str(uuid.uuid4()) + d["WeatherUid"] = str(uuid.uuid4()) + d["DistPriceUid"] = str(uuid.uuid4()) + d["StartYearUtc"] = start.year + d["StartMonthUtc"] = start.month + d["StartDayUtc"] = start.day + d["StartHourUtc"] = start.hour + d["StartMinuteUtc"] = start.minute + d["StartingIdx"] = storage_idx + d["FloParamsUid"] = str(uuid.uuid4()) + return FloParams(**d) + + +def initialize_slot_stuff( + slot: MarketSlot, + atn_params: AtnParams, + atn_gni_id: str, +) -> SlotStuff: + flo_params = get_flo_params(atn_params=atn_params, slot=slot, storage_idx=0) + + bid = AtnBid( + BidderAlias=atn_params.GNodeAlias, + BidderGNodeInstanceId=atn_gni_id, + MarketSlotName=atn_utils.name_from_market_slot(slot), + PqPairs=[], + InjectionIsPositive=False, + PriceUnit=slot.Type.PriceUnit, + QuantityUnit=slot.Type.QuantityUnit, + SignedMarketFeeTxn=atn_utils.DUMMY_ALGO_TXN, + ) + return SlotStuff(Slot=slot, BidParams=flo_params, Bid=bid) diff --git a/src/gwatn/strategies/simple_resistive_hydronic/edge.py b/src/gwatn/strategies/simple_resistive_hydronic/edge.py new file mode 100644 index 0000000..1314e37 --- /dev/null +++ b/src/gwatn/strategies/simple_resistive_hydronic/edge.py @@ -0,0 +1,33 @@ +""" Heatpumpwithbooststore Flo Edge Class Definition """ +from typing import Optional +from typing import no_type_check + +from gwatn.data_classes.d_edge import DEdge + + +SIG_FIGS_FOR_OUTPUT = 6 + + +class Edge__BrickStorageHeater(DEdge): + def __init__( + self, + start_ts_idx: int, + start_idx: int, + end_idx: int, + avg_kw: Optional[float] = None, + cost: Optional[float] = None, + ): + DEdge.__init__( + self, + start_ts_idx=start_ts_idx, + start_idx=start_idx, + end_idx=end_idx, + cost=cost, + ) + self.avg_kw = avg_kw + + def __repr__(self) -> str: + rep = f"DEdge => Time Slice Idx: {self.start_ts_idx}, StartIdx: {self.start_idx}, EndIdx: {self.end_idx}" + if self.cost is not None: + rep += f" Cost => {self.cost}" + return rep diff --git a/src/gwatn/strategies/simple_resistive_hydronic/flo.py b/src/gwatn/strategies/simple_resistive_hydronic/flo.py new file mode 100644 index 0000000..2fc47c4 --- /dev/null +++ b/src/gwatn/strategies/simple_resistive_hydronic/flo.py @@ -0,0 +1,193 @@ +# type: ignore +import math +import random +import time +from typing import Dict +from typing import List +from typing import Optional +from typing import no_type_check + +import numpy as np +import pendulum + +import gwatn.brick_storage_heater.strategy_utils as strategy_utils +from gwatn.brick_storage_heater.edge import Edge__BrickStorageHeater as Edge +from gwatn.brick_storage_heater.node import Node_BrickStorageHeater as Node +from gwatn.data_classes.d_graph import DGraph +from gwatn.types import FloParamsBrickstorageheater as FloParams + + +class Flo__BrickStorageHeater(DGraph): + MAGIC_HEAT_PUMP_DELTA_F = 20 + FAILED_HEATING_PENALTY_DOLLARS = 10**6 + + def __init__( + self, + params: FloParams, + d_graph_id: str, + ): + self.params = params + + self.RealtimeElectricityPrice = np.array(self.params.RealtimeElectricityPrice) + self.DistributionPrice = np.array(self.params.DistributionPrice) + if self.params.IsRegulating: + self.reg_price_per_mwh = np.array(self.params.RegulationPrice) + else: + self.reg_price_per_mwh = np.array( + [0] * len(self.params.RealtimeElectricityPrice) + ) + + self.max_energy_kwh_th = strategy_utils.get_max_store_kwh_th(self.params) + + self.currency_unit = self.params.CurrencyUnit + self.temp_unit = self.params.TempUnit + DGraph.__init__( + self, + d_graph_id=d_graph_id, + graph_strategy_alias="SpaceHeat__HeatPumpWithBoostStore__Flo", + flo_start_unix_time_s=pendulum.datetime( + year=self.params.StartYearUtc, + month=self.params.StartMonthUtc, + day=self.params.StartDayUtc, + hour=self.params.StartHourUtc, + minute=self.params.StartMinuteUtc, + ).timestamp(), + slice_duration_hrs=list(np.array(self.params.SliceDurationMinutes) / 60), + default_storage_steps=self.params.StorageSteps, + starting_store_idx=self.params.StartingStoreIdx, + timezone_string=self.params.TimezoneString, + home_city=self.params.HomeCity, + currency_unit=self.params.CurrencyUnit, + max_storage=self.max_energy_kwh_th, + max_power_in=self.params.RatedMaxPowerKw, + wh_exponent=3, + ) + self.e_step: float = self.max_energy_kwh_th / self.params.StorageSteps + room_temp_c = (self.params.RoomTempF - 32) * 5 / 9 + temp_range_c = self.params.MaxBrickTempC - room_temp_c + self.temp_step_c: float = temp_range_c / self.params.StorageSteps + self.store_kwh_per_deg_c = self.max_energy_kwh_th / temp_range_c + self.energy_cost_per_kwh: Dict[int, float] = {} + self.set_energy_cost_per_kwh() + self.uncosted_edges: Dict[Edge, int] = {} + self.failed_in_cost_boost_preferred: int = 0 + self.failed_in_cost_hp_preferred: int = 0 + + self.create_graph() + self.solve_dijkstra() + + def create_slice_nodes(self, ts_idx: int) -> None: + """Creates nodes for time slice ts_idx, equally spaced by self.e_step, + running from params.ZeroPotentialEnergyWaterTempF up to params.MaxStoreTempF. + + Sets the energy store in kwh of enthalpy (latent heat in this case) as well + as the average boost water temp. + + """ + ... + + def create_nodes(self) -> None: + for ts_idx in range(self.time_slices + 1): + self.create_slice_nodes(ts_idx) + + def set_energy_cost_per_kwh(self): + for ts_idx in range(self.time_slices): + self.energy_cost_per_kwh[ts_idx] = self.get_energy_cost_per_kwh(ts_idx) + + ###################################################### + # Functions of edges + ###################################################### + + # Cost related + + def get_energy_cost_per_kwh(self, ts_idx: int) -> float: + energy_cost_per_mwh = self.params.RealtimeElectricityPrice[ts_idx] + dist_cost_per_mwh = self.params.DistributionPrice[ts_idx] + + cost_per_mwh = energy_cost_per_mwh + dist_cost_per_mwh + return cost_per_mwh / 1000 + + ###################################################### + # Functions of edges - Water temp related + ##################################################### + + def set_failed_edge_properties(self, edge: Edge): + """Set costs""" + edge.cost = self.FAILED_HEATING_PENALTY_DOLLARS + edge.hp_electricity_avg_kw = 10000 + edge.avg_kw = 0 + + def get_failed_edge(self, node: Node) -> Edge: + edge = Edge( + start_ts_idx=node.ts_idx, start_idx=node.store_idx, end_idx=node.store_idx + ) + self.set_failed_edge_properties(edge) + return edge + + def get_uncosted_edge( + self, node: Node, delta_energy_kwh: float, existing_edges: List[Edge] = [] + ) -> Optional[Edge]: + """Finds a randomized passing edge consistent with delta_energy_kwh. + For example, if delta_energy_kwh falls exactly half way between + two energy steps it will return each step with probability 50% + """ + if node.store_enthalpy_kwh + delta_energy_kwh < 0: + return None + delta_idx_as_float = delta_energy_kwh / self.e_step + lower_delta_idx = math.floor(delta_idx_as_float) + frac = delta_idx_as_float - lower_delta_idx + random_adder = random.choices(population=[0, 1], weights=[1 - frac, frac], k=1)[ + 0 + ] + if node.store_idx + lower_delta_idx == self.params.StorageSteps: + delta_idx = lower_delta_idx + else: + delta_idx = lower_delta_idx + random_adder + edge = Edge( + start_ts_idx=node.ts_idx, + start_idx=node.store_idx, + end_idx=node.store_idx + delta_idx, + ) + if edge not in existing_edges: + self.uncosted_edges[edge] = 1 + return edge + else: + # in this case the edge is already in the existing_edges, so return None + return None + + def get_edges_from_node(self, node: Node) -> List[Edge]: + """Creates edges starting at a node.""" + ts_idx = node.ts_idx + return [] + + def set_edge_cost(self, edge: Edge) -> None: + """Given an edge: + - set cost to penalty if there is no way to meet the energy requirements + - otherwise set the cost for the optimal choice of using the boost and + the heat pump + """ + ... + + def create_graph(self) -> None: + # print(f"Creating graph nodes, edges and edge weights") + st = time.time() + self.create_nodes() + nt = time.time() + # print(f"{time.time() - st:1.0f} seconds to make nodes") + for jj in range(self.time_slices): + # if jj % 500 == 0: + # print(f"{time.time() - st:1.0f} seconds for {jj} slices") + for node in self.node[jj].values(): + edges = self.get_edges_from_node(node=node) + self.edges[node] = edges + et = time.time() + # print(f"{et - nt:1.0f} seconds for building edges") + for edge in self.uncosted_edges.keys(): + self.set_edge_cost(edge) + self.uncosted_edges = {} + ct = time.time() + # print(f"{ct - et:1.0f} seconds for costing edges") + tt = time.time() - st + # print(f"time per 100 slices:{tt*100000/self.time_slices:2.0f} ms ") + # print(f"total time to build graph: {tt:2.0f} s") + # print(f"Total number of slices: {self.time_slices}") diff --git a/src/gwatn/strategies/simple_resistive_hydronic/flo_output.py b/src/gwatn/strategies/simple_resistive_hydronic/flo_output.py new file mode 100644 index 0000000..ff2a5f9 --- /dev/null +++ b/src/gwatn/strategies/simple_resistive_hydronic/flo_output.py @@ -0,0 +1,491 @@ +import time + +import gridworks.conversion_factors as cf +import pendulum +import xlsxwriter + +import gwatn.brick_storage_heater.strategy_utils as strategy_utils +from gwatn.brick_storage_heater.edge import Edge__BrickStorageHeater as Edge +from gwatn.brick_storage_heater.flo import Flo__BrickStorageHeater as Flo +from gwatn.brick_storage_heater.node import Node_BrickStorageHeater as Node +from gwatn.enums import RecognizedCurrencyUnit +from gwatn.types import AtnParamsBrickstorageheater as AtnParams + + +OUTPUT_FOLDER = "output_data" + +ON_PEAK_DIST_PRICE_PER_MWH_CUTOFF = 100 +SHOULDER_PEAK_DIST_PRICE_PER_MWH_CUTOFF = 50 + + +def export_xlsx(alias: str, flo: Flo, atn_params: AtnParams): + local_start = pendulum.timezone(flo.timezone_string).convert(flo.flo_start_utc) + date_str = local_start.strftime("%Y%m%d") + hour_str = local_start.strftime("%H") + file = ( + OUTPUT_FOLDER + f"/result_{alias}_{date_str}_{hour_str}_{int(time.time())}.xlsx" + ) + # flo.graph_strategy_alias, + + file = file.lower() + print(file) + + workbook = xlsxwriter.Workbook(file) + starting_store_idx = flo.starting_store_idx + + # Add to gsr for more blank rows + gsr = 30 + w = export_best_path_info( + alias=alias, flo=flo, workbook=workbook, starting_store_idx=starting_store_idx + ) + + export_flo_graph( + flo=flo, + workbook=workbook, + worksheet=w, + starting_store_idx=starting_store_idx, + graph_start_row=gsr, + ) + + export_params_xlsx(flo=flo, atn_params=atn_params, workbook=workbook) + workbook.close() + + +def export_best_path_info( + alias: str, + flo: Flo, + workbook: xlsxwriter.workbook.Workbook, + starting_store_idx: int, +): + w = workbook.add_worksheet( + f"start {100 * starting_store_idx / flo.params.StorageSteps}%" + ) + w.freeze_panes(0, 2) + title_format = workbook.add_format({"bold": True}) + title_format.set_font_size(14) + bold_format = workbook.add_format({"bold": True}) + gray_filler_format = workbook.add_format({"bg_color": "#edf0f2"}) + header_format = workbook.add_format({"bold": True, "text_wrap": True}) + mwh_format = workbook.add_format({"bold": True, "num_format": '0.00" MWh"'}) + if flo.params.CurrencyUnit == RecognizedCurrencyUnit.USD: + currency_format = workbook.add_format({"num_format": "$#,##0.00"}) + currency_bold_format = workbook.add_format( + {"bold": True, "num_format": "$#,##0.00"} + ) + elif flo.params.CurrencyUnit == RecognizedCurrencyUnit.GBP: + currency_format = workbook.add_format( + {"num_format": "_-[$£-en-GB]* #,##0.00_-"} + ) + currency_bold_format = workbook.add_format( + {"bold": True, "num_format": "_-[$£-en-GB]* #,##0.00_-"} + ) + + w.set_column("A:A", 26) + w.set_column("B:B", 15) + OPT_PATH_STATE_VAR_ROW = 14 + + swt_list = flo_utils.get_source_water_temp_f_list(flo.params) + w.write(0, 0, f"GNode Alias: {alias}", title_format) + w.write(1, 0, flo.graph_strategy_alias) + w.write(2, 0, f"FLO start {flo.params.TimezoneString} ", header_format) + local_start = pendulum.timezone(flo.timezone_string).convert(flo.flo_start_utc) + w.write(2, 1, local_start.strftime("%Y/%m/%d %H:%M")) + + w.write(3, 0, "Total hours", header_format) + w.write(3, 1, sum(flo.slice_duration_hrs), bold_format) + + w.write(4, 0, "Rt Energy Price ($/MWh)", header_format) + w.write( + 4, + 1, + sum(flo.RealtimeElectricityPrice) / len(flo.RealtimeElectricityPrice), + currency_bold_format, + ) + # w.write(5, 0, "Flat rate for hp ($/MWh)", header_format) + # if flo.params.IsRegulating: + # w.write(5, 0, "Regulation Price ($/MWh)", header_format) + # w.write( + # 5, 1, sum(flo.reg_price_per_mwh) / len(flo.reg_price_per_mwh), currency_bold_format + # ) + + w.write(6, 0, "Dist Price ($/MWh)", header_format) + w.write( + 6, + 1, + sum(flo.DistributionPrice) / len(flo.DistributionPrice), + currency_bold_format, + ) + w.write(7, 0, "Outside Temp F", header_format) + w.write( + 7, + 1, + round(sum(flo.params.OutsideTempF) / len(flo.params.OutsideTempF), 2), + bold_format, + ) + w.write(8, 0, "COP", header_format) + avg_cop = sum(flo.cop.values()) / len(flo.cop) + w.write(8, 1, round(avg_cop, 2), bold_format) + w.write(9, 0, "House Power Required AvgKw", header_format) + w.write( + 9, + 1, + round(sum(flo.params.PowerRequiredByHouseFromSystemAvgKwList), 2), + bold_format, + ) + w.write(10, 0, "Required Source Water Temp F", header_format) + avg_swt = round((sum(swt_list) / len(swt_list)), 0) + w.write(10, 1, avg_swt, bold_format) + w.write(11, 0, "Max HeatPump kWh thermal", header_format) + avg_max_thermal_hp_kwh = round( + (sum(flo.max_thermal_hp_kwh.values()) / len(flo.max_thermal_hp_kwh)), 2 + ) + w.write(11, 1, avg_max_thermal_hp_kwh, bold_format) + + w.write(12, 0, "Outputs", header_format) + + for jj in range(flo.time_slices): + hours_since_start = sum(flo.slice_duration_hrs[0:jj]) + local_time = local_start.add(hours=hours_since_start) + w.write(2, jj + 2, local_time.strftime("%m/%d")) + w.write(3, jj + 2, local_time.strftime("%H:%M")) + w.write(4, jj + 2, flo.RealtimeElectricityPrice[jj], currency_format) + if flo.params.IsRegulating: + w.write(5, jj + 2, flo.reg_price_per_mwh[jj], currency_format) + else: + w.write(5, jj + 2, "", gray_filler_format) + dp = flo.DistributionPrice[jj] + LIGHT_GREEN_HEX = "#bbe3a6" + LIGHT_RED_HEX = "#ff6363" + if dp > ON_PEAK_DIST_PRICE_PER_MWH_CUTOFF: + if flo.params.CurrencyUnit == RecognizedCurrencyUnit.USD: + dist_format = workbook.add_format( + {"bg_color": LIGHT_RED_HEX, "num_format": "$#,##0.00"} + ) + elif flo.params.CurrencyUnit == RecognizedCurrencyUnit.GBP: + dist_format = workbook.add_format( + { + "bg_color": LIGHT_RED_HEX, + "num_format": "_-[$£-en-GB]* #,##0.00_-", + } + ) + elif dp > SHOULDER_PEAK_DIST_PRICE_PER_MWH_CUTOFF: + if flo.params.CurrencyUnit == RecognizedCurrencyUnit.USD: + dist_format = workbook.add_format( + {"bg_color": "yellow", "num_format": "$#,##0.00"} + ) + elif flo.params.CurrencyUnit == RecognizedCurrencyUnit.GBP: + dist_format = workbook.add_format( + {"bg_color": "yellow", "num_format": "_-[$£-en-GB]* #,##0.00_-"} + ) + else: + if flo.params.CurrencyUnit == RecognizedCurrencyUnit.USD: + dist_format = workbook.add_format( + {"bg_color": LIGHT_GREEN_HEX, "num_format": "$#,##0.00"} + ) + elif flo.params.CurrencyUnit == RecognizedCurrencyUnit.GBP: + dist_format = workbook.add_format( + { + "bg_color": LIGHT_GREEN_HEX, + "num_format": "_-[$£-en-GB]* #,##0.00_-", + } + ) + w.write(6, jj + 2, flo.DistributionPrice[jj], dist_format) + w.write(7, jj + 2, flo.params.OutsideTempF[jj]) + w.write(8, jj + 2, flo.cop[jj]) + w.write( + 9, jj + 2, round(flo.params.PowerRequiredByHouseFromSystemAvgKwList[jj], 2) + ) + w.write(10, jj + 2, round(swt_list[jj], 0)) + w.write(11, jj + 2, round(flo.max_thermal_hp_kwh[jj], 2)) + + w.write(12, jj + 2, "", gray_filler_format) + + node: Node = flo.node[0][starting_store_idx] + w.write(OPT_PATH_STATE_VAR_ROW, 0, "Store Temp (F)", header_format) + w.write(OPT_PATH_STATE_VAR_ROW + 1, 0, "HeatPump kWh thermal", header_format) + w.write(OPT_PATH_STATE_VAR_ROW + 2, 0, "HeatPump kWh electric", header_format) + w.write(OPT_PATH_STATE_VAR_ROW + 3, 0, "Boost kWh electric", header_format) + w.write(OPT_PATH_STATE_VAR_ROW + 4, 0, "Energy cost (¢)", header_format) + w.write(OPT_PATH_STATE_VAR_ROW + 5, 0, "Hours Since Start", header_format) + + store_temp_f = [] + opt_heatpump_electricity_used_kwh = [] + opt_boost_electricity_used_kwh = [] + opt_energy_cost_dollars = [] + + best_idx = starting_store_idx + dist_cost = [] + min_dist_price_per_mwh = min(flo.DistributionPrice) + + for jj in range(flo.time_slices): + edge: Edge = flo.best_edge[node] + store_temp_f.append(node.store_avg_water_temp_f) + hp_kwh = edge.hp_electricity_avg_kw + boost_kwh = edge.boost_electricity_used_avg_kw + opt_heatpump_electricity_used_kwh.append(hp_kwh) + opt_boost_electricity_used_kwh.append(boost_kwh) + opt_energy_cost_dollars.append(edge.cost) + hours_since_start = sum(flo.slice_duration_hrs[0:jj]) + + w.write(OPT_PATH_STATE_VAR_ROW, jj + 2, round(node.store_avg_water_temp_f, 2)) + w.write( + OPT_PATH_STATE_VAR_ROW + 1, + jj + 2, + round(edge.hp_thermal_energy_generated_avg_kw, 3), + ) + w.write( + OPT_PATH_STATE_VAR_ROW + 2, jj + 2, round(edge.hp_electricity_avg_kw, 3) + ) + w.write( + OPT_PATH_STATE_VAR_ROW + 3, + jj + 2, + round(edge.boost_electricity_used_avg_kw, 3), + ) + w.write(OPT_PATH_STATE_VAR_ROW + 4, jj + 2, round(edge.cost * 100, 2)) + w.write(OPT_PATH_STATE_VAR_ROW + 5, jj + 2, round(hours_since_start, 1)) + node = flo.node[jj + 1][edge.end_idx] + + w.write( + OPT_PATH_STATE_VAR_ROW, + 1, + round(sum(store_temp_f) / len(store_temp_f), 0), + bold_format, + ) + w.write( + OPT_PATH_STATE_VAR_ROW + 2, + 1, + sum(opt_heatpump_electricity_used_kwh) / 1000, + mwh_format, + ) + w.write( + OPT_PATH_STATE_VAR_ROW + 3, + 1, + sum(opt_boost_electricity_used_kwh) / 1000, + mwh_format, + ) + w.write( + OPT_PATH_STATE_VAR_ROW + 4, + 1, + sum(opt_energy_cost_dollars), + currency_bold_format, + ) + + w.write("H1", "Electricity cost of this path") + total_cost = sum(opt_energy_cost_dollars) + w.write("G1", total_cost, currency_bold_format) + w.write("H2", "Total electricity MWh") + total_electricity_mwh = ( + sum(opt_boost_electricity_used_kwh) + sum(opt_heatpump_electricity_used_kwh) + ) / 1000 + w.write("G2", total_electricity_mwh, mwh_format) + + total_btu = sum(flo.params.PowerRequiredByHouseFromSystemAvgKwList) * cf.BTU_PER_KWH + gallons_oil = total_btu / cf.BTU_PER_GALLON_OF_OIL / 0.85 + # assumes 85% efficient oil boiler + + w.write("L1", "Gallons of Oil") + w.write("K1", round(gallons_oil), bold_format) + + w.write("L2", "Equivalent Price of Oil") + w.write("K2", total_cost / gallons_oil, currency_bold_format) + # w.write("H3", "Flat rate comparison") + + return w + + +def export_flo_graph( + flo: Flo, + workbook: xlsxwriter.workbook.Workbook, + worksheet: xlsxwriter.workbook.Worksheet, + starting_store_idx: int, + graph_start_row: int, +): + # worksheet.freeze_panes(graph_start_row - 1, 2) + header_format = workbook.add_format({"bold": True, "text_wrap": True}) + percent_format = workbook.add_format({"num_format": '00.0"%"'}) + best_path_format = workbook.add_format({"bold": True, "bg_color": "#CDEBA6"}) + best_idx = starting_store_idx + + worksheet.write(graph_start_row - 1, 0, "Percent full", header_format) + + for mm in range(flo.params.StorageSteps + 1): + kk = flo.params.StorageSteps - mm + percent = round(100 * kk / flo.params.StorageSteps, 1) + worksheet.write(graph_start_row + mm, 0, percent, percent_format) + for jj in range(flo.time_slices): + best_node = flo.node[jj][best_idx] + for mm in range(flo.params.StorageSteps + 1): + kk = flo.params.StorageSteps - mm + node = flo.node[jj][kk] + if kk == best_idx: + try: + worksheet.write( + graph_start_row + mm, + jj + 2, + -round(node.path_benefit, 4), + best_path_format, + ) + except: + print(f"failed for best {jj},{kk}") + else: + try: + worksheet.write( + graph_start_row + mm, jj + 2, round(node.path_cost, 4) + ) + except: + print(f"failed for {jj},{kk}") + best_idx = flo.best_edge[best_node].end_idx + + +def export_params_xlsx( + flo: Flo, + atn_params: AtnParams, + workbook: xlsxwriter.workbook.Workbook, +): + bold = workbook.add_format({"bold": True}) + w = workbook.add_worksheet("Params") + derived_format_bold = workbook.add_format({"bold": True, "font_color": "green"}) + derived_format = workbook.add_format({"font_color": "green"}) + swt_list = flo_utils.get_source_water_temp_f_list(flo.params) + w.set_column("A:A", 31) + w.set_column("D:D", 31) + w.set_column("G:G", 31) + w.write("A1", "Key Parameters", bold) + + t = flo.params.OutsideTempF + w.write("A4", "This Run ColdestTempF ", derived_format_bold) + w.write("B4", min(t), derived_format) + + w.write("A5", "HouseWorstCaseTempF ", bold) + w.write("B5", atn_params.HouseWorstCaseTempF) + + w.write("A7", "SystemMaxHeatOutputKwAvg", bold) + w.write("B7", round(flo.params.SystemMaxHeatOutputKwAvg, 2)) + + p = flo.params.PowerRequiredByHouseFromSystemAvgKwList + w.write("A8", "This Run MaxHeatOutputKwAvg", derived_format_bold) + w.write("B8", round(max(p), 2), derived_format) + + house_wc_kw = flo_utils.get_house_worst_case_heat_output_avg_kw(flo.params) + w.write("A9", "HouseWorstCaseHeatOuputAvgKw", derived_format_bold) + w.write("B9", round(house_wc_kw, 1), derived_format) + + w.write("A10", "HouseWorstCaseHeatOuput BTU/hr", derived_format_bold) + w.write("B10", round(house_wc_kw * cf.BTU_PER_KWH), derived_format) + + w.write("A12", "EmitterMaxSafeSwtF", bold) + w.write("B12", flo.params.EmitterMaxSafeSwtF) + + w.write("A13", "This Run SystemMaxHeatOutputSwtF", bold) + w.write("B13", round(max(swt_list))) + + w.write("A14", "SystemMaxHeatOutputSWTF ", bold) + w.write("B14", flo.params.SystemMaxHeatOutputSwtF) + + w.write("A15", "HeatPumpMaxWaterTempF ", bold) + w.write("B15", flo.params.MaxHeatpumpSourceWaterTempF) + + w.write("A16", "RatedHeatpumpElectricityKw", bold) + w.write("B16", flo.params.RatedHeatpumpElectricityKw) + + w.write("A17", "StoreMaxPowerKw", bold) + w.write("B17", flo.params.StoreMaxPowerKw) + + if flo.params.EmitterPumpFeedbackModel == EmitterPumpFeedbackModel.ConstantDeltaT: + w.write("A19", "SystemMaxHeatOutputDeltaTempF", bold) + w.write("B19", flo.params.SystemMaxHeatOutputDeltaTempF) + + w.write("A20", "SystemMaxHeatOutputGpm", derived_format_bold) + w.write("B20", round(flo.params.SystemMaxHeatOutputGpm, 2), derived_format) + else: + w.write("A19", "SystemMaxHeatOutputDeltaTempF", bold) + w.write("B19", flo.params.SystemMaxHeatOutputDeltaTempF) + + w.write("A20", "SystemMaxHeatOutputGpm", derived_format_bold) + w.write("B20", round(flo.params.SystemMaxHeatOutputGpm, 2), derived_format) + + w.write("A21", "Cop1TempF", bold) + w.write("B21", flo.params.Cop1TempF) + + w.write("A22", "Cop4TempF", bold) + w.write("B22", flo.params.Cop4TempF) + + w.write("A23", "StorePassiveLossRatio", bold) + w.write("B23", flo.params.StorePassiveLossRatio) + + w.write("A25", "StorageSteps", bold) + w.write("B25", flo.params.StorageSteps) + + ############# + + annual_kwh = atn_params.AnnualHvacKwhTh + annual_btu = round(cf.BTU_PER_KWH * annual_kwh) + w.write("D3", "Annual HVAC kWhTh", bold) + w.write("E3", annual_kwh) + + w.write("D4", "Annual HVAC MBTU", derived_format_bold) + w.write("E4", round(annual_btu / 10**6), derived_format) + + w.write("D6", "StoreSizeGallons", bold) + w.write("E6", flo.params.StoreSizeGallons) + + w.write("D7", "MaxStoreTempF", bold) + w.write("E7", flo.params.MaxStoreTempF) + + w.write("D8", "ZeroPotentialEnergyWaterTempF", bold) + w.write("E8", flo.params.ZeroPotentialEnergyWaterTempF) + + w.write("D9", "TotalStorageKwh", derived_format_bold) + w.write("E9", round(flo.max_energy_kwh_th, 1), derived_format) + + w.write("D10", "TotalStorage BTU", derived_format_bold) + w.write("E10", round(cf.BTU_PER_KWH * flo.max_energy_kwh_th), derived_format) + + w.write("D12", "EmitterPumpFeedbackModel", bold) + w.write("E12", flo.params.EmitterPumpFeedbackModel.value) + + w.write("D13", "MixingValveFeedbackModel", bold) + w.write("E13", flo.params.MixingValveFeedbackModel.value) + + w.write("D14", "IsRegulating", bold) + w.write("E14", flo.params.IsRegulating) + + w.write("D19", "RoomTempF", bold) + w.write("E19", flo.params.RoomTempF) + + w.write("D20", "AmbientPowerInKw", bold) + w.write("E20", flo.params.AmbientPowerInKw) + + ############### + + w.write("G3", "HeatpumpTariff", bold) + w.write("H3", flo.params.HeatpumpTariff.value) + + w.write("G4", "HeatpumpEnergySupplyType", bold) + w.write("H4", flo.params.HeatpumpEnergySupplyType.value) + + w.write("G5", "BoostTariff", bold) + w.write("H5", flo.params.BoostTariff.value) + + w.write("G6", "BoostEnergySupplyType", bold) + w.write("H6", flo.params.BoostEnergySupplyType.value) + + w.write("G7", "StandardOfferPriceDollarsPerMwh", bold) + w.write("H7", flo.params.StandardOfferPriceDollarsPerMwh) + + w.write("G8", "DistributionTariffDollarsPerMwh", bold) + w.write("H8", flo.params.DistributionTariffDollarsPerMwh) + + w.write("A29", "WeatherUid", bold) + w.write("B29", flo.params.WeatherUid) + w.write("A31", "RtElecPriceUid", bold) + w.write("B31", flo.params.RtElecPriceUid) + + w.write("A33", "DistPriceUid", bold) + w.write("B33", flo.params.DistPriceUid) + + if flo.params.IsRegulating: + w.write("A34", "LocalRegulationFile", bold) + w.write("B34", regp_sync_100_handler.csv_file_by_uid(flo.params.RegPriceUid)) + w.write("A35", "RegPriceUid", bold) + w.write("B35", flo.params.RegPriceUid) diff --git a/src/gwatn/strategies/simple_resistive_hydronic/flo_utils.py b/src/gwatn/strategies/simple_resistive_hydronic/flo_utils.py new file mode 100644 index 0000000..c8d2c37 --- /dev/null +++ b/src/gwatn/strategies/simple_resistive_hydronic/flo_utils.py @@ -0,0 +1,230 @@ +from typing import List + +import gridworks.conversion_factors as cf # TODO change to from gwatn import conversion_factors as cf +from satn.enums import ShDistPumpFeedbackModel +from satn.enums import ShMixingValveFeedbackModel +from satn.types import FloParamsHeatpumpwithbooststore as FloParams + +import gwatn.errors as errors + + +def get_max_store_kwh_th(params: FloParams) -> float: + """Can be duck-typed with AtnParams as well""" + return ( + cf.KWH_TH_PER_GALLON_PER_DEG_F + * params.StoreSizeGallons + * (params.MaxStoreTempF - params.ZeroPotentialEnergyWaterTempF) + ) + + +def get_house_worst_case_heat_output_avg_kw(params: FloParams) -> float: + design_t = params.HouseWorstCaseTempF + this_run_t = min(params.OutsideTempF) + room_t = params.RoomTempF + p = params.PowerRequiredByHouseFromSystemAvgKwList + this_run_max_system_kw = max(p) + this_run_max_kw_in = this_run_max_system_kw + params.AmbientPowerInKw + dd_max_kw_in = this_run_max_kw_in * (room_t - design_t) / (room_t - this_run_t) + return dd_max_kw_in - params.AmbientPowerInKw + + +def get_source_water_temp_f_list(params: FloParams) -> List[float]: + """The SourceWaterTemp or SWT is the water in the hydronic pipes + going into the emitters, after the mixing valve. This + temperature is determined by the outside temperature and + is regulated by the emitter circulator pump feedback mechanism and, + when the SWT is above the MaxHeatPumpSourceWaterTempF, the mixing valve + that mixes the water coming from the boost and the + IntermediateWaterTemp (see graphic in explanatory artifact) + + Explanatory artifact: LINK + + Args: + params: TeaParams. + power_required_by_house_from_system_avg_kw: A list (by time slice) of the + power required by the house from the system. This will be less than the + the actual power required by the house by the ambient power supplied from + other sources (other electrical appliances, ambient solar, animals) + """ + + swt: List[float] = [] + system_heat_list = params.PowerRequiredByHouseFromSystemAvgKwList + for i in range(len(system_heat_list)): + system_heat_avg_kw = system_heat_list[i] + try: + this_slice_swt = get_source_water_temp_f( + params=params, system_heat_kw=system_heat_avg_kw + ) + except errors.PhysicalSystemFailure: + raise Exception( + f"Trouble for slice {i} and system_heat_avg_kw {system_heat_avg_kw}" + ) + swt.append(this_slice_swt) + swt.append(params.ZeroPotentialEnergyWaterTempF) + return swt + + +def get_source_water_temp_f(params: FloParams, system_heat_kw: float) -> float: + """Returns SourceWaterTempF for given this system_heat_avg_kw, + and ShDistPumpFeedbackModel of ConstantGpm. Does not + let SourceWaterTempF go below params.ZeroPotentialEnergyWaterTempF + + Args: + params (FloParams): Params for the Flo. + system_heat_avg_kw (float): the heat the system is putting + into the house + + Raises: + errors.PhysicalSystemFailure: raised if derived + SourceWaterTempF exceeds EmitterMaxSafeSwtF + + Returns: + float: SourceWaterTempF + """ + if system_heat_kw < 0: + raise Exception(f"System does not TAKE heat from house") + if system_heat_kw == 0: + return params.ZeroPotentialEnergyWaterTempF + if params.DistPumpFeedbackModel == ShDistPumpFeedbackModel.ConstantDeltaT: + return get_constant_delta_t_swt(params=params, system_heat_kw=system_heat_kw) + else: + return get_constant_gpm_swt(params=params, system_heat_kw=system_heat_kw) + + +def get_constant_gpm_swt(params: FloParams, system_heat_kw: float) -> float: + """Returns SourceWaterTempF for given this system_heat_avg_kw, + and ShDistPumpFeedbackModel of ConstantGpm. Does not + let SourceWaterTempF go below params.ZeroPotentialEnergyWaterTempF. + + Args: + params (FloParams): Params for the Flo. Uses + - RoomTempF + - SystemMaxHeatOutputDeltaTempF + - EmitterMaxSafeSwtF + - SystemMaxHeatOutputSwtF + - SystemMaxHeatOutputGpm + system_heat_avg_kw (float): The heat provided by the system + into the house + system_heat_avg_kw (float): the heat the system ixs putting + into the house + + Raises: + errors.PhysicalSystemFailure: raised if derived + SourceWaterTempF exceeds EmitterMaxSafeSwtF + + Returns: + float: SourceWaterTempF + """ + rt = params.RoomTempF + ddd = params.SystemMaxHeatOutputDeltaTempF + dd_gpm = params.SystemMaxHeatOutputGpm + c = cf.POUNDS_OF_WATER_PER_GALLON * cf.MINUTES_PER_HOUR / cf.BTU_PER_KWH + dd_swt = params.SystemMaxHeatOutputSwtF + denominator = c * dd_gpm * (1 - (dd_swt - rt - ddd) / (dd_swt - rt)) + constant_running_swt = rt + (system_heat_kw / denominator) + constant_running_swt = max( + constant_running_swt, params.ZeroPotentialEnergyWaterTempF + ) + + if params.MixingValveFeedbackModel == ShMixingValveFeedbackModel.ConstantSwt: + return params.SystemMaxHeatOutputSwtF + elif params.MixingValveFeedbackModel == ShMixingValveFeedbackModel.NaiveVariableSwt: + if constant_running_swt > params.EmitterMaxSafeSwtF: + raise errors.PhysicalSystemFailure( + "Pump strategy: ConstantGpm. MixingValve: " + f"NaiveVariable. Constant running swt {constant_running_swt} F exceeds" + f" EmitterMaxSafeSwtF {params.EmitterMaxSafeSwtF}!" + ) + return constant_running_swt + elif ( + params.MixingValveFeedbackModel + == ShMixingValveFeedbackModel.CautiousVariableSwt + ): + cautious_swt = constant_running_swt + params.CautiousMixingValveTempDeltaF + if cautious_swt > params.EmitterMaxSafeSwtF: + raise errors.PhysicalSystemFailure( + "Pump strategy: ConstantGpm. MixingValve: " + f"CautiousVariable. Cautious {cautious_swt} F (hotter by {params.CautiousMixingValveTempDeltaF}" + f" than constant running temp) exceeds" + f" EmitterMaxSafeSwtF {params.EmitterMaxSafeSwtF}!" + ) + return cautious_swt + else: + raise Exception( + f"Unknown ShMixingValveFeedbackModel {params.MixingValveFeedbackModel}" + ) + + +def get_constant_delta_t_swt(params: FloParams, system_heat_kw: float) -> float: + """Calculates Source Water Temp (SWT) for a system with + a constant delta T feedback control mechanism for its circulator + pump/thermostat. Does not + let SourceWaterTempF go below params.ZeroPotentialEnergyWaterTempF. + + params (FloParams): Params for the Flo. Uses + - RoomTempF + - SystemMaxHeatOutputDeltaTempF + - EmitterMaxSafeSwtF + - SystemMaxHeatOutputSwtF + - SystemMaxHeatOutputGpm + system_heat_avg_kw (float): The heat provided by the system + into the house + system_heat_avg_kw (float): the heat the system is putting + into the house + + Raises: + errors.PhysicalSystemFailure: raised if derived + SourceWaterTempF exceeds EmitterMaxSafeSwtF + + Returns: + float: SourceWaterTempF + """ + p_req = system_heat_kw + if p_req <= 0: + return params.ZeroPotentialEnergyWaterTempF + rt = params.RoomTempF + ddd = params.SystemMaxHeatOutputDeltaTempF + + max_e_out = params.SystemMaxHeatOutputKwAvg + dd_swt = params.SystemMaxHeatOutputSwtF + base = (dd_swt - rt - ddd) / (dd_swt - rt) + exp = max_e_out / p_req + + numerator = rt + ddd - rt * (base**exp) + denominator = 1 - (base**exp) + if denominator == 0: + raise Exception(f"About to divide by zero. base = {base} exp = {exp}") + constant_running_swt: float = numerator / denominator + constant_running_swt = max( + constant_running_swt, params.ZeroPotentialEnergyWaterTempF + ) + if params.MixingValveFeedbackModel == ShMixingValveFeedbackModel.ConstantSwt: + return params.SystemMaxHeatOutputSwtF + elif params.MixingValveFeedbackModel == ShMixingValveFeedbackModel.NaiveVariableSwt: + if constant_running_swt > params.EmitterMaxSafeSwtF: + raise errors.PhysicalSystemFailure( + "Pump: ConstantDeltaT, MixingValve: " + f"NaiveVariable. {constant_running_swt} F exceeds" + f" EmitterMaxSafeSwtF {params.EmitterMaxSafeSwtF}!" + ) + + return constant_running_swt + elif ( + params.MixingValveFeedbackModel + == ShMixingValveFeedbackModel.CautiousVariableSwt + ): + cautious_swt: float = ( + constant_running_swt + params.CautiousMixingValveTempDeltaF + ) + if cautious_swt > params.EmitterMaxSafeSwtF: + raise errors.PhysicalSystemFailure( + "Pump strategy: ConstantDeltaT. MixingValve: " + f"CautiousVariable. Cautious {cautious_swt} F (hotter by {params.CautiousMixingValveTempDeltaF}" + f" than constant running temp) exceeds" + f" EmitterMaxSafeSwtF {params.EmitterMaxSafeSwtF}!" + ) + return cautious_swt + else: + raise Exception( + f"Unknown ShMixingValveFeedbackModel {params.MixingValveFeedbackModel}" + ) diff --git a/src/gwatn/strategies/simple_resistive_hydronic/make_dev_input_data.py b/src/gwatn/strategies/simple_resistive_hydronic/make_dev_input_data.py new file mode 100644 index 0000000..6785a31 --- /dev/null +++ b/src/gwatn/strategies/simple_resistive_hydronic/make_dev_input_data.py @@ -0,0 +1,58 @@ +import csv +import json + +import pendulum +from satn.types import AtnParamsHeatpumpwithbooststore as AtnParams +from satn.types import AtnParamsHeatpumpwithbooststore_Maker as AtnParams_Maker + +from gwatn.types import AtnParamsReport_Maker + + +params_file = "input_data/atn_params_data.csv" +params = [] +with open(params_file) as csv_file: + reader = csv.reader(csv_file) + for row in reader: + params.append(row) +start = pendulum.datetime(year=2020, month=1, day=1, hour=4) + +for j in range(1, 19): + alias = params[1][j] + atn_params = AtnParams( + GNodeAlias=alias, + BetaOt=params[2][j], + HouseHeatingCapacity=params[3][j], + AmbientPowerInKw=params[4][j], + AnnualHvacKwhTh=params[5][j], + StoreSizeGallons=params[8][j], + RatedHeatpumpElectricityKw=params[9][j], + StoreMaxPowerKw=params[10][j], + SystemMaxHeatOutputDeltaTempF=params[11][j], + SystemMaxHeatOutputGpm=params[12][j], + SystemMaxHeatOutputSwtF=params[13][j], + FloSlices=48, + SliceDurationMinutes=60, + ) + + params_dict = atn_params.as_dict() + try: + report = AtnParamsReport_Maker( + g_node_alias=alias, + g_node_instance_id="00000000-0000-0000-0000-000000000000", + atn_params_type_name=AtnParams_Maker.type_name, + time_unix_s=start.int_timestamp, + params=atn_params, + irl_time_unix_s=None, + ).tuple + except: + print(f"Problem with {alias}") + + file_name = f"input_data/eventstore/{alias}-{AtnParamsReport_Maker.type_name}-{start.int_timestamp * 1000}.json" + + r = report.as_dict() + r["Params"] = atn_params.as_dict() + json_object = json.dumps(r, indent=4) + + # Writing to sample.json + with open(file_name, "w") as outfile: + outfile.write(json_object) diff --git a/src/gwatn/strategies/simple_resistive_hydronic/node.py b/src/gwatn/strategies/simple_resistive_hydronic/node.py new file mode 100644 index 0000000..639994b --- /dev/null +++ b/src/gwatn/strategies/simple_resistive_hydronic/node.py @@ -0,0 +1,33 @@ +""" HeatPumpWithBoostStore DNode Definition +""" +from typing import Optional + +from gwatn.data_classes.d_node import DNode + + +SIG_FIGS_FOR_OUTPUT = 6 + + +class Node_BrickStorageHeater(DNode): + def __init__( + self, + ts_idx: int, + store_idx: int, + store_enthalpy_kwh: Optional[int] = None, + store_avg_brick_temp_c: Optional[float] = None, + ): + DNode.__init__( + self, + ts_idx=ts_idx, + store_idx=store_idx, + ) + self.store_enthalpy_kwh = store_enthalpy_kwh + self.store_avg_brick_temp_c = store_avg_brick_temp_c + + def __repr__( + self, + ) -> str: + rep = f"DNode => TimeSliceIdx: {self.ts_idx}, StoreIdx: {self.store_idx}" + if self.path_cost: + rep += f", Path cost: ${round(self.path_cost, 3)}" + return rep diff --git a/src/gwatn/strategies/simple_resistive_hydronic/output_data/__init__.py b/src/gwatn/strategies/simple_resistive_hydronic/output_data/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/src/gwatn/strategies/simple_resistive_hydronic/simple_scada_sim.py b/src/gwatn/strategies/simple_resistive_hydronic/simple_scada_sim.py new file mode 100644 index 0000000..bfb823d --- /dev/null +++ b/src/gwatn/strategies/simple_resistive_hydronic/simple_scada_sim.py @@ -0,0 +1,643 @@ +""" SCADA Actor """ +import functools +import logging +import random +import time +from typing import Optional +from typing import no_type_check + +import dotenv +import gridworks.algo_utils as algo_utils +import pendulum +import requests +from algosdk.atomic_transaction_composer import TransactionWithSigner +from algosdk.future.transaction import * +from algosdk.v2client.algod import AlgodClient +from beaker.client import ApplicationClient +from gridworks.actor_base import ActorBase +from gridworks.algo_utils import BasicAccount +from gridworks.enums import GNodeRole +from gridworks.message import as_enum +from pydantic import BaseModel + +import gwatn.api_types as api_types +import gwatn.config as config +from gwatn import DispatchContract +from gwatn.enums import AlgoCertType +from gwatn.enums import MessageCategorySymbol +from gwatn.enums import UniverseType +from gwatn.types import AtnParamsBrickstorageheater as AtnParams +from gwatn.types import DispatchContractConfirmed +from gwatn.types import DispatchContractConfirmed_Maker +from gwatn.types import GtDispatchBoolean +from gwatn.types import GtDispatchBoolean_Maker +from gwatn.types import GwCertId +from gwatn.types import GwCertId_Maker +from gwatn.types import HeartbeatA +from gwatn.types import HeartbeatA_Maker +from gwatn.types import HeartbeatB +from gwatn.types import HeartbeatB_Maker +from gwatn.types import JoinDispatchContract_Maker +from gwatn.types import ScadaCertTransfer_Maker +from gwatn.types import SimScadaDriverReportBsh as SimScadaDriverReport +from gwatn.types import SimScadaDriverReportBsh_Maker as SimScadaDriverReport_Maker +from gwatn.types import SimTimestep +from gwatn.types import SimTimestep_Maker +from gwatn.types import SnapshotBrickstorageheater as Snapshot + + +DISPATCH_CONTRACT_REPORTING_ALGOS = 5 + +LOG_FORMAT = ( + "%(levelname) -10s %(sasctime)s %(name) -30s %(funcName) " + "-35s %(lineno) -5d: %(message)s" +) +LOGGER = logging.getLogger(__name__) + +LOGGER.setLevel(logging.INFO) + + +def get_max_store_kwh_th(params: AtnParams) -> float: + room_temp_c = (params.RoomTempF - 32) * 5 / 9 + return params.C * (params.MaxBrickTempC - room_temp_c) + + +class AtnHbStatus(BaseModel): + LastHeartbeatReceivedMs: int + AtnLastHex: Optional[str] = None + ScadaLastHex: str = "0" + + +class ScadaActor(ActorBase): + def __init__( + self, + settings: config.ScadaSettings = config.ScadaSettings( + _env_file=dotenv.find_dotenv() + ), + ): + super().__init__(settings=settings) + self.api_type_maker_by_name = ( + api_types.TypeMakerByName + ) # overwrites base class to include types used in this repo + self.settings = settings + self.atn_gni_id = settings.atn_gni_id + self.acct: BasicAccount = BasicAccount(settings.sk.get_secret_value()) + self.client: AlgodClient = AlgodClient( + settings.algo_api_secrets.algod_token.get_secret_value(), + settings.public.algod_address, + ) + self.atn_params_type_name: str = "atn.params.brickstorageheater" + self.sp = self.client.suggested_params() + self.cert_id: Optional[GwCertId] = None + self.talking_with: bool = False + self.atn_hb_status = AtnHbStatus( + LastHeartbeatReceivedMs=int(1000 * time.time()) + ) + # The dc_client (DispatchContract client) builds on top of the Algorand beaker ApplicationClient + self.dc_client: Optional[ApplicationClient] = None + self.dc_app_id: Optional[int] = None + self.algo_init_check() + self.universe_type = as_enum( + self.settings.universe_type_value, UniverseType, UniverseType.default() + ) + self._time: float = self.get_initial_time_s() + + self.boost_power_kw: float = 0 + self.power_watts: float = 0 + self.cop: float = 0 + self.store_kwh: int = 0 + self.max_store_kwh: int = 0 + self.atn_params: Optional[AtnParams] = None + + @property + def atn_alias(self): + """Removes `scada` from the end of the SCADA's GNodeAlias""" + return self.alias[:-6] + + @property + def ta_alias(self): + """Removes `scada` from the end of the SCADA's GNodeAlias""" + return self.atn_alias + ".ta" + + def local_rabbit_startup(self) -> None: + rjb = MessageCategorySymbol.rjb.value + tc_alias_lrh = self.settings.time_coordinator_alias.replace(".", "-") + binding = f"{rjb}.{tc_alias_lrh}.timecoordinator.sim-timestep" + + cb = functools.partial(self.on_timecoordinator_bindok, binding=binding) + self._consume_channel.queue_bind( + self.queue_name, "timecoordinatormic_tx", routing_key=binding, callback=cb + ) + LOGGER.info( + f"Queue {self.queue_name} bound to timecoordinatormic_tx with {binding} " + ) + + @no_type_check + def on_timecoordinator_bindok(self, _unused_frame, binding) -> None: + LOGGER.info(f"Queue {self.queue_name} bound with {binding}") + + def time(self) -> float: + if self.universe_type == UniverseType.Dev: + return self._time + else: + return time.time() + + def get_initial_time_s(self) -> float: + if self.universe_type == UniverseType.Dev: + return self.settings.initial_time_unix_s + else: + return time.time() + + def prepare_for_death(self) -> None: + self.actor_main_stopped = True + + ######################## + ## Receives + ######################## + + def route_message( + self, from_alias: str, from_role: GNodeRole, payload: HeartbeatB + ) -> None: + if payload.TypeName == DispatchContractConfirmed_Maker.type_name: + if from_role != GNodeRole.AtomicTNode: + LOGGER.info( + f"Ignoring DispatchContractConfrimed from GNode with role {from_role}; expects AtomicTNode" + ) + try: + self.dispatch_contract_confirmed_received(payload) + except: + LOGGER.exception("Error in dispatch_contract_confirmed_received") + elif payload.TypeName == GtDispatchBoolean_Maker.type_name: + if from_role != GNodeRole.AtomicTNode: + LOGGER.info( + f"Ignoring GtDispatchBooleanfrom GNode with role {from_role}; expects AtomicTNode" + ) + try: + self.dispatch_received(payload) + except: + LOGGER.exception("Error in dispatch_received") + + elif payload.TypeName == HeartbeatA_Maker.type_name: + if from_role != GNodeRole.Supervisor: + LOGGER.info( + f"Ignoring HeartbeatA from GNode with role {from_role}; expects Supervisor" + ) + try: + self.heartbeat_from_super(from_alias=from_alias, ping=payload) + except: + LOGGER.exception("Error in heartbeat_from_super") + elif payload.TypeName == HeartbeatB_Maker.type_name: + if from_role != GNodeRole.AtomicTNode: + LOGGER.info( + f"Ignoring HeartbeatB from GNode with role {from_role}; expects AtomicTNode" + ) + try: + self.heartbeat_from_atn(ping=payload) + except: + LOGGER.exception("Error in heartbeat_from_atn") + elif payload.TypeName == SimScadaDriverReport_Maker.type_name: + try: + self.sim_scada_driver_report_received(payload) + except: + LOGGER.exception("Error in sim_scada_driver_report_received") + elif payload.TypeName == SimTimestep_Maker.type_name: + if from_role != GNodeRole.TimeCoordinator: + LOGGER.info( + f"Ignoring SimTimestep from GNode with role {from_role}; expects TimeCoordinator" + ) + try: + self.timestep_from_timecoordinator(payload) + except: + LOGGER.exception("Error in timestep_from_timecoordinator") + + def heartbeat_from_atn(self, ping: HeartbeatB) -> None: + """ + This is the Scada's half of the DispatchContract Heartbeat pattern. + It: + - Checks that it has a DispatchContract (owns that SmartContract) + - Checks the GNodeAlias and GNodeInstanceId to validate partner + - Sends a reply HeartbeatB immediately back using a RabbitJsonDirect message + - Sends an audit report of its action to the DispatchContract + # TODO: save audit report for sending in a batch if SmartContract + # exists but is not reachable (i.e. blockchain down) + [more info](https://gridworks.readthedocs.io/en/latest/dispatch-contract.html) + + Args: + payload (HeartbeatB): The latest heartbeat received from its + AtomicTNode partner + + """ + self.ping = ping + + received_ms = int(time.time() * 1000) + if not self.in_dispatch_contract(): + LOGGER.info(f"Not in Dispatch Contract. Ignoring") + if ping.FromGNodeInstanceId != self.atn_gni_id: + raise Exception("In dispatch contract but mismatched Atn Gni Id!") + if self.atn_hb_status.AtnLastHex is not None: + if ping.YourLastHex != self.atn_hb_status.ScadaLastHex: + LOGGER.info("Received incorrect ping. Ignoring") + return + self.talking_with = True + self.atn_hb_status.LastHeartbeatReceivedMs = int(time.time() * 1000) + self.atn_hb_status.AtnLastHex = ping.MyHex + self.atn_hb_status.ScadaLastHex = str(random.choice("0123456789abcdef")) + LOGGER.info(f"Got {ping.MyHex}. Sending {self.atn_hb_status.ScadaLastHex}") + pong = HeartbeatB_Maker( + from_g_node_alias=self.alias, + from_g_node_instance_id=self.g_node_instance_id, + my_hex=self.atn_hb_status.ScadaLastHex, + your_last_hex=self.atn_hb_status.AtnLastHex, + last_received_time_unix_ms=self.atn_hb_status.LastHeartbeatReceivedMs, + send_time_unix_ms=int(1000 * time.time()), + ).tuple + self.send_message( + payload=pong, to_role=GNodeRole.AtomicTNode, to_g_node_alias=self.atn_alias + ) + # Report to DispatchContract with heartbeat.algo.audit + ptxn = PaymentTxn(self.acct.addr, self.sp, self.dc_client.app_addr, 1000) + self.dc_client.call( + DispatchContract.heartbeat_algo_audit, + signed_proof=TransactionWithSigner(ptxn, self.acct.as_signer()), + heartbeat=pong.as_dict(), + ) + + def heartbeat_from_super(self, from_alias: str, ping: HeartbeatA) -> None: + pong = HeartbeatA_Maker( + my_hex=str(random.choice("0123456789abcdef")), your_last_hex=ping.MyHex + ).tuple + + self.send_message( + payload=pong, + to_role=GNodeRole.Supervisor, + to_g_node_alias=self.settings.my_super_alias, + ) + + LOGGER.debug( + f"[{self.alias}] Sent HB: SuHex {pong.YourLastHex}, AtnHex {pong.MyHex}" + ) + + def timestep_from_timecoordinator(self, payload: SimTimestep): + if self._time == 0: + self._time = payload.TimeUnixS + self.new_timestep(payload) + LOGGER.info(f"TIME STARTED: {self.time_str()}") + elif self._time < payload.TimeUnixS: + self._time = payload.TimeUnixS + self.new_timestep(payload) + LOGGER.debug(f"Time is now {self.time_str()}") + elif self._time == payload.TimeUnixS: + self.repeat_timestep(payload) + + ######################################################## + # Related to Dispatch Contract and Scada Cert + ######################################################## + + def algo_init_check(self): + """Upon startup, see if the dispatch contract has already been created + and also make sure Scada is appropriately funded""" + + # First, request cert transfer. This is idempotent - if scada + # addr already owns the ScadaCertificate it does nothing + + self.request_cert_transfer() + apps = self.client.account_info(self.acct.addr)["created-apps"] + balances = algo_utils.get_balances(self.client, self.acct.addr) + micro_algos = balances[0] + if micro_algos < 100_000: + raise Exception(f"Insufficient funds to even opt into SCADA cert!") + if len(apps) > 1: + raise Exception("Should only be part of one app") + if not self.dispatch_contract_created(): + # Create, fund and do bootstrap1 of the Dispatch contract + dc_algos = DISPATCH_CONTRACT_REPORTING_ALGOS + ( + DispatchContract.min_balance / 1_000_000 + ) + if micro_algos < (dc_algos + 2) * 1_000_000: + raise Exception(f"Insufficient funding! Need at least {dc_algos + 2}") + self.dc_client = ApplicationClient( + self.client, DispatchContract(), signer=self.acct.as_signer() + ) + app_id, app_address, transaction_id = self.dc_client.create() + self.dc_app_id = app_id + LOGGER.info(f"Created Dispatch Contract, app id {self.dc_app_id}") + sp = self.dc_client.get_suggested_params() + sp.flat_fee = True + sp.fee = 2000 + ptxn = PaymentTxn( + self.acct.addr, + sp, + self.dc_client.app_addr, + DispatchContract.min_balance, + ) + + # Now do the first half of bootstrapping the contract by providing it + # with enough money to fund about 32kB in boxes, along with + # our ScadaCertId (which can be shown to match our address in the sig) + + result = self.dc_client.call( + DispatchContract.bootstrap1, + scada_seed=TransactionWithSigner(ptxn, self.acct.as_signer()), + ScadaCert=self.cert_id.Idx, + ) + LOGGER.info("Called bootstrap1 method of DispatchContract") + LOGGER.info( + f"Providing the scada certificate, and funding with {dc_algos} Algos" + ) + if result.return_value != self.ta_alias: + raise Exception( + f"Expected {self.ta_alias} back from bootstrapping app {self.dc_app_id}, but" + f" got {result.return_value}" + ) + else: + created_apps = self.client.account_info(self.acct.addr)["created-apps"] + self.dc_app_id = created_apps[0]["id"] + self.dc_client = ApplicationClient( + client=self.client, + app=DispatchContract(), + signer=self.acct.as_signer(), + app_id=self.dc_app_id, + ) + LOGGER.info(f"Dispatch contract already exists, app id {self.dc_app_id}") + self.in_dispatch_contract() + + def request_cert_transfer(self) -> None: + """ + Opts into certificate and then request transfer of the ScadaCert from the GNodeFactory + + Raises: + Exception if the scada certificate does not exist, or transfer is not successful + """ + cert_type = AlgoCertType(self.settings.cert_type_value) + if cert_type == AlgoCertType.SmartSig: + raise NotImplementedError("Not prepared for SmartSig Certificates yet") + + # Look for cert id + gnf = self.settings.public.gnf_admin_addr + a = self.client.account_info(gnf)["created-assets"] + scada_nft_ids = list( + filter( + lambda x: x["params"]["unit-name"] == "SCADA" + and x["params"]["name"] == self.ta_alias, + a, + ) + ) + if len(scada_nft_ids) == 0: + raise Exception("Scada certificate does not exist") + + self.cert_id = GwCertId_Maker( + type=AlgoCertType.ASA, idx=scada_nft_ids[0]["index"], addr=None + ).tuple + + balances = algo_utils.get_balances(self.client, self.acct.addr) + + # First, opt into sig + txn = AssetOptInTxn(self.acct.addr, self.sp, self.cert_id.Idx) + + signed_txn = txn.sign(self.acct.sk) + if self.cert_id.Idx not in balances.keys(): + try: + self.client.send_transaction(signed_txn) + except: + raise Exception( + "Failure sending transaction on Algo blockchain. Check sandbox and funding" + ) + algo_utils.wait_for_transaction(self.client, signed_txn.get_txid()) + + balances = algo_utils.get_balances(self.client, self.acct.addr) + + if balances[self.cert_id.Idx] == 1: + # already own the cert + return + payload = ScadaCertTransfer_Maker( + ta_alias="d1.isone.ver.keene.holly.ta", # REPLACE WITH api_util.get_alias_from_scada_cert + signed_proof=encoding.msgpack_encode(signed_txn), + ).tuple + + api_endpoint = f"{self.settings.public.gnf_api_root}/scada-cert-transfer/" + try: + r = requests.post(url=api_endpoint, json=payload.as_dict()) + except: + raise Exception(f"Post to {api_endpoint} failed") + if r.status_code > 200: + if r.status_code == 422: + note = f"Error entering SLA: " + r.json()["detail"] + else: + note = r.reason + raise Exception(f"Post to {api_endpoint} failed: {note} ") + + # check that scada now owns cert + balances = algo_utils.get_balances(self.client, self.acct.addr) + if balances[self.cert_id.Idx] != 1: + raise Exception(f"Gnf claimed successful transfer but I do not own cert") + + LOGGER.info(f"Successfully received Scada Cert ASA {self.cert_id.Idx}") + + def initialize_dispatch_contract(self): + """Starts up Dispatch contract to be used with AtomicTNode. + + + Raises exception if Scada Addr + - scada cert from settings matches what is in my acct + - already has created an app + - does not already have a unique Scada Cert ASA + """ + + if self.in_dispatch_contract(): + LOGGER.warning( + f"Already in dispatch contract. Ignoring initialize_dispatch_contract!" + ) + return + + balances = algo_utils.get_balances(self.client, self.acct.addr) + non_algo_asas = list(set(balances.keys()) - {0}) + scada_certs = list( + filter( + lambda x: self.client.asset_info(x)["params"]["unit-name"] == "SCADA", + non_algo_asas, + ) + ) + + if len(scada_certs) > 1: + raise Exception( + f"Scada {self.alias} has 2 scada certs. This should not happen!" + ) + if len(scada_certs) == 0: + raise Exception( + "Scada acct needs to be created and funded by TaOwner, and Scada GNode " + "needs to be authorized (with GNodeFactory sending Scada cert) " + f"before starting up {self.alias}" + ) + + if self.cert_id.Idx != scada_certs[0]: + raise Exception( + f"Scada cert in settings {self.cert_id.Idx} does not match" + f"scada cert in my acct {scada_certs[0]}" + ) + + txn = PaymentTxn( + sender=self.acct.addr, + receiver=self.dc_client.app_addr, + sp=self.sp, + amt=1000, + ) + signed_txn = txn.sign(self.acct.sk) + # Don't need to actually send it in order to establish signature + payload = JoinDispatchContract_Maker( + from_g_node_alias=self.alias, + from_g_node_instance_id=self.g_node_instance_id, + dispatch_contract_app_id=self.dc_app_id, + signed_proof=encoding.msgpack_encode(signed_txn), + ).tuple + + self.send_message( + payload=payload, + to_role=GNodeRole.AtomicTNode, + to_g_node_alias=self.atn_alias, + ) + LOGGER.info(f"Sent d = {payload.as_dict()}") + + def dispatch_contract_confirmed_received(self, payload: DispatchContractConfirmed): + """Check the state of the app to confirm that the Atn has finished the bootstrap. + Then + - set the atn_gni_id to GNodeInstanceId from the payload (this is how the Scada + will confirm identity of the AtomicTNode + - set the atn_params to those shared by the AtomicTNode + - opt into the Dispatch Contract (now the DispatchContract is live)""" + if self.dc_app_id is None: + raise Exception( + "Thats odd ... dont have a Dispatch Contract App Id and I need to be" + "the creator" + ) + if self.in_dispatch_contract(): + LOGGER.warning( + f"Ignoring DispatchContractConfirmed - already in dispatch contract {self.dc_app_id}" + ) + return + + if payload.AtnParamsTypeName != self.atn_params_type_name: + LOGGER.info( + f"Unexpected AtnParamsTypeName {payload.AtnParamsTypeName}. Expects {self.atn_params_type_name}" + ) + return + self.atn_params = payload.Params + app_state = self.dc_client.get_application_state() + if "ta_trading_rights_idx" not in app_state.keys(): + LOGGER.warning(f"Atn bootstrap is not finished. Ignoring") + return + + self.atn_gni_id = payload.FromGNodeInstanceId + + with open(".env", "a") as file: + file.write(f'SCADA_ATN_GNI_ID="{payload.FromGNodeInstanceId}"\n') + self.dc_client.opt_in() + LOGGER.info(f"Dispatch Contract {self.dc_app_id} is live") + + ######################################################### + # Make the below into abstractmethods if pulling out base class + ######################################################### + + def sim_scada_driver_report_received(self, payload: SimScadaDriverReport) -> None: + """This gets received right before the top of the hour, from our + best simulation of the TerminalAsset (which is happening in the + AtomicTNode).""" + if payload.FromGNodeInstanceId != self.atn_gni_id: + LOGGER.info(f"Igoring {payload} - incorrect GNodeInstanceId") + self.power_watts = payload.PowerWatts + self.store_kwh = payload.StoreKwh + self.max_store_kwh = payload.MaxStoreKwh + self.send_snapshot() + + def send_snapshot(self): + """Send a snapshot of current core sensed values to AtomicTNode. + This is done every hour, and also on sensed power change.""" + if self.atn_params is None: + return + report_payload = Snapshot( + FromGNodeAlias=self.alias, + FromGNodeInstanceId=self.g_node_instance_id, + PowerWatts=self.power_watts, + StoreKwh=self.store_kwh, + MaxStoreKwh=get_max_store_kwh_th(self.atn_params), + AboutTerminalAssetAlias=self.ta_alias, + ) + self.send_message( + payload=report_payload, + to_role=GNodeRole.AtomicTNode, + to_g_node_alias=self.atn_alias, + ) + + def dispatch_received(self, payload: GtDispatchBoolean) -> None: + """ + Dispatch received from AtomicTNode + + - Checks that the GNodeAlias and GNodeInstanceId belong to its + AtomicTNode and that we have a dispatch contract + - Sets talking_with to true + - Follows instructions (turns on or turns off). Will turn on or + off boost unless AboutNodeName is "a.element" + + For hourly sim: + - Updatespower + - Send status to AtomicTNode + """ + if self.atn_params is None or self.in_dispatch_contract() is False: + LOGGER.info("Igoring dispatch command, DispatchContract is not started") + if payload.FromGNodeInstanceId != self.atn_gni_id: + LOGGER.info(f"Igoring {payload}, not my Atn's GNodeInstanceId") + self.talking_with = True + if payload.AboutNodeName == "a.elements": + # Making the grossly simplifying assumption that the heat pump turns on immediately + if payload.RelayState == 1: + new_power_watts = self.atn_params.RatedMaxPowerKw * 1000 + else: + new_power_watts = 0 + if self.power_watts != new_power_watts: + self.power_watts = new_power_watts + self.send_snapshot() + + def new_timestep(self, payload: SimTimestep) -> None: + # LOGGER.info("New timestep") + if not self.in_dispatch_contract(): + self.initialize_dispatch_contract() + + def repeat_timestep(self, payload: SimTimestep) -> None: + LOGGER.info("Timestep received again") + + def time_str(self) -> str: + return pendulum.from_timestamp(self.time()).strftime("%m/%d/%Y, %H:%M") + + def dispatch_contract_created(self) -> bool: + """The SCADA only creates one kind of smart contract, which is its dispatch contract. + So this just checks for a created app + """ + created_apps = self.client.account_info(self.acct.addr)["created-apps"] + if len(created_apps) > 1: + raise NotImplementedError( + f"SCADA not designed yet to create multiple dispatch contracts" + ) + if len(created_apps) > 0: + return True + return False + + def in_dispatch_contract(self) -> bool: + """Checks that bootstrap 2 was completed w addition of ta_trading_rights_idx, + and also that Scada is opted in""" + app_state = self.dc_client.get_application_state() + if "ta_trading_rights_idx" not in app_state.keys(): + LOGGER.info( + "Not in Dispatch Contract because ta_trading_rights_idx isn't in app_state" + ) + LOGGER.info( + "AtomicTNode has not finished its part in bootstrapping the contract" + ) + return False + # Todo: also check that atn is opted in + apps = self.client.account_info(self.acct.addr)["apps-local-state"] + app_ids = list(map(lambda x: x["id"], apps)) + if self.dc_client.app_id in app_ids: + return True + LOGGER.info( + "Not in Dispatch Contract because scada has not yet opted into contract" + ) + return False diff --git a/src/gwatn/strategies/simple_resistive_hydronic/strategy_utils.py b/src/gwatn/strategies/simple_resistive_hydronic/strategy_utils.py new file mode 100644 index 0000000..2b8c0a9 --- /dev/null +++ b/src/gwatn/strategies/simple_resistive_hydronic/strategy_utils.py @@ -0,0 +1,128 @@ +from typing import Optional + +import gridworks.conversion_factors as cf # TODO change to from gwatn import conversion_factors as cf +import numpy as np +from pydantic import BaseModel +from satn.strategies.heatpumpwithbooststore.flo import ( + Flo__HeatpumpWithBoostStore as Flo, +) +from satn.types import AtnParamsHeatpumpwithbooststore as AtnParams +from satn.types import FloParamsHeatpumpwithbooststore as FloParams + +from gwatn import atn_utils +from gwatn.types import AtnBid +from gwatn.types import MarketSlot + + +################################# +# Bidding related +################################# + + +class CostAndQuantityBought(BaseModel): + QuantityBought: float + Cost: float + + +class SlotStuff(BaseModel): + Slot: MarketSlot + BidParams: Optional[FloParams] = None + Flo: Optional[Flo] = None + Bid: Optional[AtnBid] = None + Price: Optional[float] = None + + +def dummy_bid() -> AtnBid: + return AtnBid( + BidderAlias="d1.isone.dummy.ta", + BidderGNodeInstanceId="00000000-0000-0000-0000-000000000000", + MarketSlotName="rt60gate30b.d1.isone.ver.keene.1577836800", + PqPairs=[], + SignedMarketFeeTxn=atn_utils.DUMMY_ALGO_TXN, + ) + + +def dummy_atn_params() -> AtnParams: + return AtnParams( + SliceDurationMinutes=60, + FloSlices=48, + GNodeAlias="d1.isone.dummy.ta", + GNodeInstanceId="00000000-0000-0000-0000-000000000000", + TypeName="atn.params.heatpumpwithbooststore", + Version="000", + ) + + +def dummy_flo_params() -> FloParams: + return FloParams( + GNodeAlias="d1.isone.dummy.ta", + FloParamsUid="00000000-0000-0000-0000-000000000000", + RtElecPriceUid="00000000-0000-0000-0000-000000000000", + WeatherUid="00000000-0000-0000-0000-000000000000", + ) + + +def dummy_slot_stuff(slot: MarketSlot) -> SlotStuff: + return SlotStuff( + Slot=slot, FloStartIdx=50, BidParams=dummy_flo_params(), AtnBid=dummy_bid() + ) + + +def is_dummy_slot_stuff(bid_stuff: SlotStuff) -> bool: + if is_dummy_flo_params(bid_stuff.BidParams): + return True + return False + + +def is_dummy_atn_params(atn_params: AtnParams) -> bool: + if atn_params.GNodeAlias == "d1.isone.dummy.ta": + return True + return False + + +def is_dummy_flo_params(flo_params: FloParams) -> bool: + if flo_params.RtElecPriceUid == "00000000-0000-0000-0000-000000000000": + return True + return False + + +########################################## +# Flo prep (uses AtnParams as variable) +########################################## + + +def get_k( + system_max_heat_output_delta_temp_f: int, + system_max_heat_output_gpm: float, + system_max_heat_output_swt_f: int, + room_temp_f: int, +) -> float: + dt = system_max_heat_output_delta_temp_f + gpm = system_max_heat_output_gpm + swt = system_max_heat_output_swt_f + rt = room_temp_f + return float(gpm * np.log(1 - dt / (swt - rt))) + + +def get_system_max_heat_output_kw_avg( + system_max_heat_output_gpm: float, system_max_heat_output_delta_temp_f: float +) -> float: + """What is the max heat that the system put out? ASSUMES that the hydronic + fluid is water. + + Args: + tea_params: Uses the system_max_heat_output_delta_temp_f and + the system_max_heat_output_gpm. + + Returns: + float: max heat ouput of the heating system, in kw + """ + gpm = system_max_heat_output_gpm + delta_temp = system_max_heat_output_delta_temp_f + c = cf.POUNDS_OF_WATER_PER_GALLON * cf.MINUTES_PER_HOUR / cf.BTU_PER_KWH + if gpm == 0 or delta_temp == 0: + raise Exception( + f"max gpm is {gpm}, max delta_temp is {delta_temp}. Cannot calculate system heat output." + ) + pwr = c * delta_temp * gpm + return pwr diff --git a/src/gwatn/strategies/simple_resistive_hydronic/tea.py b/src/gwatn/strategies/simple_resistive_hydronic/tea.py new file mode 100644 index 0000000..8e25ab2 --- /dev/null +++ b/src/gwatn/strategies/simple_resistive_hydronic/tea.py @@ -0,0 +1,264 @@ +import csv +import uuid +from typing import List + +import numpy as np +import pendulum +import satn.dev_utils.price.distp_sync_100_handler as distp_sync_100_handler +import satn.dev_utils.price.eprt_sync_100_handler as eprt_sync_100_handler +import satn.dev_utils.weather.weather_forecast_sync_100_handler as weather_forecast_sync_100_handler +import satn.strategies.heatpumpwithbooststore.flo_utils as flo_utils +import satn.strategies.heatpumpwithbooststore.strategy_utils as strategy_utils +from satn.enums import ShDistPumpFeedbackModel +from satn.enums import ShMixingValveFeedbackModel +from satn.strategies.heatpumpwithbooststore.flo import Flo__HeatpumpWithBoostStore +from satn.strategies.heatpumpwithbooststore.tea_config import TeaParams +from satn.strategies.heatpumpwithbooststore.tea_output import export_xlsx +from satn.types.flo_params_heatpumpwithbooststore import ( + FloParamsHeatpumpwithbooststore as FloParams, +) + +import gwatn.errors as errors +from gwatn.enums import DistributionTariff +from gwatn.enums import EnergySupplyType +from gwatn.enums import RecognizedCurrencyUnit +from gwatn.enums import RecognizedTemperatureUnit +from gwatn.types.ps_distprices_gnode.r_distp_sync.r_distp_sync_1_0_0 import ( + Payload as DistpSync100Payload, +) +from gwatn.types.ps_electricityprices_gnode.r_eprt_sync.r_eprt_sync_1_0_0 import ( + Payload as EprtSync100Payload, +) +from gwatn.types.ws_forecast_gnode.r_weather_forecast_sync.r_weather_forecast_sync_1_0_0 import ( + Payload as RWeatherForecastSync100Payload, +) + + +def get_electricity_prices(tea_params: TeaParams) -> EprtSync100Payload: + ep = eprt_sync_100_handler.payload_from_file( + eprt_type_name=tea_params.real_time_electricity_price_type_name, + eprt_csv=tea_params.real_time_electricity_price_csv, + csv_starting_offset_hours=tea_params.price_csv_starting_offset_hours, + flo_total_time_hrs=tea_params.flo_total_time_hrs, + ) + if ep.CurrencyUnit != tea_params.currency_unit: + raise Exception( + f"Currency unit for {tea_params.real_time_electricity_price_csv} does not match params.currency_unit of {tea_params.currency_unit}" + ) + return ep + + +def get_flo_start_utc(tea_params: TeaParams) -> pendulum.datetime: + ep = get_electricity_prices(tea_params) + return pendulum.datetime( + year=ep.StartYearUtc, + month=ep.StartMonthUtc, + day=ep.StartDayUtc, + hour=ep.StartHourUtc, + minute=ep.StartMinuteUtc, + ) + + +def get_distribution_prices( + tea_params: TeaParams, flo_start_utc: pendulum.datetime +) -> DistpSync100Payload: + dp = distp_sync_100_handler.payload_from_file( + distp_type_name=tea_params.dist_price_type_name, + distp_csv=tea_params.dist_price_csv, + flo_start_utc=flo_start_utc, + flo_total_time_hrs=tea_params.flo_total_time_hrs, + ) + if dp.CurrencyUnit != tea_params.currency_unit: + raise Exception( + f"currency unit for {tea_params.dist_price_type_name} does not match params.currency_unit of {tea_params.currency_unit}" + ) + return dp + + +def get_weather_forecasts( + tea_params: TeaParams, flo_start_utc: pendulum.datetime +) -> RWeatherForecastSync100Payload: + wp = weather_forecast_sync_100_handler.payload_from_file( + file_name=tea_params.weather_csv, + request_start_datetime_utc=flo_start_utc, + total_time_hrs=tea_params.flo_total_time_hrs, + ) + if wp.TempUnit != tea_params.temp_unit: + raise Exception( + f"temp unit for {tea_params.weather_csv} does not match params.temp_unit of {tea_params.temp_unit}" + ) + return wp + + +def get_desired_heat_from_csv(tea_params: TeaParams) -> List[float]: + ep = get_electricity_prices(tea_params) + if tea_params.price_csv_starting_offset_hours != 0: + raise NotImplementedError( + f"heat profile has not been adjusted to offsets from start of year" + ) + + with open(tea_params.scaled_heat_profile_csv, newline="") as csvfile: + reader = csv.reader(csvfile, delimiter=",") + p = [] + for row in reader: + p.append(float(row[0])) + if len(p) != 8784: + raise Exception( + f"scaled heat profile {tea_params.scaled_heat_profile_csv} should have 8784 hours" + ) + annual_desired_house_heat_profile = ( + tea_params.annual_hvac_kwh_th * np.array(p) + ).tolist() + return annual_desired_house_heat_profile[0 : len(ep.Prices)] + + +def get_flo_params(tea_params: TeaParams) -> FloParams: + ep = get_electricity_prices(tea_params) + flo_start_utc = get_flo_start_utc(tea_params) + dp = get_distribution_prices(tea_params, flo_start_utc) + weather = get_weather_forecasts(tea_params, flo_start_utc) + pump_model = ShDistPumpFeedbackModel(tea_params.emitter_pump_feedback_model_value) + mixing_valve_model = ShMixingValveFeedbackModel( + tea_params.mixing_valve_feedback_model_value + ) + # PowerRequiredByHouseFromSystemAvgKw + power_required_by_house_from_system_avg_kw_list: List[ + float + ] = get_desired_heat_from_csv(tea_params) + + flo_params = FloParams( + GNodeAlias="d1.tea.atn", + FloParamsUid=str(uuid.uuid4()), + PowerRequiredByHouseFromSystemAvgKwList=power_required_by_house_from_system_avg_kw_list, + HouseWorstCaseTempF=tea_params.house_worst_case_temp_f, + EmitterMaxSafeSwtF=tea_params.emitter_max_safe_swt_f, + CirculatorPumpMaxGpm=tea_params.circulator_pump_max_gpm, + SystemMaxHeatOutputKwAvg=strategy_utils.get_system_max_heat_output_kw_avg( + system_max_heat_output_gpm=tea_params.system_max_heat_output_gpm, + system_max_heat_output_delta_temp_f=tea_params.system_max_heat_output_delta_temp_f, + ), + RtElecPriceUid=ep.PriceUid, + DistPriceUid=dp.PriceUid, + RegPriceUid=None, + WeatherUid=weather.WeatherUid, + K=strategy_utils.get_k( + system_max_heat_output_delta_temp_f=tea_params.system_max_heat_output_delta_temp_f, + system_max_heat_output_gpm=tea_params.system_max_heat_output_gpm, + system_max_heat_output_swt_f=tea_params.system_max_heat_output_swt_f, + room_temp_f=tea_params.room_temp_f, + ), + StartYearUtc=ep.StartYearUtc, + StartMonthUtc=ep.StartMonthUtc, + StartDayUtc=ep.StartDayUtc, + StartHourUtc=ep.StartHourUtc, + StartMinuteUtc=ep.StartMinuteUtc, + TimezoneString=tea_params.timezone_string, + SliceDurationMinutes=[ep.UniformSliceDurationHrs * 60] * len(ep.Prices), + HomeCity=tea_params.home_city, + AmbientTempStoreF=tea_params.ambient_temp_store_f, + StorePassiveLossRatio=tea_params.store_passive_loss_ratio, + SystemMaxHeatOutputDeltaTempF=tea_params.system_max_heat_output_delta_temp_f, + SystemMaxHeatOutputGpm=tea_params.system_max_heat_output_gpm, + SystemMaxHeatOutputSwtF=tea_params.system_max_heat_output_swt_f, + IsRegulating=False, + OutsideTempF=weather.Temperatures, + HeatpumpTariff=DistributionTariff(tea_params.heatpump_tariff_value), + HeatpumpEnergySupplyType=EnergySupplyType( + tea_params.heatpump_energy_supply_type_value + ), + BoostTariff=DistributionTariff(tea_params.boost_tariff_value), + BoostEnergySupplyType=EnergySupplyType( + tea_params.boost_energy_supply_type_value + ), + StandardOfferPriceDollarsPerMwh=tea_params.standard_offer_price_dollars_per_mwh, + DistributionTariffDollarsPerMwh=tea_params.flat_tariff_dollars_per_mwh, + RealtimeElectricityPrice=ep.Prices, + DistributionPrice=dp.Prices, + RoomTempF=tea_params.room_temp_f, + AmbientPowerInKw=tea_params.ambient_power_in_kw, + MaxHeatpumpSourceWaterTempF=tea_params.max_heatpump_source_water_temp_f, + ZeroPotentialEnergyWaterTempF=tea_params.zero_potential_energy_water_temp_f, + MaxStoreTempF=tea_params.max_store_temp_f, + StorageSteps=tea_params.storage_steps, + StoreSizeGallons=tea_params.store_size_gallons, + RatedHeatpumpElectricityKw=tea_params.rated_heatpump_electricity_kw, + StoreMaxPowerKw=tea_params.store_max_power_kw, + DistPumpFeedbackModel=pump_model, + MixingValveFeedbackModel=mixing_valve_model, + CautiousMixingValveTempDeltaF=tea_params.cautious_mixing_valve_temp_delta_f, + TempUnit=RecognizedTemperatureUnit(tea_params.temp_unit), + CurrencyUnit=RecognizedCurrencyUnit(tea_params.currency_unit), + StartingIdx=int(tea_params.storage_steps / 2), + Cop1TempF=tea_params.cop_1_temp_f, + Cop4TempF=tea_params.cop_4_temp_f, + RegulationPrice=[], + ) + + return flo_params + + +def get_flo(flo_params: FloParams) -> Flo__HeatpumpWithBoostStore: + return Flo__HeatpumpWithBoostStore( + params=flo_params, + d_graph_id=str(uuid.uuid4()), + ) + + +if __name__ == "__main__": + tea_params = TeaParams(_env_file="tea_params/heatpumpwithbooststore.env") + flo_params = get_flo_params(tea_params) + p = flo_params.PowerRequiredByHouseFromSystemAvgKwList + t = flo_params.OutsideTempF + house_dd_max_heat_kw = flo_utils.get_house_worst_case_heat_output_avg_kw(flo_params) + + print(f"Max required heat this run: {round(max(p),2)} kW") + print(f"Max required heat on design day:{round(house_dd_max_heat_kw,2)} kW") + print( + f"flo_params.SystemMaxHeatOutputKwAvg: {round(flo_params.SystemMaxHeatOutputKwAvg,2)} kW" + ) + if max(p) > flo_params.SystemMaxHeatOutputKwAvg: + # if house_dd_max_heat_kw > flo_params.SystemMaxHeatOutputKwAvg: + raise errors.PhysicalSystemFailure( + f"House requires {round(house_dd_max_heat_kw,2)} kW on the" + f" design day but flo_params.SystemMaxHeatOutputKwAvg is only {round(flo_params.SystemMaxHeatOutputKwAvg,2)} kW!" + ) + if flo_params.CirculatorPumpMaxGpm < flo_params.SystemMaxHeatOutputGpm: + raise errors.PhysicalSystemFailure( + f"CirculatorPumpMaxGpm {flo_params.CirculatorPumpMaxGpm} is less than" + f" SystemMaxHeatOutputKwAvg {round(flo_params.SystemMaxHeatOutputKwAvg,2)}" + ) + if ( + flo_params.MixingValveFeedbackModel + == ShMixingValveFeedbackModel.NaiveVariableSwt + ): + if flo_params.EmitterMaxSafeSwtF < flo_params.SystemMaxHeatOutputSwtF: + raise errors.PhysicalSystemFailure( + f".EmitterMaxSafeSwtF {flo_params.EmitterMaxSafeSwtF} is less than" + f" SystemMaxHeatOutputSwtF {round(flo_params.SystemMaxHeatOutputSwtF,1)}" + ) + if ( + flo_params.MixingValveFeedbackModel + == ShMixingValveFeedbackModel.CautiousVariableSwt + ): + if ( + flo_params.EmitterMaxSafeSwtF + < flo_params.SystemMaxHeatOutputSwtF + + flo_params.CautiousMixingValveTempDeltaF + ): + raise errors.PhysicalSystemFailure( + "MixingValveModel: CautiousVariableSwt." + f" EmitterMaxSafeSwtF {flo_params.EmitterMaxSafeSwtF} is less than" + " SystemMaxHeatOutputSwtF - CautiousMixingValveTempDeltaF: " + f"{round(flo_params.SystemMaxHeatOutputSwtF - flo_params.CautiousMixingValveTempDeltaF,1)}" + ) + + print(f"Coldest temp this run: {min(t)} F") + print(f"Design day temp: {flo_params.HouseWorstCaseTempF} F") + swt = flo_utils.get_source_water_temp_f_list(params=flo_params) + + print(f"Max swt : {round(max(swt),1)}") + flo = get_flo(flo_params) + + if flo.node[0][50].path_cost > 100000: + print("FAILED") + export_xlsx(tea_params=tea_params, flo=flo, export_graph=tea_params.export_graph) diff --git a/src/gwatn/strategies/simple_resistive_hydronic/tea_config.py b/src/gwatn/strategies/simple_resistive_hydronic/tea_config.py new file mode 100644 index 0000000..4e542c0 --- /dev/null +++ b/src/gwatn/strategies/simple_resistive_hydronic/tea_config.py @@ -0,0 +1,76 @@ +"""Settings for the GridWorks Scada, readable from environment and/or from env files.""" +from pydantic import BaseSettings +from satn.enums import ShDistPumpFeedbackModel +from satn.enums import ShMixingValveFeedbackModel + + +DEFAULT_ENV_FILE = "../tea_settings/heatpumpwithbooststore.env" + + +class TeaParams(BaseSettings): + """Settings for the HeatPumpWithBoostStore Technoeconomic analysis.""" + + house_worst_case_temp_f: int = -7 + emitter_max_safe_swt_f: int = 160 + system_max_heat_output_swt_f: int = 150 + circulator_pump_max_gpm: float = 6 + system_max_heat_output_delta_temp_f: int + system_max_heat_output_gpm: float + max_store_temp_f: int = 210 + room_temp_f: int = 70 + ambient_power_in_kw: float = 1.14 + flo_total_time_hrs: int = 48 + export_graph: bool = False + storage_steps: int = 100 + edge_options: int = 2 + price_csv_starting_offset_hours: int = 0 + timezone_string: str = "US/Eastern" + zero_heat_delta_f: int = 3 + max_heatpump_source_water_temp_f: int = 140 + design_day_temp_f: int = -7 + store_size_gallons: int = 240 + rated_heatpump_electricity_kw: float = 5.5 + store_max_power_kw: float = 9 + cop_1_temp_f: int = 0 + cop_4_temp_f: int = 50 + store_passive_loss_ratio: float = 0.003 + house_no_energy_needed_temp_f: int = 65 + space_heat_thermostat_setpoint_f: int = 68 + annual_hvac_kwh_th: int = 25000 + annual_solar_gain_kwh_th: int = 5000 + ambient_temp_store_f: int = 65 + currency_unit: str = "USD" + temp_unit: str = "F" + home_city: str = "MILLINOCKET_ME" + standard_offer_price_dollars_per_mwh: float = 110 + flat_tariff_dollars_per_mwh: float = 70 + real_time_electricity_price_type_name: str = "csv.eprt.sync.1_0_0" + real_time_electricity_price_csv: str = "../gridworks-ps/input_data/electricity_prices/isone/eprt__w.isone.stetson__2020.csv" + dist_price_type_name: str = "csv.distp.sync.1_0_0" + dist_price_csv: str = "input_data/electricity_prices/isone/distp__w.isone.stetson__2020__gw.me.versant.a1.res.ets.csv" + weather_type_name: str = "csv.weather.forecast.sync.1_0_0" + weather_csv: str = ( + "input_data/weather/us/me/temp__ws.us.me.millinocketairport__2020.csv" + ) + scaled_heat_profile_csv: str = "input_data/misc/millinocket_heat_profile_2020.csv" + zero_potential_energy_water_temp_f: int = 100 + emitter_pump_feedback_model_value: ShDistPumpFeedbackModel = ( + ShDistPumpFeedbackModel.ConstantGpm.value + ) + mixing_valve_feedback_model_value: ShMixingValveFeedbackModel = ( + ShMixingValveFeedbackModel.ConstantSwt.value + ) + cautious_mixing_valve_temp_delta_f: int = 5 + heatpump_tariff_value: str = "CmpHeatTariff" + heatpump_energy_supply_type_value: str = "StandardOffer" + boost_tariff_value: str = "CmpStorageHeatTariff" + boost_energy_supply_type_value: str = "RealtimeLocalLmp" + # When this is uncommented, timezone_string disappears + # @validator('timezone_string') + # def is_recognized_timezone(cls, v): + # assert pytz.timezone(v) + + class Config: + env_prefix = "TEA_" + env_nested_delimiter = "__" + use_enum_values = True diff --git a/src/gwatn/strategies/simple_resistive_hydronic/tea_output.py b/src/gwatn/strategies/simple_resistive_hydronic/tea_output.py new file mode 100644 index 0000000..401c0c3 --- /dev/null +++ b/src/gwatn/strategies/simple_resistive_hydronic/tea_output.py @@ -0,0 +1,561 @@ +import time + +import pendulum +import satn.dev_utils.price.distp_sync_100_handler as distp_sync_100_handler +import satn.dev_utils.price.eprt_sync_100_handler as eprt_sync_100_handler +import satn.dev_utils.price.regp_sync_100_handler as regp_sync_100_handler +import satn.dev_utils.weather.weather_forecast_sync_100_handler as weather_forecast_sync_100_handler +import satn.strategies.heatpumpwithbooststore.flo_utils as flo_utils +import xlsxwriter +from satn.enums import ShDistPumpFeedbackModel +from satn.strategies.heatpumpwithbooststore.edge import ( + Edge__HeatpumpWithBoostStore as Edge, +) +from satn.strategies.heatpumpwithbooststore.flo import Flo__HeatpumpWithBoostStore +from satn.strategies.heatpumpwithbooststore.node import ( + Node__HeatpumpWithBoostStore as Node, +) +from satn.strategies.heatpumpwithbooststore.tea_config import TeaParams + +import gwatn.conversion_factors as cf +from gwatn.enums import RecognizedCurrencyUnit + + +OUTPUT_FOLDER = "src/satn/strategies/heatpumpwithbooststore/output_data" + +ON_PEAK_DIST_PRICE_PER_MWH_CUTOFF = 100 +SHOULDER_PEAK_DIST_PRICE_PER_MWH_CUTOFF = 50 + + +def export_xlsx( + tea_params: TeaParams, flo: Flo__HeatpumpWithBoostStore, export_graph: bool +): + file = OUTPUT_FOLDER + "/result_{}_{}_{}_{}.xlsx".format( + flo.home_city, + flo.flo_start_utc.year, + flo.graph_strategy_alias, + int(time.time()), + ) + file = file.lower() + print(file) + + workbook = xlsxwriter.Workbook(file) + start_idx = int(flo.params.StorageSteps / 2) + + # Add to gsr for more blank rows + gsr = 30 + w = export_best_path_info(flo=flo, workbook=workbook, start_idx=start_idx) + if export_graph: + export_flo_graph( + flo=flo, + workbook=workbook, + worksheet=w, + start_idx=start_idx, + graph_start_row=gsr, + ) + + export_params_xlsx(tea_params=tea_params, flo=flo, workbook=workbook) + workbook.close() + + +def export_best_path_info( + flo: Flo__HeatpumpWithBoostStore, + workbook: xlsxwriter.workbook.Workbook, + start_idx: int, +): + w = workbook.add_worksheet(f"start {100 * start_idx / flo.params.StorageSteps}%") + w.freeze_panes(0, 2) + title_format = workbook.add_format({"bold": True}) + title_format.set_font_size(14) + bold_format = workbook.add_format({"bold": True}) + gray_filler_format = workbook.add_format({"bg_color": "#edf0f2"}) + header_format = workbook.add_format({"bold": True, "text_wrap": True}) + mwh_format = workbook.add_format({"bold": True, "num_format": '0.00" MWh"'}) + + currency_format = workbook.add_format({"num_format": "$#,##0.00"}) + currency_bold_format = workbook.add_format( + {"bold": True, "num_format": "$#,##0.00"} + ) + if flo.params.CurrencyUnit == RecognizedCurrencyUnit.GBP: + currency_format = workbook.add_format( + {"num_format": "_-[$£-en-GB]* #,##0.00_-"} + ) + currency_bold_format = workbook.add_format( + {"bold": True, "num_format": "_-[$£-en-GB]* #,##0.00_-"} + ) + + w.set_column("A:A", 26) + w.set_column("B:B", 15) + OPT_PATH_STATE_VAR_ROW = 14 + + swt_list = flo_utils.get_source_water_temp_f_list(flo.params) + w.write(0, 0, "GNode Alias: millinocket.moss", title_format) + w.write(1, 0, flo.graph_strategy_alias) + w.write(2, 0, f"FLO start {flo.params.TimezoneString} ", header_format) + local_start = pendulum.timezone(flo.timezone_string).convert(flo.flo_start_utc) + w.write(2, 1, local_start.strftime("%Y/%m/%d %H:%M")) + + w.write(3, 0, "Total hours", header_format) + w.write(3, 1, sum(flo.slice_duration_hrs), bold_format) + + w.write(4, 0, "Rt Energy Price ($/MWh)", header_format) + w.write( + 4, + 1, + sum(flo.RealtimeElectricityPrice) / len(flo.RealtimeElectricityPrice), + currency_bold_format, + ) + # w.write(5, 0, "Flat rate for hp ($/MWh)", header_format) + # if flo.params.IsRegulating: + # w.write(5, 0, "Regulation Price ($/MWh)", header_format) + # w.write( + # 5, 1, sum(flo.reg_price_per_mwh) / len(flo.reg_price_per_mwh), currency_bold_format + # ) + + w.write(6, 0, "Dist Price ($/MWh)", header_format) + w.write( + 6, + 1, + sum(flo.DistributionPrice) / len(flo.DistributionPrice), + currency_bold_format, + ) + w.write(7, 0, "Outside Temp F", header_format) + w.write( + 7, + 1, + round(sum(flo.params.OutsideTempF) / len(flo.params.OutsideTempF), 2), + bold_format, + ) + w.write(8, 0, "COP", header_format) + avg_cop = sum(flo.cop.values()) / len(flo.cop) + w.write(8, 1, round(avg_cop, 2), bold_format) + w.write(9, 0, "House Power Required AvgKw", header_format) + w.write( + 9, + 1, + round(sum(flo.params.PowerRequiredByHouseFromSystemAvgKwList), 2), + bold_format, + ) + w.write(10, 0, "Required Source Water Temp F", header_format) + avg_swt = round((sum(swt_list) / len(swt_list)), 0) + w.write(10, 1, avg_swt, bold_format) + w.write(11, 0, "Max HeatPump kWh thermal", header_format) + avg_max_thermal_hp_kwh = round( + (sum(flo.max_thermal_hp_kwh.values()) / len(flo.max_thermal_hp_kwh)), 2 + ) + w.write(11, 1, avg_max_thermal_hp_kwh, bold_format) + + w.write(12, 0, "Outputs", header_format) + + for jj in range(flo.time_slices): + hours_since_start = sum(flo.slice_duration_hrs[0:jj]) + local_time = local_start.add(hours=hours_since_start) + w.write(2, jj + 2, local_time.strftime("%m/%d")) + w.write(3, jj + 2, local_time.strftime("%H:%M")) + w.write(4, jj + 2, flo.RealtimeElectricityPrice[jj], currency_format) + if flo.params.IsRegulating: + w.write(5, jj + 2, flo.reg_price_per_mwh[jj], currency_format) + else: + w.write(5, jj + 2, "", gray_filler_format) + dp = flo.DistributionPrice[jj] + LIGHT_GREEN_HEX = "#bbe3a6" + LIGHT_RED_HEX = "#ff6363" + if dp > ON_PEAK_DIST_PRICE_PER_MWH_CUTOFF: + if flo.params.CurrencyUnit == RecognizedCurrencyUnit.USD: + dist_format = workbook.add_format( + {"bg_color": LIGHT_RED_HEX, "num_format": "$#,##0.00"} + ) + elif flo.params.CurrencyUnit == RecognizedCurrencyUnit.GBP: + dist_format = workbook.add_format( + { + "bg_color": LIGHT_RED_HEX, + "num_format": "_-[$£-en-GB]* #,##0.00_-", + } + ) + elif dp > SHOULDER_PEAK_DIST_PRICE_PER_MWH_CUTOFF: + if flo.params.CurrencyUnit == RecognizedCurrencyUnit.USD: + dist_format = workbook.add_format( + {"bg_color": "yellow", "num_format": "$#,##0.00"} + ) + elif flo.params.CurrencyUnit == RecognizedCurrencyUnit.GBP: + dist_format = workbook.add_format( + {"bg_color": "yellow", "num_format": "_-[$£-en-GB]* #,##0.00_-"} + ) + else: + if flo.params.CurrencyUnit == RecognizedCurrencyUnit.USD: + dist_format = workbook.add_format( + {"bg_color": LIGHT_GREEN_HEX, "num_format": "$#,##0.00"} + ) + elif flo.params.CurrencyUnit == RecognizedCurrencyUnit.GBP: + dist_format = workbook.add_format( + { + "bg_color": LIGHT_GREEN_HEX, + "num_format": "_-[$£-en-GB]* #,##0.00_-", + } + ) + w.write(6, jj + 2, flo.DistributionPrice[jj], dist_format) + w.write(7, jj + 2, flo.params.OutsideTempF[jj]) + w.write(8, jj + 2, flo.cop[jj]) + w.write( + 9, jj + 2, round(flo.params.PowerRequiredByHouseFromSystemAvgKwList[jj], 2) + ) + w.write(10, jj + 2, round(swt_list[jj], 0)) + w.write(11, jj + 2, round(flo.max_thermal_hp_kwh[jj], 2)) + + w.write(12, jj + 2, "", gray_filler_format) + + node: Node = flo.node[0][start_idx] + w.write(OPT_PATH_STATE_VAR_ROW, 0, "Store Temp (F)", header_format) + w.write(OPT_PATH_STATE_VAR_ROW + 1, 0, "HeatPump kWh thermal", header_format) + w.write(OPT_PATH_STATE_VAR_ROW + 2, 0, "HeatPump kWh electric", header_format) + w.write(OPT_PATH_STATE_VAR_ROW + 3, 0, "Boost kWh electric", header_format) + w.write(OPT_PATH_STATE_VAR_ROW + 4, 0, "Energy cost (¢)", header_format) + w.write(OPT_PATH_STATE_VAR_ROW + 5, 0, "Hours Since Start", header_format) + + store_temp_f = [] + opt_heatpump_electricity_used_kwh = [] + opt_boost_electricity_used_kwh = [] + opt_energy_cost_dollars = [] + + best_idx = start_idx + dist_cost = [] + min_dist_price_per_mwh = min(flo.DistributionPrice) + + for jj in range(flo.time_slices): + edge: Edge = flo.best_edge[node] + store_temp_f.append(node.store_avg_water_temp_f) + hp_kwh = edge.hp_electricity_avg_kw + boost_kwh = edge.boost_electricity_used_avg_kw + opt_heatpump_electricity_used_kwh.append(hp_kwh) + opt_boost_electricity_used_kwh.append(boost_kwh) + opt_energy_cost_dollars.append(edge.cost) + hours_since_start = sum(flo.slice_duration_hrs[0:jj]) + + w.write(OPT_PATH_STATE_VAR_ROW, jj + 2, round(node.store_avg_water_temp_f, 2)) + w.write( + OPT_PATH_STATE_VAR_ROW + 1, + jj + 2, + round(edge.hp_thermal_energy_generated_avg_kw, 3), + ) + w.write( + OPT_PATH_STATE_VAR_ROW + 2, jj + 2, round(edge.hp_electricity_avg_kw, 3) + ) + w.write( + OPT_PATH_STATE_VAR_ROW + 3, + jj + 2, + round(edge.boost_electricity_used_avg_kw, 3), + ) + w.write(OPT_PATH_STATE_VAR_ROW + 4, jj + 2, round(edge.cost * 100, 2)) + w.write(OPT_PATH_STATE_VAR_ROW + 5, jj + 2, round(hours_since_start, 1)) + node = flo.node[jj + 1][edge.end_idx] + + w.write( + OPT_PATH_STATE_VAR_ROW, + 1, + round(sum(store_temp_f) / len(store_temp_f), 0), + bold_format, + ) + w.write( + OPT_PATH_STATE_VAR_ROW + 2, + 1, + sum(opt_heatpump_electricity_used_kwh) / 1000, + mwh_format, + ) + w.write( + OPT_PATH_STATE_VAR_ROW + 3, + 1, + sum(opt_boost_electricity_used_kwh) / 1000, + mwh_format, + ) + w.write( + OPT_PATH_STATE_VAR_ROW + 4, + 1, + sum(opt_energy_cost_dollars), + currency_bold_format, + ) + + w.write("H1", "Electricity cost of this path") + total_cost = sum(opt_energy_cost_dollars) + w.write("G1", total_cost, currency_bold_format) + w.write("H2", "Total electricity MWh") + total_electricity_mwh = ( + sum(opt_boost_electricity_used_kwh) + sum(opt_heatpump_electricity_used_kwh) + ) / 1000 + w.write("G2", total_electricity_mwh, mwh_format) + + total_btu = sum(flo.params.PowerRequiredByHouseFromSystemAvgKwList) * cf.BTU_PER_KWH + gallons_oil = total_btu / cf.BTU_PER_GALLON_OF_OIL / 0.85 + # assumes 85% efficient oil boiler + + w.write("L1", "Gallons of Oil") + w.write("K1", round(gallons_oil), bold_format) + + w.write("L2", "Equivalent Price of Oil") + w.write("K2", total_cost / gallons_oil, currency_bold_format) + # w.write("H3", "Flat rate comparison") + + return w + + +def export_pointer( + flo: Flo__HeatpumpWithBoostStore, + workbook: xlsxwriter.workbook.Workbook, + worksheet: xlsxwriter.workbook.Worksheet, + start_idx: int, + graph_start_row: int, +): + worksheet.freeze_panes(graph_start_row - 1, 2) + header_format = workbook.add_format({"bold": True, "text_wrap": True}) + percent_format = workbook.add_format({"num_format": '00.0"%"'}) + best_path_format = workbook.add_format({"bold": True, "bg_color": "#CDEBA6"}) + best_idx = start_idx + + worksheet.write(graph_start_row - 1, 0, "Percent full", header_format) + + for mm in range(flo.params.StorageSteps + 1): + kk = flo.params.StorageSteps - mm + percent = round(100 * kk / flo.params.StorageSteps, 1) + worksheet.write(graph_start_row + mm, 0, percent, percent_format) + for jj in range(flo.time_slices): + best_node = flo.node[jj][best_idx] + for mm in range(flo.params.StorageSteps + 1): + kk = flo.params.StorageSteps - mm + node = flo.node[jj][kk] + edges = flo.edges[node] + if kk == best_idx: + try: + worksheet.write( + graph_start_row + mm, + jj + 2, + -round(node.path_benefit, 4), + best_path_format, + ) + except: + print(f"failed for best {jj},{kk}") + else: + try: + worksheet.write( + graph_start_row + mm, + jj + 2, + round(node.path_cost, 4), + ) + except: + print(f"failed for {jj},{kk}") + best_idx = flo.best_edge[best_node].end_idx + + +def export_flo_graph( + flo: Flo__HeatpumpWithBoostStore, + workbook: xlsxwriter.workbook.Workbook, + worksheet: xlsxwriter.workbook.Worksheet, + start_idx: int, + graph_start_row: int, +): + worksheet.freeze_panes(graph_start_row - 1, 2) + header_format = workbook.add_format({"bold": True, "text_wrap": True}) + percent_format = workbook.add_format({"num_format": '00.0"%"'}) + best_path_format = workbook.add_format({"bold": True, "bg_color": "#CDEBA6"}) + best_idx = start_idx + + worksheet.write(graph_start_row - 1, 0, "Percent full", header_format) + + for mm in range(flo.params.StorageSteps + 1): + kk = flo.params.StorageSteps - mm + percent = round(100 * kk / flo.params.StorageSteps, 1) + worksheet.write(graph_start_row + mm, 0, percent, percent_format) + for jj in range(flo.time_slices): + best_node = flo.node[jj][best_idx] + for mm in range(flo.params.StorageSteps + 1): + kk = flo.params.StorageSteps - mm + node = flo.node[jj][kk] + if kk == best_idx: + try: + worksheet.write( + graph_start_row + mm, + jj + 2, + -round(node.path_benefit, 4), + best_path_format, + ) + except: + print(f"failed for best {jj},{kk}") + else: + try: + worksheet.write( + graph_start_row + mm, + jj + 2, + round(node.path_cost, 4), + ) + except: + print(f"failed for {jj},{kk}") + best_idx = flo.best_edge[best_node].end_idx + + +def export_params_xlsx( + tea_params: TeaParams, + flo: Flo__HeatpumpWithBoostStore, + workbook: xlsxwriter.workbook.Workbook, +): + bold = workbook.add_format({"bold": True}) + w = workbook.add_worksheet("Params") + derived_format_bold = workbook.add_format({"bold": True, "font_color": "green"}) + derived_format = workbook.add_format({"font_color": "green"}) + swt_list = flo_utils.get_source_water_temp_f_list(flo.params) + w.set_column("A:A", 31) + w.set_column("D:D", 31) + w.set_column("G:G", 31) + w.write("A1", "Key Parameters", bold) + + t = flo.params.OutsideTempF + w.write("A4", "This Run ColdestTempF ", derived_format_bold) + w.write("B4", min(t), derived_format) + + w.write("A5", "HouseWorstCaseTempF ", bold) + w.write("B5", flo.params.HouseWorstCaseTempF) + + w.write("A7", "SystemMaxHeatOutputKwAvg", bold) + w.write("B7", round(flo.params.SystemMaxHeatOutputKwAvg, 2)) + + p = flo.params.PowerRequiredByHouseFromSystemAvgKwList + w.write("A8", "This Run MaxHeatOutputKwAvg", derived_format_bold) + w.write("B8", round(max(p), 2), derived_format) + + house_wc_kw = flo_utils.get_house_worst_case_heat_output_avg_kw(flo.params) + w.write("A9", "HouseWorstCaseHeatOuputAvgKw", derived_format_bold) + w.write("B9", round(house_wc_kw, 1), derived_format) + + w.write("A10", "HouseWorstCaseHeatOuput BTU/hr", derived_format_bold) + w.write("B10", round(house_wc_kw * cf.BTU_PER_KWH), derived_format) + + w.write("A12", "EmitterMaxSafeSwtF", bold) + w.write("B12", flo.params.EmitterMaxSafeSwtF) + + w.write("A13", "This Run SystemMaxHeatOutputSwtF", bold) + w.write("B13", round(max(swt_list))) + + w.write("A14", "SystemMaxHeatOutputSWTF ", bold) + w.write("B14", tea_params.system_max_heat_output_swt_f) + + w.write("A15", "HeatPumpMaxWaterTempF ", bold) + w.write("B15", flo.params.MaxHeatpumpSourceWaterTempF) + + w.write("A16", "RatedHeatpumpElectricityKw", bold) + w.write("B16", flo.params.RatedHeatpumpElectricityKw) + + w.write("A17", "StoreMaxPowerKw", bold) + w.write("B17", flo.params.StoreMaxPowerKw) + + if flo.params.DistPumpFeedbackModel == ShDistPumpFeedbackModel.ConstantDeltaT: + w.write("A19", "SystemMaxHeatOutputDeltaTempF", bold) + w.write("B19", flo.params.SystemMaxHeatOutputDeltaTempF) + + w.write("A20", "SystemMaxHeatOutputGpm", derived_format_bold) + w.write("B20", round(flo.params.SystemMaxHeatOutputGpm, 2), derived_format) + else: + w.write("A19", "SystemMaxHeatOutputDeltaTempF", bold) + w.write("B19", flo.params.SystemMaxHeatOutputDeltaTempF) + + w.write("A20", "SystemMaxHeatOutputGpm", derived_format_bold) + w.write("B20", round(flo.params.SystemMaxHeatOutputGpm, 2), derived_format) + + w.write("A21", "Cop1TempF", bold) + w.write("B21", flo.params.Cop1TempF) + + w.write("A22", "Cop4TempF", bold) + w.write("B22", flo.params.Cop4TempF) + + w.write("A23", "StorePassiveLossRatio", bold) + w.write("B23", flo.params.StorePassiveLossRatio) + + w.write("A25", "StorageSteps", bold) + w.write("B25", flo.params.StorageSteps) + + ############# + + annual_kwh = tea_params.annual_hvac_kwh_th + annual_btu = round(cf.BTU_PER_KWH * annual_kwh) + w.write("D3", "Annual HVAC kWhTh", bold) + w.write("E3", annual_kwh) + + w.write("D4", "Annual HVAC MBTU", derived_format_bold) + w.write("E4", round(annual_btu / 10**6), derived_format) + + w.write("D6", "StoreSizeGallons", bold) + w.write("E6", flo.params.StoreSizeGallons) + + w.write("D7", "MaxStoreTempF", bold) + w.write("E7", flo.params.MaxStoreTempF) + + w.write("D8", "ZeroPotentialEnergyWaterTempF", bold) + w.write("E8", flo.params.ZeroPotentialEnergyWaterTempF) + + w.write("D9", "TotalStorageKwh", derived_format_bold) + w.write("E9", round(flo.max_energy_kwh_th, 1), derived_format) + + w.write("D10", "TotalStorage BTU", derived_format_bold) + w.write("E10", round(cf.BTU_PER_KWH * flo.max_energy_kwh_th), derived_format) + + w.write("D12", "SistPumpFeedbackModel", bold) + w.write("E12", flo.params.DistPumpFeedbackModel.value) + + w.write("D13", "MixingValveFeedbackModel", bold) + w.write("E13", flo.params.MixingValveFeedbackModel.value) + + w.write("D14", "IsRegulating", bold) + w.write("E14", flo.params.IsRegulating) + + w.write("D19", "RoomTempF", bold) + w.write("E19", flo.params.RoomTempF) + + w.write("D20", "AmbientPowerInKw", bold) + w.write("E20", flo.params.AmbientPowerInKw) + + ############### + + w.write("G3", "HeatpumpTariff", bold) + w.write("H3", flo.params.HeatpumpTariff.value) + + w.write("G4", "HeatpumpEnergySupplyType", bold) + w.write("H4", flo.params.HeatpumpEnergySupplyType.value) + + w.write("G5", "BoostTariff", bold) + w.write("H5", flo.params.BoostTariff.value) + + w.write("G6", "BoostEnergySupplyType", bold) + w.write("H6", flo.params.BoostEnergySupplyType.value) + + w.write("G7", "StandardOfferPriceDollarsPerMwh", bold) + w.write("H7", flo.params.StandardOfferPriceDollarsPerMwh) + + w.write("G8", "DistributionTariffDollarsPerMwh", bold) + w.write("H8", flo.params.DistributionTariffDollarsPerMwh) + + file = weather_forecast_sync_100_handler.csv_file_by_uid(flo.params.WeatherUid) + w.write("A28", "LocalWeatherFile", bold) + w.write("B28", file) + w.write("A29", "WeatherUid", bold) + w.write("B29", flo.params.WeatherUid) + + w.write("A30", "LocalRtPriceFile", bold) + w.write( + "B30", + eprt_sync_100_handler.csv_file_by_uid(price_uid=flo.params.RtElecPriceUid), + ) + + w.write("A31", "RtElecPriceUid", bold) + w.write("B31", flo.params.RtElecPriceUid) + + w.write("A32", "LocalDistPriceFile", bold) + w.write( + "B32", distp_sync_100_handler.csv_file_by_uid(price_uid=flo.params.DistPriceUid) + ) + w.write("A33", "DistPriceUid", bold) + w.write("B33", flo.params.DistPriceUid) + + w.write("A35", "Heat Profile", bold) + w.write("B35", tea_params.scaled_heat_profile_csv) + + if flo.params.IsRegulating: + w.write("A34", "LocalRegulationFile", bold) + w.write("B34", regp_sync_100_handler.csv_file_by_uid(flo.params.RegPriceUid)) + w.write("A35", "RegPriceUid", bold) + w.write("B35", flo.params.RegPriceUid) diff --git a/src/gwatn/two_channel_actor_base.py b/src/gwatn/two_channel_actor_base.py index eb9bdb6..9eeff32 100644 --- a/src/gwatn/two_channel_actor_base.py +++ b/src/gwatn/two_channel_actor_base.py @@ -1,6 +1,7 @@ import enum import functools import logging +import random import threading import time import uuid @@ -14,6 +15,7 @@ import gridworks.property_format as property_format import gridworks.utils as utils +import pendulum import pika # type: ignore from fastapi_utils.enums import StrEnum from gridworks.enums import GNodeRole @@ -30,6 +32,8 @@ from gwatn.scada_codec import ScadaCodec from gwatn.types import HeartbeatA from gwatn.types import HeartbeatA_Maker +from gwatn.types import SimTimestep +from gwatn.types import SimTimestep_Maker PayloadT = TypeVar("PayloadT") @@ -107,13 +111,14 @@ class DbgMsg(BaseModel): class TwoChannelActorBase(ABC): - SHUTDOWN_INTERVAL = 0.1 + SHUTDOWN_INTERVAL: float = 0.1 def __init__( self, settings: AtnSettings, api_type_maker_by_name: Dict[str, HeartbeatA_Maker] = api_types.TypeMakerByName, ): + self.settings: AtnSettings = settings self.scada_codec = ScadaCodec() self.dbg_messages: List[DbgMsg] = [] self.api_type_maker_by_name = api_type_maker_by_name @@ -124,6 +129,10 @@ def __init__( self.g_node_role: GNodeRole = GNodeRole(settings.g_node_role_value) self.rabbit_role: RabbitRole = RabbitRolebyRole[self.g_node_role] self.universe_type: UniverseType = UniverseType(settings.universe_type_value) + # Used for tracking time in simulated worlds + self._time: float = time.time() + if self.universe_type == UniverseType.Dev: + self._time = settings.initial_time_unix_s self.actor_main_stopped: bool = False adder = "-F" + str(uuid.uuid4()).split("-")[0][0:3] @@ -191,6 +200,9 @@ def local_stop(self) -> None: to stop the additional threads started in local_start""" pass + def __repr__(self) -> str: + return f"{self.alias}" + @no_type_check def on_message(self, _unused_channel, basic_deliver, properties, body) -> None: """Invoked by pika when a message is delivered from RabbitMQ. If a message @@ -297,11 +309,75 @@ def on_message(self, _unused_channel, basic_deliver, properties, body) -> None: payload=payload, ) - @abstractmethod def route_message( self, from_alias: str, from_role: GNodeRole, payload: HeartbeatA ) -> None: - raise NotImplementedError + if payload.TypeName == HeartbeatA_Maker.type_name: + if from_role != GNodeRole.Supervisor: + LOGGER.info( + f"Ignoring HeartbeatA from GNode {from_alias} with GNodeRole {from_role}; expects" + f"Supervisor as the GNodeRole" + ) + return + elif from_alias != self.settings.my_super_alias: + LOGGER.info( + f"Ignoring HeartbeatA from supervisor {from_alias}; " + f"my supervisor is {self.settings.my_super_alias}" + ) + return + + try: + self.heartbeat_from_super(from_alias, payload) + except: + LOGGER.exception("Error in heartbeat_received") + elif payload.TypeName == SimTimestep_Maker.type_name: + try: + self.timestep_from_timecoordinator(payload) + except: + LOGGER.exception("Error in timestep_from_timecoordinator") + + def heartbeat_from_super(self, from_alias: str, ping: HeartbeatA) -> None: + """ + Subordinate GNode responds to its supervisor's heartbeat with a "pong" message. + + Both the received heartbeat (ping) and the response (pong) have the type HeartbeatA + (see: https://gridworks.readthedocs.io/en/latest/apis/types.html#heartbeata). + + The subordinate GNode generates its own unique identifier (hex) and includes it + in the pong message along with the heartbeat it received from the supervisor. + + Please note that the subordinate GNode does not have the responsibility of verifying + the authenticity of the last heartbeat received from the supervisor - although typically, + the supervisor does send the last heartbeat from this GNode (except during the initial + heartbeat exchange). + + Args: + from_alias (str): the alias of the GNode that sent the ping. + ping (HeartbeatA): the heartbeat sent. + + Raises: + ValueError: If `from_alias` is not this GNode's Supervisor alias. + + """ + if from_alias != self.settings.my_super_alias: + raise ValueError( + f"from_alias {from_alias} does not match my supervisor" + f" {self.settings.my_super_alias}. This message should" + f"have been filtered out in the route_message method." + ) + pong = HeartbeatA_Maker( + my_hex=str(random.choice("0123456789abcdef")), your_last_hex=ping.MyHex + ).tuple + + self.send_message( + payload=pong, + to_role=GNodeRole.Supervisor, + to_g_node_alias=self.settings.my_super_alias, + ) + + LOGGER.debug( + f"[{self.alias}] Sent HB: SuHex {pong.YourLastHex}, AtnHex {pong.MyHex}" + ) def route_scada_message(self, from_alias: str, message: ScadaMessage) -> None: """This router should be overwritten by derived class""" @@ -1169,13 +1245,37 @@ def from_role_from_routing_key(self, routing_key: str) -> GNodeRole: return RoleByRabbitRole[rabbit_role] - ################################################# - # On receiving messages broadcast to all listners - ################################################# + ######################## + ## Time related (simulated time) + ######################## - ################################################# - # Various - ################################################# + def timestep_from_timecoordinator(self, payload: SimTimestep) -> None: + if self._time < payload.TimeUnixS: + self._time = payload.TimeUnixS + self.new_timestep(payload) + LOGGER.debug(f"Time is now {self.time_str()}") + elif self._time == payload.TimeUnixS: + self.repeat_timestep(payload) - def __repr__(self) -> str: - return f"{self.alias}" + def new_timestep(self, payload: SimTimestep) -> None: + LOGGER.info("New timestep") + + def repeat_timestep(self, payload: SimTimestep) -> None: + LOGGER.info("Timestep received again in atn_actor_base") + + def time(self) -> float: + if self.universe_type == UniverseType.Dev: + return self._time + else: + return time.time() + + def time_str(self) -> str: + return pendulum.from_timestamp(self.time()).strftime("%m/%d/%Y, %H:%M") + + ############################### + # Other GNode-related methods + ############################### + + @property + def short_alias(self) -> str: + return self.alias.split(".")[-1] diff --git a/src/gwatn/types/__init__.py b/src/gwatn/types/__init__.py index 49d8e7a..6c4be83 100644 --- a/src/gwatn/types/__init__.py +++ b/src/gwatn/types/__init__.py @@ -101,6 +101,12 @@ from gwatn.types.atn_params_brickstorageheater import AtnParamsBrickstorageheater_Maker from gwatn.types.atn_params_report import AtnParamsReport from gwatn.types.atn_params_report import AtnParamsReport_Maker +from gwatn.types.atn_params_simpleresistivehydronic import ( + AtnParamsSimpleresistivehydronic, +) +from gwatn.types.atn_params_simpleresistivehydronic import ( + AtnParamsSimpleresistivehydronic_Maker, +) from gwatn.types.basegnode_scada_create import BasegnodeScadaCreate from gwatn.types.basegnode_scada_create import BasegnodeScadaCreate_Maker from gwatn.types.discoverycert_algo_create import DiscoverycertAlgoCreate @@ -113,6 +119,12 @@ from gwatn.types.flo_params_brickstorageheater import FloParamsBrickstorageheater_Maker from gwatn.types.flo_params_report import FloParamsReport from gwatn.types.flo_params_report import FloParamsReport_Maker +from gwatn.types.flo_params_simpleresistivehydronic import ( + FloParamsSimpleresistivehydronic, +) +from gwatn.types.flo_params_simpleresistivehydronic import ( + FloParamsSimpleresistivehydronic_Maker, +) from gwatn.types.initial_tadeed_algo_create import InitialTadeedAlgoCreate from gwatn.types.initial_tadeed_algo_create import InitialTadeedAlgoCreate_Maker from gwatn.types.initial_tadeed_algo_optin import InitialTadeedAlgoOptin @@ -174,6 +186,8 @@ "AtnParamsBrickstorageheater_Maker", "AtnParamsReport", "AtnParamsReport_Maker", + "AtnParamsSimpleresistivehydronic", + "AtnParamsSimpleresistivehydronic_Maker", "BaseGNodeGt", "BaseGNodeGt_Maker", "BasegnodeScadaCreate", @@ -202,6 +216,8 @@ "FloParamsBrickstorageheater_Maker", "FloParamsReport", "FloParamsReport_Maker", + "FloParamsSimpleresistivehydronic", + "FloParamsSimpleresistivehydronic_Maker", "GNodeGt", "GNodeGt_Maker", "GNodeInstanceGt", diff --git a/src/gwatn/types/atn_params_brickstorageheater.py b/src/gwatn/types/atn_params_brickstorageheater.py index 6b48c9c..cec5691 100644 --- a/src/gwatn/types/atn_params_brickstorageheater.py +++ b/src/gwatn/types/atn_params_brickstorageheater.py @@ -25,6 +25,7 @@ class DistributionTariff000SchemaEnum: "00000000", "2127aba6", "ea5c675a", + "54aec3a7", ] @classmethod @@ -36,8 +37,9 @@ def is_symbol(cls, candidate: str) -> bool: class DistributionTariff000(StrEnum): Unknown = auto() - VersantStorageHeatTariff = auto() + VersantA1StorageHeatTariff = auto() VersantATariff = auto() + VersantA20HeatTariff = auto() @classmethod def default(cls) -> "DistributionTariff000": @@ -69,14 +71,16 @@ def local_to_type(cls, distribution_tariff: DistributionTariff) -> str: type_to_versioned_enum_dict: Dict[str, DistributionTariff000] = { "00000000": DistributionTariff000.Unknown, - "2127aba6": DistributionTariff000.VersantStorageHeatTariff, + "2127aba6": DistributionTariff000.VersantA1StorageHeatTariff, "ea5c675a": DistributionTariff000.VersantATariff, + "54aec3a7": DistributionTariff000.VersantA20HeatTariff, } versioned_enum_to_type_dict: Dict[DistributionTariff000, str] = { DistributionTariff000.Unknown: "00000000", - DistributionTariff000.VersantStorageHeatTariff: "2127aba6", + DistributionTariff000.VersantA1StorageHeatTariff: "2127aba6", DistributionTariff000.VersantATariff: "ea5c675a", + DistributionTariff000.VersantA20HeatTariff: "54aec3a7", } @@ -339,7 +343,7 @@ class AtnParamsBrickstorageheater(BaseModel): ) Tariff: DistributionTariff = Field( title="Tariff", - default=DistributionTariff.VersantStorageHeatTariff, + default=DistributionTariff.VersantA1StorageHeatTariff, ) EnergyType: EnergySupplyType = Field( title="EnergyType", diff --git a/src/gwatn/types/atn_params_simpleresistivehydronic.py b/src/gwatn/types/atn_params_simpleresistivehydronic.py new file mode 100644 index 0000000..0358cee --- /dev/null +++ b/src/gwatn/types/atn_params_simpleresistivehydronic.py @@ -0,0 +1,538 @@ +"""Type atn.params.simpleresistivehydronic, version 000""" +import json +from enum import auto +from typing import Any +from typing import Dict +from typing import List +from typing import Literal + +from fastapi_utils.enums import StrEnum +from gridworks.errors import SchemaError +from gridworks.message import as_enum +from pydantic import BaseModel +from pydantic import Field +from pydantic import validator + +from gwatn.enums import DistributionTariff +from gwatn.enums import EnergySupplyType +from gwatn.enums import RecognizedCurrencyUnit + + +class DistributionTariff000SchemaEnum: + enum_name: str = "distribution.tariff.000" + symbols: List[str] = [ + "00000000", + "2127aba6", + "ea5c675a", + "54aec3a7", + ] + + @classmethod + def is_symbol(cls, candidate: str) -> bool: + if candidate in cls.symbols: + return True + return False + + +class DistributionTariff000(StrEnum): + Unknown = auto() + VersantA1StorageHeatTariff = auto() + VersantATariff = auto() + VersantA20HeatTariff = auto() + + @classmethod + def default(cls) -> "DistributionTariff000": + return cls.Unknown + + @classmethod + def values(cls) -> List[str]: + return [elt.value for elt in cls] + + +class DistributionTariffMap: + @classmethod + def type_to_local(cls, symbol: str) -> DistributionTariff: + if not DistributionTariff000SchemaEnum.is_symbol(symbol): + raise SchemaError(f"{symbol} must belong to DistributionTariff000 symbols") + versioned_enum = cls.type_to_versioned_enum_dict[symbol] + return as_enum(versioned_enum, DistributionTariff, DistributionTariff.default()) + + @classmethod + def local_to_type(cls, distribution_tariff: DistributionTariff) -> str: + if not isinstance(distribution_tariff, DistributionTariff): + raise SchemaError( + f"{distribution_tariff} must be of type {DistributionTariff}" + ) + versioned_enum = as_enum( + distribution_tariff, DistributionTariff000, DistributionTariff000.default() + ) + return cls.versioned_enum_to_type_dict[versioned_enum] + + type_to_versioned_enum_dict: Dict[str, DistributionTariff000] = { + "00000000": DistributionTariff000.Unknown, + "2127aba6": DistributionTariff000.VersantA1StorageHeatTariff, + "ea5c675a": DistributionTariff000.VersantATariff, + "54aec3a7": DistributionTariff000.VersantA20HeatTariff, + } + + versioned_enum_to_type_dict: Dict[DistributionTariff000, str] = { + DistributionTariff000.Unknown: "00000000", + DistributionTariff000.VersantA1StorageHeatTariff: "2127aba6", + DistributionTariff000.VersantATariff: "ea5c675a", + DistributionTariff000.VersantA20HeatTariff: "54aec3a7", + } + + +class RecognizedCurrencyUnit000SchemaEnum: + enum_name: str = "recognized.currency.unit.000" + symbols: List[str] = [ + "00000000", + "e57c5143", + "f7b38fc5", + ] + + @classmethod + def is_symbol(cls, candidate: str) -> bool: + if candidate in cls.symbols: + return True + return False + + +class RecognizedCurrencyUnit000(StrEnum): + UNKNOWN = auto() + USD = auto() + GBP = auto() + + @classmethod + def default(cls) -> "RecognizedCurrencyUnit000": + return cls.UNKNOWN + + @classmethod + def values(cls) -> List[str]: + return [elt.value for elt in cls] + + +class RecognizedCurrencyUnitMap: + @classmethod + def type_to_local(cls, symbol: str) -> RecognizedCurrencyUnit: + if not RecognizedCurrencyUnit000SchemaEnum.is_symbol(symbol): + raise SchemaError( + f"{symbol} must belong to RecognizedCurrencyUnit000 symbols" + ) + versioned_enum = cls.type_to_versioned_enum_dict[symbol] + return as_enum( + versioned_enum, RecognizedCurrencyUnit, RecognizedCurrencyUnit.default() + ) + + @classmethod + def local_to_type(cls, recognized_currency_unit: RecognizedCurrencyUnit) -> str: + if not isinstance(recognized_currency_unit, RecognizedCurrencyUnit): + raise SchemaError( + f"{recognized_currency_unit} must be of type {RecognizedCurrencyUnit}" + ) + versioned_enum = as_enum( + recognized_currency_unit, + RecognizedCurrencyUnit000, + RecognizedCurrencyUnit000.default(), + ) + return cls.versioned_enum_to_type_dict[versioned_enum] + + type_to_versioned_enum_dict: Dict[str, RecognizedCurrencyUnit000] = { + "00000000": RecognizedCurrencyUnit000.UNKNOWN, + "e57c5143": RecognizedCurrencyUnit000.USD, + "f7b38fc5": RecognizedCurrencyUnit000.GBP, + } + + versioned_enum_to_type_dict: Dict[RecognizedCurrencyUnit000, str] = { + RecognizedCurrencyUnit000.UNKNOWN: "00000000", + RecognizedCurrencyUnit000.USD: "e57c5143", + RecognizedCurrencyUnit000.GBP: "f7b38fc5", + } + + +class EnergySupplyType000SchemaEnum: + enum_name: str = "energy.supply.type.000" + symbols: List[str] = [ + "00000000", + "cb18f937", + "e9dc99a6", + ] + + @classmethod + def is_symbol(cls, candidate: str) -> bool: + if candidate in cls.symbols: + return True + return False + + +class EnergySupplyType000(StrEnum): + Unknown = auto() + StandardOffer = auto() + RealtimeLocalLmp = auto() + + @classmethod + def default(cls) -> "EnergySupplyType000": + return cls.Unknown + + @classmethod + def values(cls) -> List[str]: + return [elt.value for elt in cls] + + +class EnergySupplyTypeMap: + @classmethod + def type_to_local(cls, symbol: str) -> EnergySupplyType: + if not EnergySupplyType000SchemaEnum.is_symbol(symbol): + raise SchemaError(f"{symbol} must belong to EnergySupplyType000 symbols") + versioned_enum = cls.type_to_versioned_enum_dict[symbol] + return as_enum(versioned_enum, EnergySupplyType, EnergySupplyType.default()) + + @classmethod + def local_to_type(cls, energy_supply_type: EnergySupplyType) -> str: + if not isinstance(energy_supply_type, EnergySupplyType): + raise SchemaError( + f"{energy_supply_type} must be of type {EnergySupplyType}" + ) + versioned_enum = as_enum( + energy_supply_type, EnergySupplyType000, EnergySupplyType000.default() + ) + return cls.versioned_enum_to_type_dict[versioned_enum] + + type_to_versioned_enum_dict: Dict[str, EnergySupplyType000] = { + "00000000": EnergySupplyType000.Unknown, + "cb18f937": EnergySupplyType000.StandardOffer, + "e9dc99a6": EnergySupplyType000.RealtimeLocalLmp, + } + + versioned_enum_to_type_dict: Dict[EnergySupplyType000, str] = { + EnergySupplyType000.Unknown: "00000000", + EnergySupplyType000.StandardOffer: "cb18f937", + EnergySupplyType000.RealtimeLocalLmp: "e9dc99a6", + } + + +def check_is_left_right_dot(v: str) -> None: + """ + LeftRightDot format: Lowercase alphanumeric words separated by periods, + most significant word (on the left) starting with an alphabet character. + + Raises: + ValueError: if not LeftRightDot format + """ + from typing import List + + try: + x: List[str] = v.split(".") + except: + raise ValueError(f"Failed to seperate {v} into words with split'.'") + first_word = x[0] + first_char = first_word[0] + if not first_char.isalpha(): + raise ValueError(f"Most significant word of {v} must start with alphabet char.") + for word in x: + if not word.isalnum(): + raise ValueError(f"words of {v} split by by '.' must be alphanumeric.") + if not v.islower(): + raise ValueError(f"All characters of {v} must be lowercase.") + + +class AtnParamsSimpleresistivehydronic(BaseModel): + """ """ + + GNodeAlias: str = Field( + title="GNodeAlias", + ) + HomeCity: str = Field( + title="HomeCity", + ) + TimezoneString: str = Field( + title="TimezoneString", + ) + StorageSteps: int = Field( + title="StorageSteps", + default=100, + ) + FloSlices: int = Field( + title="FloSlices", + default=48, + ) + SliceDurationMinutes: int = Field( + title="SliceDurationMinutes", + default=60, + ) + CurrencyUnit: RecognizedCurrencyUnit = Field( + title="CurrencyUnit", + default=RecognizedCurrencyUnit.USD, + ) + Tariff: DistributionTariff = Field( + title="Tariff", + default=DistributionTariff.VersantA1StorageHeatTariff, + ) + EnergyType: EnergySupplyType = Field( + title="EnergyType", + default=EnergySupplyType.RealtimeLocalLmp, + ) + StandardOfferPriceDollarsPerMwh: int = Field( + title="StandardOfferPriceDollarsPerMwh", + default=110, + ) + DistributionTariffDollarsPerMwh: int = Field( + title="DistributionTariffDollarsPerMwh", + default=113, + ) + StoreSizeGallons: int = Field( + title="StoreSizeGallons", + default=240, + ) + MaxStoreTempF: int = Field( + title="MaxStoreTempF", + default=210, + ) + ElementMaxPowerKw: float = Field( + title="ElementMaxPowerKw", + default=9.5, + ) + RequiredSourceWaterTempF: int = Field( + title="RequiredSourceWaterTempF", + default=120, + ) + FixedPumpGpm: float = Field( + title="FixedPumpGpm", + default=5.5, + ) + ReturnWaterFixedDeltaT: int = Field( + title="ReturnWaterFixedDeltaT", + default=20, + ) + AnnualHvacKwhTh: int = Field( + title="AnnualHvacKwhTh", + default=25000, + ) + AmbientPowerInKw: float = Field( + title="AmbientPowerInKw", + default=1.2, + ) + HouseWorstCaseTempF: int = Field( + title="HouseWorstCaseTempF", + default=-7, + ) + RoomTempF: int = Field( + title="RoomTempF", + default=68, + ) + TypeName: Literal[ + "atn.params.simpleresistivehydronic" + ] = "atn.params.simpleresistivehydronic" + Version: str = "000" + + @validator("GNodeAlias") + def _check_g_node_alias(cls, v: str) -> str: + try: + check_is_left_right_dot(v) + except ValueError as e: + raise ValueError(f"GNodeAlias failed LeftRightDot format validation: {e}") + return v + + @validator("CurrencyUnit") + def _check_currency_unit(cls, v: RecognizedCurrencyUnit) -> RecognizedCurrencyUnit: + return as_enum(v, RecognizedCurrencyUnit, RecognizedCurrencyUnit.UNKNOWN) + + @validator("Tariff") + def _check_tariff(cls, v: DistributionTariff) -> DistributionTariff: + return as_enum(v, DistributionTariff, DistributionTariff.Unknown) + + @validator("EnergyType") + def _check_energy_type(cls, v: EnergySupplyType) -> EnergySupplyType: + return as_enum(v, EnergySupplyType, EnergySupplyType.Unknown) + + def as_dict(self) -> Dict[str, Any]: + d = self.dict() + del d["CurrencyUnit"] + CurrencyUnit = as_enum( + self.CurrencyUnit, RecognizedCurrencyUnit, RecognizedCurrencyUnit.default() + ) + d["CurrencyUnitGtEnumSymbol"] = RecognizedCurrencyUnitMap.local_to_type( + CurrencyUnit + ) + del d["Tariff"] + Tariff = as_enum(self.Tariff, DistributionTariff, DistributionTariff.default()) + d["TariffGtEnumSymbol"] = DistributionTariffMap.local_to_type(Tariff) + del d["EnergyType"] + EnergyType = as_enum( + self.EnergyType, EnergySupplyType, EnergySupplyType.default() + ) + d["EnergyTypeGtEnumSymbol"] = EnergySupplyTypeMap.local_to_type(EnergyType) + return d + + def as_type(self) -> str: + return json.dumps(self.as_dict()) + + def __hash__(self): + return hash((type(self),) + tuple(self.__dict__.values())) # noqa + + +class AtnParamsSimpleresistivehydronic_Maker: + type_name = "atn.params.simpleresistivehydronic" + version = "000" + + def __init__( + self, + g_node_alias: str, + home_city: str, + timezone_string: str, + storage_steps: int, + flo_slices: int, + slice_duration_minutes: int, + currency_unit: RecognizedCurrencyUnit, + tariff: DistributionTariff, + energy_type: EnergySupplyType, + standard_offer_price_dollars_per_mwh: int, + distribution_tariff_dollars_per_mwh: int, + store_size_gallons: int, + max_store_temp_f: int, + element_max_power_kw: float, + required_source_water_temp_f: int, + fixed_pump_gpm: float, + return_water_fixed_delta_t: int, + annual_hvac_kwh_th: int, + ambient_power_in_kw: float, + house_worst_case_temp_f: int, + room_temp_f: int, + ): + self.tuple = AtnParamsSimpleresistivehydronic( + GNodeAlias=g_node_alias, + HomeCity=home_city, + TimezoneString=timezone_string, + StorageSteps=storage_steps, + FloSlices=flo_slices, + SliceDurationMinutes=slice_duration_minutes, + CurrencyUnit=currency_unit, + Tariff=tariff, + EnergyType=energy_type, + StandardOfferPriceDollarsPerMwh=standard_offer_price_dollars_per_mwh, + DistributionTariffDollarsPerMwh=distribution_tariff_dollars_per_mwh, + StoreSizeGallons=store_size_gallons, + MaxStoreTempF=max_store_temp_f, + ElementMaxPowerKw=element_max_power_kw, + RequiredSourceWaterTempF=required_source_water_temp_f, + FixedPumpGpm=fixed_pump_gpm, + ReturnWaterFixedDeltaT=return_water_fixed_delta_t, + AnnualHvacKwhTh=annual_hvac_kwh_th, + AmbientPowerInKw=ambient_power_in_kw, + HouseWorstCaseTempF=house_worst_case_temp_f, + RoomTempF=room_temp_f, + # + ) + + @classmethod + def tuple_to_type(cls, tuple: AtnParamsSimpleresistivehydronic) -> str: + """ + Given a Python class object, returns the serialized JSON type object + """ + return tuple.as_type() + + @classmethod + def type_to_tuple(cls, t: str) -> AtnParamsSimpleresistivehydronic: + """ + Given a serialized JSON type object, returns the Python class object + """ + try: + d = json.loads(t) + except TypeError: + raise SchemaError("Type must be string or bytes!") + if not isinstance(d, dict): + raise SchemaError(f"Deserializing {t} must result in dict!") + return cls.dict_to_tuple(d) + + @classmethod + def dict_to_tuple(cls, d: dict[str, Any]) -> AtnParamsSimpleresistivehydronic: + d2 = dict(d) + if "GNodeAlias" not in d2.keys(): + raise SchemaError(f"dict {d2} missing GNodeAlias") + if "HomeCity" not in d2.keys(): + raise SchemaError(f"dict {d2} missing HomeCity") + if "TimezoneString" not in d2.keys(): + raise SchemaError(f"dict {d2} missing TimezoneString") + if "StorageSteps" not in d2.keys(): + raise SchemaError(f"dict {d2} missing StorageSteps") + if "FloSlices" not in d2.keys(): + raise SchemaError(f"dict {d2} missing FloSlices") + if "SliceDurationMinutes" not in d2.keys(): + raise SchemaError(f"dict {d2} missing SliceDurationMinutes") + if "CurrencyUnitGtEnumSymbol" not in d2.keys(): + raise SchemaError(f"dict {d2} missing CurrencyUnitGtEnumSymbol") + if ( + d2["CurrencyUnitGtEnumSymbol"] + in RecognizedCurrencyUnit000SchemaEnum.symbols + ): + d2["CurrencyUnit"] = RecognizedCurrencyUnitMap.type_to_local( + d2["CurrencyUnitGtEnumSymbol"] + ) + else: + d2["CurrencyUnit"] = RecognizedCurrencyUnit.default() + if "TariffGtEnumSymbol" not in d2.keys(): + raise SchemaError(f"dict {d2} missing TariffGtEnumSymbol") + if d2["TariffGtEnumSymbol"] in DistributionTariff000SchemaEnum.symbols: + d2["Tariff"] = DistributionTariffMap.type_to_local(d2["TariffGtEnumSymbol"]) + else: + d2["Tariff"] = DistributionTariff.default() + if "EnergyTypeGtEnumSymbol" not in d2.keys(): + raise SchemaError(f"dict {d2} missing EnergyTypeGtEnumSymbol") + if d2["EnergyTypeGtEnumSymbol"] in EnergySupplyType000SchemaEnum.symbols: + d2["EnergyType"] = EnergySupplyTypeMap.type_to_local( + d2["EnergyTypeGtEnumSymbol"] + ) + else: + d2["EnergyType"] = EnergySupplyType.default() + if "StandardOfferPriceDollarsPerMwh" not in d2.keys(): + raise SchemaError(f"dict {d2} missing StandardOfferPriceDollarsPerMwh") + if "DistributionTariffDollarsPerMwh" not in d2.keys(): + raise SchemaError(f"dict {d2} missing DistributionTariffDollarsPerMwh") + if "StoreSizeGallons" not in d2.keys(): + raise SchemaError(f"dict {d2} missing StoreSizeGallons") + if "MaxStoreTempF" not in d2.keys(): + raise SchemaError(f"dict {d2} missing MaxStoreTempF") + if "ElementMaxPowerKw" not in d2.keys(): + raise SchemaError(f"dict {d2} missing ElementMaxPowerKw") + if "RequiredSourceWaterTempF" not in d2.keys(): + raise SchemaError(f"dict {d2} missing RequiredSourceWaterTempF") + if "FixedPumpGpm" not in d2.keys(): + raise SchemaError(f"dict {d2} missing FixedPumpGpm") + if "ReturnWaterFixedDeltaT" not in d2.keys(): + raise SchemaError(f"dict {d2} missing ReturnWaterFixedDeltaT") + if "AnnualHvacKwhTh" not in d2.keys(): + raise SchemaError(f"dict {d2} missing AnnualHvacKwhTh") + if "AmbientPowerInKw" not in d2.keys(): + raise SchemaError(f"dict {d2} missing AmbientPowerInKw") + if "HouseWorstCaseTempF" not in d2.keys(): + raise SchemaError(f"dict {d2} missing HouseWorstCaseTempF") + if "RoomTempF" not in d2.keys(): + raise SchemaError(f"dict {d2} missing RoomTempF") + if "TypeName" not in d2.keys(): + raise SchemaError(f"dict {d2} missing TypeName") + + return AtnParamsSimpleresistivehydronic( + GNodeAlias=d2["GNodeAlias"], + HomeCity=d2["HomeCity"], + TimezoneString=d2["TimezoneString"], + StorageSteps=d2["StorageSteps"], + FloSlices=d2["FloSlices"], + SliceDurationMinutes=d2["SliceDurationMinutes"], + CurrencyUnit=d2["CurrencyUnit"], + Tariff=d2["Tariff"], + EnergyType=d2["EnergyType"], + StandardOfferPriceDollarsPerMwh=d2["StandardOfferPriceDollarsPerMwh"], + DistributionTariffDollarsPerMwh=d2["DistributionTariffDollarsPerMwh"], + StoreSizeGallons=d2["StoreSizeGallons"], + MaxStoreTempF=d2["MaxStoreTempF"], + ElementMaxPowerKw=d2["ElementMaxPowerKw"], + RequiredSourceWaterTempF=d2["RequiredSourceWaterTempF"], + FixedPumpGpm=d2["FixedPumpGpm"], + ReturnWaterFixedDeltaT=d2["ReturnWaterFixedDeltaT"], + AnnualHvacKwhTh=d2["AnnualHvacKwhTh"], + AmbientPowerInKw=d2["AmbientPowerInKw"], + HouseWorstCaseTempF=d2["HouseWorstCaseTempF"], + RoomTempF=d2["RoomTempF"], + TypeName=d2["TypeName"], + Version="000", + ) diff --git a/src/gwatn/types/flo_params_brickstorageheater.py b/src/gwatn/types/flo_params_brickstorageheater.py index 1139a5c..2980720 100644 --- a/src/gwatn/types/flo_params_brickstorageheater.py +++ b/src/gwatn/types/flo_params_brickstorageheater.py @@ -267,7 +267,7 @@ class FloParamsBrickstorageheater(BaseModel): title="PowerRequiredByHouseFromSystemAvgKwList", default=[3.42], ) - C: Optional[float] = Field( + C: float = Field( title="C", default=200, ) @@ -293,9 +293,8 @@ class FloParamsBrickstorageheater(BaseModel): WeatherUid: str = Field( title="WeatherUid", ) - DistPriceUid: Optional[str] = Field( + DistPriceUid: str = Field( title="DistPriceUid", - default=None, ) RegPriceUid: Optional[str] = Field( title="RegPriceUid", @@ -373,9 +372,7 @@ def _check_weather_uid(cls, v: str) -> str: return v @validator("DistPriceUid") - def _check_dist_price_uid(cls, v: Optional[str]) -> Optional[str]: - if v is None: - return v + def _check_dist_price_uid(cls, v: str) -> str: try: check_is_uuid_canonical_textual(v) except ValueError as e: @@ -430,10 +427,6 @@ def as_dict(self) -> Dict[str, Any]: RecognizedTemperatureUnit.default(), ) d["TempUnitGtEnumSymbol"] = RecognizedTemperatureUnitMap.local_to_type(TempUnit) - if d["C"] is None: - del d["C"] - if d["DistPriceUid"] is None: - del d["DistPriceUid"] if d["RegPriceUid"] is None: del d["RegPriceUid"] return d @@ -464,14 +457,14 @@ def __init__( storage_steps: int, slice_duration_minutes: List[int], power_required_by_house_from_system_avg_kw_list: List[float], - c: Optional[float], + c: float, realtime_electricity_price: List[float], outside_temp_f: List[float], distribution_price: List[float], rt_elec_price_uid: str, regulation_price: List[float], weather_uid: str, - dist_price_uid: Optional[str], + dist_price_uid: str, reg_price_uid: Optional[str], start_year_utc: int, start_month_utc: int, @@ -587,7 +580,7 @@ def dict_to_tuple(cls, d: dict[str, Any]) -> FloParamsBrickstorageheater: f"dict {d2} missing PowerRequiredByHouseFromSystemAvgKwList" ) if "C" not in d2.keys(): - d2["C"] = None + raise SchemaError(f"dict {d2} missing C") if "RealtimeElectricityPrice" not in d2.keys(): raise SchemaError(f"dict {d2} missing RealtimeElectricityPrice") if "OutsideTempF" not in d2.keys(): @@ -601,7 +594,7 @@ def dict_to_tuple(cls, d: dict[str, Any]) -> FloParamsBrickstorageheater: if "WeatherUid" not in d2.keys(): raise SchemaError(f"dict {d2} missing WeatherUid") if "DistPriceUid" not in d2.keys(): - d2["DistPriceUid"] = None + raise SchemaError(f"dict {d2} missing DistPriceUid") if "RegPriceUid" not in d2.keys(): d2["RegPriceUid"] = None if "StartYearUtc" not in d2.keys(): diff --git a/src/gwatn/types/flo_params_simpleresistivehydronic.py b/src/gwatn/types/flo_params_simpleresistivehydronic.py new file mode 100644 index 0000000..3a76893 --- /dev/null +++ b/src/gwatn/types/flo_params_simpleresistivehydronic.py @@ -0,0 +1,480 @@ +"""Type flo.params.simpleresistivehydronic, version 000""" +import json +from enum import auto +from typing import Any +from typing import Dict +from typing import List +from typing import Literal + +from fastapi_utils.enums import StrEnum +from gridworks.errors import SchemaError +from gridworks.message import as_enum +from pydantic import BaseModel +from pydantic import Field +from pydantic import validator + +from gwatn.enums import RecognizedCurrencyUnit + + +class RecognizedCurrencyUnit000SchemaEnum: + enum_name: str = "recognized.currency.unit.000" + symbols: List[str] = [ + "00000000", + "e57c5143", + "f7b38fc5", + ] + + @classmethod + def is_symbol(cls, candidate: str) -> bool: + if candidate in cls.symbols: + return True + return False + + +class RecognizedCurrencyUnit000(StrEnum): + UNKNOWN = auto() + USD = auto() + GBP = auto() + + @classmethod + def default(cls) -> "RecognizedCurrencyUnit000": + return cls.UNKNOWN + + @classmethod + def values(cls) -> List[str]: + return [elt.value for elt in cls] + + +class RecognizedCurrencyUnitMap: + @classmethod + def type_to_local(cls, symbol: str) -> RecognizedCurrencyUnit: + if not RecognizedCurrencyUnit000SchemaEnum.is_symbol(symbol): + raise SchemaError( + f"{symbol} must belong to RecognizedCurrencyUnit000 symbols" + ) + versioned_enum = cls.type_to_versioned_enum_dict[symbol] + return as_enum( + versioned_enum, RecognizedCurrencyUnit, RecognizedCurrencyUnit.default() + ) + + @classmethod + def local_to_type(cls, recognized_currency_unit: RecognizedCurrencyUnit) -> str: + if not isinstance(recognized_currency_unit, RecognizedCurrencyUnit): + raise SchemaError( + f"{recognized_currency_unit} must be of type {RecognizedCurrencyUnit}" + ) + versioned_enum = as_enum( + recognized_currency_unit, + RecognizedCurrencyUnit000, + RecognizedCurrencyUnit000.default(), + ) + return cls.versioned_enum_to_type_dict[versioned_enum] + + type_to_versioned_enum_dict: Dict[str, RecognizedCurrencyUnit000] = { + "00000000": RecognizedCurrencyUnit000.UNKNOWN, + "e57c5143": RecognizedCurrencyUnit000.USD, + "f7b38fc5": RecognizedCurrencyUnit000.GBP, + } + + versioned_enum_to_type_dict: Dict[RecognizedCurrencyUnit000, str] = { + RecognizedCurrencyUnit000.UNKNOWN: "00000000", + RecognizedCurrencyUnit000.USD: "e57c5143", + RecognizedCurrencyUnit000.GBP: "f7b38fc5", + } + + +def check_is_uuid_canonical_textual(v: str) -> None: + """ + UuidCanonicalTextual format: A string of hex words separated by hyphens + of length 8-4-4-4-12. + + Raises: + ValueError: if not UuidCanonicalTextual format + """ + try: + x = v.split("-") + except AttributeError as e: + raise ValueError(f"Failed to split on -: {e}") + if len(x) != 5: + raise ValueError(f"{v} split by '-' did not have 5 words") + for hex_word in x: + try: + int(hex_word, 16) + except ValueError: + raise ValueError(f"Words of {v} are not all hex") + if len(x[0]) != 8: + raise ValueError(f"{v} word lengths not 8-4-4-4-12") + if len(x[1]) != 4: + raise ValueError(f"{v} word lengths not 8-4-4-4-12") + if len(x[2]) != 4: + raise ValueError(f"{v} word lengths not 8-4-4-4-12") + if len(x[3]) != 4: + raise ValueError(f"{v} word lengths not 8-4-4-4-12") + if len(x[4]) != 12: + raise ValueError(f"{v} word lengths not 8-4-4-4-12") + + +def check_is_left_right_dot(v: str) -> None: + """ + LeftRightDot format: Lowercase alphanumeric words separated by periods, + most significant word (on the left) starting with an alphabet character. + + Raises: + ValueError: if not LeftRightDot format + """ + from typing import List + + try: + x: List[str] = v.split(".") + except: + raise ValueError(f"Failed to seperate {v} into words with split'.'") + first_word = x[0] + first_char = first_word[0] + if not first_char.isalpha(): + raise ValueError(f"Most significant word of {v} must start with alphabet char.") + for word in x: + if not word.isalnum(): + raise ValueError(f"words of {v} split by by '.' must be alphanumeric.") + if not v.islower(): + raise ValueError(f"All characters of {v} must be lowercase.") + + +class FloParamsSimpleresistivehydronic(BaseModel): + """ """ + + GNodeAlias: str = Field( + title="GNodeAlias", + ) + FloParamsUid: str = Field( + title="FloParamsUid", + ) + HomeCity: str = Field( + title="HomeCity", + default="MILLINOCKET_ME", + ) + TimezoneString: str = Field( + title="TimezoneString", + default="US/Eastern", + ) + StartYearUtc: int = Field( + title="StartYearUtc", + default=2020, + ) + StartMonthUtc: int = Field( + title="StartMonthUtc", + default=1, + ) + StartDayUtc: int = Field( + title="StartDayUtc", + default=1, + ) + StartHourUtc: int = Field( + title="StartHourUtc", + default=0, + ) + StartMinuteUtc: int = Field( + title="StartMinuteUtc", + default=0, + ) + StoreSizeGallons: int = Field( + title="StoreSizeGallons", + default=240, + ) + MaxStoreTempF: int = Field( + title="MaxStoreTempF", + default=190, + ) + ElementMaxPowerKw: float = Field( + title="ElementMaxPowerKw", + default=9.5, + ) + RequiredSourceWaterTempF: int = Field( + title="RequiredSourceWaterTempF", + default=120, + ) + FixedPumpGpm: float = Field( + title="FixedPumpGpm", + default=4.5, + ) + ReturnWaterFixedDeltaT: int = Field( + title="ReturnWaterFixedDeltaT", + default=20, + ) + SliceDurationMinutes: List[int] = Field( + title="SliceDurationMinutes", + default=[60], + ) + PowerLostFromHouseKwList: List[float] = Field( + title="PowerLostFromHouseKwList", + default=[3.42], + ) + OutsideTempF: List[float] = Field( + title="OutsideTempF", + default=[-5.1], + ) + DistributionPrice: List[float] = Field( + title="DistributionPrice", + default=[40.0], + ) + RealtimeElectricityPrice: List[float] = Field( + title="RealtimeElectricityPrice", + default=[10.35], + ) + RtElecPriceUid: str = Field( + title="RtElecPriceUid", + ) + WeatherUid: str = Field( + title="WeatherUid", + ) + DistPriceUid: str = Field( + title="DistPriceUid", + ) + CurrencyUnit: RecognizedCurrencyUnit = Field( + title="CurrencyUnit", + default=RecognizedCurrencyUnit.USD, + ) + TypeName: Literal[ + "flo.params.simpleresistivehydronic" + ] = "flo.params.simpleresistivehydronic" + Version: str = "000" + + @validator("GNodeAlias") + def _check_g_node_alias(cls, v: str) -> str: + try: + check_is_left_right_dot(v) + except ValueError as e: + raise ValueError(f"GNodeAlias failed LeftRightDot format validation: {e}") + return v + + @validator("FloParamsUid") + def _check_flo_params_uid(cls, v: str) -> str: + try: + check_is_uuid_canonical_textual(v) + except ValueError as e: + raise ValueError( + f"FloParamsUid failed UuidCanonicalTextual format validation: {e}" + ) + return v + + @validator("RtElecPriceUid") + def _check_rt_elec_price_uid(cls, v: str) -> str: + try: + check_is_uuid_canonical_textual(v) + except ValueError as e: + raise ValueError( + f"RtElecPriceUid failed UuidCanonicalTextual format validation: {e}" + ) + return v + + @validator("WeatherUid") + def _check_weather_uid(cls, v: str) -> str: + try: + check_is_uuid_canonical_textual(v) + except ValueError as e: + raise ValueError( + f"WeatherUid failed UuidCanonicalTextual format validation: {e}" + ) + return v + + @validator("DistPriceUid") + def _check_dist_price_uid(cls, v: str) -> str: + try: + check_is_uuid_canonical_textual(v) + except ValueError as e: + raise ValueError( + f"DistPriceUid failed UuidCanonicalTextual format validation: {e}" + ) + return v + + @validator("CurrencyUnit") + def _check_currency_unit(cls, v: RecognizedCurrencyUnit) -> RecognizedCurrencyUnit: + return as_enum(v, RecognizedCurrencyUnit, RecognizedCurrencyUnit.UNKNOWN) + + def as_dict(self) -> Dict[str, Any]: + d = self.dict() + del d["CurrencyUnit"] + CurrencyUnit = as_enum( + self.CurrencyUnit, RecognizedCurrencyUnit, RecognizedCurrencyUnit.default() + ) + d["CurrencyUnitGtEnumSymbol"] = RecognizedCurrencyUnitMap.local_to_type( + CurrencyUnit + ) + return d + + def as_type(self) -> str: + return json.dumps(self.as_dict()) + + def __hash__(self): + return hash((type(self),) + tuple(self.__dict__.values())) # noqa + + +class FloParamsSimpleresistivehydronic_Maker: + type_name = "flo.params.simpleresistivehydronic" + version = "000" + + def __init__( + self, + g_node_alias: str, + flo_params_uid: str, + home_city: str, + timezone_string: str, + start_year_utc: int, + start_month_utc: int, + start_day_utc: int, + start_hour_utc: int, + start_minute_utc: int, + store_size_gallons: int, + max_store_temp_f: int, + element_max_power_kw: float, + required_source_water_temp_f: int, + fixed_pump_gpm: float, + return_water_fixed_delta_t: int, + slice_duration_minutes: List[int], + power_lost_from_house_kw_list: List[float], + outside_temp_f: List[float], + distribution_price: List[float], + realtime_electricity_price: List[float], + rt_elec_price_uid: str, + weather_uid: str, + dist_price_uid: str, + currency_unit: RecognizedCurrencyUnit, + ): + self.tuple = FloParamsSimpleresistivehydronic( + GNodeAlias=g_node_alias, + FloParamsUid=flo_params_uid, + HomeCity=home_city, + TimezoneString=timezone_string, + StartYearUtc=start_year_utc, + StartMonthUtc=start_month_utc, + StartDayUtc=start_day_utc, + StartHourUtc=start_hour_utc, + StartMinuteUtc=start_minute_utc, + StoreSizeGallons=store_size_gallons, + MaxStoreTempF=max_store_temp_f, + ElementMaxPowerKw=element_max_power_kw, + RequiredSourceWaterTempF=required_source_water_temp_f, + FixedPumpGpm=fixed_pump_gpm, + ReturnWaterFixedDeltaT=return_water_fixed_delta_t, + SliceDurationMinutes=slice_duration_minutes, + PowerLostFromHouseKwList=power_lost_from_house_kw_list, + OutsideTempF=outside_temp_f, + DistributionPrice=distribution_price, + RealtimeElectricityPrice=realtime_electricity_price, + RtElecPriceUid=rt_elec_price_uid, + WeatherUid=weather_uid, + DistPriceUid=dist_price_uid, + CurrencyUnit=currency_unit, + # + ) + + @classmethod + def tuple_to_type(cls, tuple: FloParamsSimpleresistivehydronic) -> str: + """ + Given a Python class object, returns the serialized JSON type object + """ + return tuple.as_type() + + @classmethod + def type_to_tuple(cls, t: str) -> FloParamsSimpleresistivehydronic: + """ + Given a serialized JSON type object, returns the Python class object + """ + try: + d = json.loads(t) + except TypeError: + raise SchemaError("Type must be string or bytes!") + if not isinstance(d, dict): + raise SchemaError(f"Deserializing {t} must result in dict!") + return cls.dict_to_tuple(d) + + @classmethod + def dict_to_tuple(cls, d: dict[str, Any]) -> FloParamsSimpleresistivehydronic: + d2 = dict(d) + if "GNodeAlias" not in d2.keys(): + raise SchemaError(f"dict {d2} missing GNodeAlias") + if "FloParamsUid" not in d2.keys(): + raise SchemaError(f"dict {d2} missing FloParamsUid") + if "HomeCity" not in d2.keys(): + raise SchemaError(f"dict {d2} missing HomeCity") + if "TimezoneString" not in d2.keys(): + raise SchemaError(f"dict {d2} missing TimezoneString") + if "StartYearUtc" not in d2.keys(): + raise SchemaError(f"dict {d2} missing StartYearUtc") + if "StartMonthUtc" not in d2.keys(): + raise SchemaError(f"dict {d2} missing StartMonthUtc") + if "StartDayUtc" not in d2.keys(): + raise SchemaError(f"dict {d2} missing StartDayUtc") + if "StartHourUtc" not in d2.keys(): + raise SchemaError(f"dict {d2} missing StartHourUtc") + if "StartMinuteUtc" not in d2.keys(): + raise SchemaError(f"dict {d2} missing StartMinuteUtc") + if "StoreSizeGallons" not in d2.keys(): + raise SchemaError(f"dict {d2} missing StoreSizeGallons") + if "MaxStoreTempF" not in d2.keys(): + raise SchemaError(f"dict {d2} missing MaxStoreTempF") + if "ElementMaxPowerKw" not in d2.keys(): + raise SchemaError(f"dict {d2} missing ElementMaxPowerKw") + if "RequiredSourceWaterTempF" not in d2.keys(): + raise SchemaError(f"dict {d2} missing RequiredSourceWaterTempF") + if "FixedPumpGpm" not in d2.keys(): + raise SchemaError(f"dict {d2} missing FixedPumpGpm") + if "ReturnWaterFixedDeltaT" not in d2.keys(): + raise SchemaError(f"dict {d2} missing ReturnWaterFixedDeltaT") + if "SliceDurationMinutes" not in d2.keys(): + raise SchemaError(f"dict {d2} missing SliceDurationMinutes") + if "PowerLostFromHouseKwList" not in d2.keys(): + raise SchemaError(f"dict {d2} missing PowerLostFromHouseKwList") + if "OutsideTempF" not in d2.keys(): + raise SchemaError(f"dict {d2} missing OutsideTempF") + if "DistributionPrice" not in d2.keys(): + raise SchemaError(f"dict {d2} missing DistributionPrice") + if "RealtimeElectricityPrice" not in d2.keys(): + raise SchemaError(f"dict {d2} missing RealtimeElectricityPrice") + if "RtElecPriceUid" not in d2.keys(): + raise SchemaError(f"dict {d2} missing RtElecPriceUid") + if "WeatherUid" not in d2.keys(): + raise SchemaError(f"dict {d2} missing WeatherUid") + if "DistPriceUid" not in d2.keys(): + raise SchemaError(f"dict {d2} missing DistPriceUid") + if "CurrencyUnitGtEnumSymbol" not in d2.keys(): + raise SchemaError(f"dict {d2} missing CurrencyUnitGtEnumSymbol") + if ( + d2["CurrencyUnitGtEnumSymbol"] + in RecognizedCurrencyUnit000SchemaEnum.symbols + ): + d2["CurrencyUnit"] = RecognizedCurrencyUnitMap.type_to_local( + d2["CurrencyUnitGtEnumSymbol"] + ) + else: + d2["CurrencyUnit"] = RecognizedCurrencyUnit.default() + if "TypeName" not in d2.keys(): + raise SchemaError(f"dict {d2} missing TypeName") + + return FloParamsSimpleresistivehydronic( + GNodeAlias=d2["GNodeAlias"], + FloParamsUid=d2["FloParamsUid"], + HomeCity=d2["HomeCity"], + TimezoneString=d2["TimezoneString"], + StartYearUtc=d2["StartYearUtc"], + StartMonthUtc=d2["StartMonthUtc"], + StartDayUtc=d2["StartDayUtc"], + StartHourUtc=d2["StartHourUtc"], + StartMinuteUtc=d2["StartMinuteUtc"], + StoreSizeGallons=d2["StoreSizeGallons"], + MaxStoreTempF=d2["MaxStoreTempF"], + ElementMaxPowerKw=d2["ElementMaxPowerKw"], + RequiredSourceWaterTempF=d2["RequiredSourceWaterTempF"], + FixedPumpGpm=d2["FixedPumpGpm"], + ReturnWaterFixedDeltaT=d2["ReturnWaterFixedDeltaT"], + SliceDurationMinutes=d2["SliceDurationMinutes"], + PowerLostFromHouseKwList=d2["PowerLostFromHouseKwList"], + OutsideTempF=d2["OutsideTempF"], + DistributionPrice=d2["DistributionPrice"], + RealtimeElectricityPrice=d2["RealtimeElectricityPrice"], + RtElecPriceUid=d2["RtElecPriceUid"], + WeatherUid=d2["WeatherUid"], + DistPriceUid=d2["DistPriceUid"], + CurrencyUnit=d2["CurrencyUnit"], + TypeName=d2["TypeName"], + Version="000", + ) diff --git a/src/gwatn/types/simplesim_driver_report.py b/src/gwatn/types/simplesim_driver_report.py index b17ced7..e40709a 100644 --- a/src/gwatn/types/simplesim_driver_report.py +++ b/src/gwatn/types/simplesim_driver_report.py @@ -75,8 +75,8 @@ class SimplesimDriverReport(BaseModel): FromGNodeAlias: str = Field( title="FromGNodeAlias", ) - FromGNodeInstanecId: str = Field( - title="FromGNodeInstanecId", + FromGNodeInstanceId: str = Field( + title="FromGNodeInstanceId", ) DriverDataTypeName: str = Field( title="DriverDataTypeName", @@ -97,13 +97,13 @@ def _check_from_g_node_alias(cls, v: str) -> str: ) return v - @validator("FromGNodeInstanecId") - def _check_from_g_node_instanec_id(cls, v: str) -> str: + @validator("FromGNodeInstanceId") + def _check_from_g_node_instance_id(cls, v: str) -> str: try: check_is_uuid_canonical_textual(v) except ValueError as e: raise ValueError( - f"FromGNodeInstanecId failed UuidCanonicalTextual format validation: {e}" + f"FromGNodeInstanceId failed UuidCanonicalTextual format validation: {e}" ) return v @@ -136,13 +136,13 @@ class SimplesimDriverReport_Maker: def __init__( self, from_g_node_alias: str, - from_g_node_instanec_id: str, + from_g_node_instance_id: str, driver_data_type_name: str, driver_data: SimplesimDriverData, ): self.tuple = SimplesimDriverReport( FromGNodeAlias=from_g_node_alias, - FromGNodeInstanecId=from_g_node_instanec_id, + FromGNodeInstanceId=from_g_node_instance_id, DriverDataTypeName=driver_data_type_name, DriverData=driver_data, # @@ -173,8 +173,8 @@ def dict_to_tuple(cls, d: dict[str, Any]) -> SimplesimDriverReport: d2 = dict(d) if "FromGNodeAlias" not in d2.keys(): raise SchemaError(f"dict {d2} missing FromGNodeAlias") - if "FromGNodeInstanecId" not in d2.keys(): - raise SchemaError(f"dict {d2} missing FromGNodeInstanecId") + if "FromGNodeInstanceId" not in d2.keys(): + raise SchemaError(f"dict {d2} missing FromGNodeInstanceId") if "DriverDataTypeName" not in d2.keys(): raise SchemaError(f"dict {d2} missing DriverDataTypeName") if "DriverData" not in d2.keys(): @@ -190,7 +190,7 @@ def dict_to_tuple(cls, d: dict[str, Any]) -> SimplesimDriverReport: return SimplesimDriverReport( FromGNodeAlias=d2["FromGNodeAlias"], - FromGNodeInstanecId=d2["FromGNodeInstanecId"], + FromGNodeInstanceId=d2["FromGNodeInstanceId"], DriverDataTypeName=d2["DriverDataTypeName"], DriverData=d2["DriverData"], TypeName=d2["TypeName"], diff --git a/src/gwatn_test/__init__.py b/src/gwatn_test/__init__.py new file mode 100644 index 0000000..dac573a --- /dev/null +++ b/src/gwatn_test/__init__.py @@ -0,0 +1,6 @@ +from gwatn_test.simple_scada_sim_actor_base import SimpleScadaSimActorBase + + +__all__ = [ + "SimpleScadaSimActorBase", +] diff --git a/src/gwatn/simple_scada_sim_actor_base.py b/src/gwatn_test/simple_scada_sim_actor_base.py similarity index 99% rename from src/gwatn/simple_scada_sim_actor_base.py rename to src/gwatn_test/simple_scada_sim_actor_base.py index 8103c97..f672d97 100644 --- a/src/gwatn/simple_scada_sim_actor_base.py +++ b/src/gwatn_test/simple_scada_sim_actor_base.py @@ -41,9 +41,9 @@ from gwatn.types import ScadaCertTransfer_Maker from gwatn.types import SimplesimDriverReport from gwatn.types import SimplesimDriverReport_Maker +from gwatn.types import SimplesimSnapshotBrickstorageheater as Snapshot from gwatn.types import SimTimestep from gwatn.types import SimTimestep_Maker -from gwatn.types import SnapshotBrickstorageheater as Snapshot DISPATCH_CONTRACT_REPORTING_ALGOS = 5 diff --git a/src/gwatn/brick_storage_heater/simple_scada_sim.py b/tests/brick_storage_heater/simple_scada_sim.py similarity index 97% rename from src/gwatn/brick_storage_heater/simple_scada_sim.py rename to tests/brick_storage_heater/simple_scada_sim.py index 7efeea5..46de493 100644 --- a/src/gwatn/brick_storage_heater/simple_scada_sim.py +++ b/tests/brick_storage_heater/simple_scada_sim.py @@ -1,7 +1,6 @@ """ SCADA Actor """ -import functools + import logging -import time from typing import Optional from typing import cast @@ -12,7 +11,6 @@ import gwatn.api_types as api_types import gwatn.config as config -from gwatn.simple_scada_sim_actor_base import SimpleScadaSimActorBase from gwatn.types import AtnParamsBrickstorageheater as AtnParams from gwatn.types import AtnParamsBrickstorageheater_Maker from gwatn.types import GtDispatchBoolean @@ -21,6 +19,7 @@ from gwatn.types import SimplesimDriverReport from gwatn.types import SimplesimSnapshotBrickstorageheater as Snapshot from gwatn.types import SimTimestep +from gwatn_test import SimpleScadaSimActorBase DISPATCH_CONTRACT_REPORTING_ALGOS = 5 @@ -72,7 +71,7 @@ def simplesim_driver_report_received(self, payload: SimplesimDriverReport) -> No """This gets received right before the top of the hour, from our best simulation of the TerminalAsset (which is happening in the AtomicTNode).""" - if payload.FromGNodeInstanecId != self.atn_gni_id: + if payload.FromGNodeInstanceId != self.atn_gni_id: LOGGER.info(f"Igoring {payload} - incorrect GNodeInstanceId") if payload.DriverDataTypeName != SimplesimDriverDataBsh_Maker.type_name: diff --git a/tests/enums/distribution_tariff_test.py b/tests/enums/distribution_tariff_test.py index dbcc1d5..7aa417d 100644 --- a/tests/enums/distribution_tariff_test.py +++ b/tests/enums/distribution_tariff_test.py @@ -5,8 +5,9 @@ def test_distribution_tariff() -> None: assert set(DistributionTariff.values()) == { "Unknown", - "VersantStorageHeatTariff", + "VersantA1StorageHeatTariff", "VersantATariff", + "VersantA20HeatTariff", } assert DistributionTariff.default() == DistributionTariff.Unknown diff --git a/tests/test_simple_atn.py b/tests/test_simple_atn.py new file mode 100644 index 0000000..091601d --- /dev/null +++ b/tests/test_simple_atn.py @@ -0,0 +1,163 @@ +import time +import uuid + +import pendulum +import pika +from gridworks_test import TimeCoordinatorStubRecorder +from gridworks_test import load_rabbit_exchange_bindings +from gridworks_test import wait_for + +from gwatn import atn_utils +from gwatn.config import AtnSettings +from gwatn.enums import MessageCategory +from gwatn.enums import UniverseType +from gwatn.simple_atn_actor import SimpleAtnActor as Atn +from gwatn.types import LatestPrice_Maker +from gwatn.types import SimTimestep_Maker + + +# TODO: fix ci issue, no exchange timecoordinatormic_tx + + +# def test_atn(): +# atn = Atn(AtnSettings()) +# atn.start() +# wait_for(lambda: atn._consuming, 2, "actor is consuming") +# wait_for(lambda: atn._publish_channel, 2, "actor publish channel exists") +# wait_for(lambda: atn._publish_channel.is_open, 2, "actor publish channel exists") +# +# load_rabbit_exchange_bindings(atn._consume_channel) +# ################## +# # test receiving simulated timestep - add stub time coordinator +# ################### +# assert atn_utils.is_dummy_atn_params(atn.atn_params) +# assert atn.universe_type == UniverseType.Dev +# assert atn._time == atn.settings.initial_time_unix_s +# assert atn.time() == atn._time +# +# d = pendulum.datetime(year=2020, month=1, day=1, hour=5) +# t = d.int_timestamp +# payload = SimTimestep_Maker( +# from_g_node_alias="d1.time", +# from_g_node_instance_id=str(uuid.uuid4()), +# time_unix_s=t, +# timestep_created_ms=1000 * int(time.time()), +# message_id=str(uuid.uuid4()), +# ).tuple +# routing_key = "rjb.d1-time.timecoordinator.sim-timestep" +# properties = pika.BasicProperties( +# # reply_to=self.queue_name, +# # app_id=self.alias, +# type=MessageCategory.RabbitJsonBroadcast, +# correlation_id=str(uuid.uuid4()), +# ) +# +# atn._publish_channel.basic_publish( +# exchange="timecoordinatormic_tx", +# routing_key=routing_key, +# body=payload.as_type(), +# properties=properties, +# ) +# # Wait and Check that not atn_utils.is_dummy_atn_params(atn.atn_params) +# # Check that broadcast "atn.params.report.heatpumpwithbooststore" happened +# atn.stop() + + +# +# def test_price_received_before_timestep(): +# atn = Atn(Settings()) +# tc = TimeCoordinatorStubRecorder(Settings()) +# atn.start() +# tc.start() +# +# wait_for(lambda: atn._publish_channel, 2, "actor publish channel exists") +# +# wait_for(lambda: atn._publish_channel.is_open, 2, "actor publish channel exists") +# +# midnight = pendulum.datetime(year=2020, month=1, day=1, hour=5).int_timestamp +# next_slot = atn.next_run.Slot +# assert next_slot.StartUnixS == midnight +# assert atn.time() < midnight +# assert atn.time() > atn.active_run.Slot.StartUnixS +# assert atn.next_run.Price is None +# +# midnight_price = LatestPrice_Maker( +# from_g_node_alias="d1.isone.ver.keene", +# from_g_node_instance_id=str(uuid.uuid4()), +# price_times1000=22600, +# price_unit=MarketPriceUnit.USDPerMWh, +# market_slot_name=atn_utils.name_from_market_slot(next_slot), +# irl_time_utc=pendulum.from_timestamp(time.time()).to_iso8601_string(), +# message_id=str(uuid.uuid4()), +# ).tuple +# +# midnight_timestep = SimTimestep_Maker( +# from_g_node_alias="d1.time", +# from_g_node_instance_id=str(uuid.uuid4()), +# time_unix_s=midnight, +# timestep_created_ms=int(time.time() * 1000), +# message_id=str(uuid.uuid4()), +# ).tuple +# +# # Atn receives midnight price before midnight timestep +# atn.latest_price_from_market_maker(midnight_price) +# assert tc.atn_ready == False +# assert atn.time() < midnight +# assert atn.next_run.Price == 22.6 +# atn.timestep_from_timecoordinator(midnight_timestep) +# wait_for(lambda: tc.atn_ready, 2, "timecoordinator got ready from atn") +# assert tc.atn_ready == True +# tc.stop() +# atn.stop() +# +# +# def test_timestep_received_before_price(): +# atn = Atn(Settings()) +# tc = TimeCoordinatorStubRecorder(Settings()) +# atn.start() +# tc.start() +# +# wait_for(lambda: atn._publish_channel, 2, "actor publish channel exists") +# +# wait_for(lambda: atn._publish_channel.is_open, 2, "actor publish channel exists") +# +# midnight = pendulum.datetime(year=2020, month=1, day=1, hour=5).int_timestamp +# next_slot = atn.next_run.Slot +# assert next_slot.StartUnixS == midnight +# assert atn.time() < midnight +# assert atn.time() > atn.active_run.Slot.StartUnixS +# assert atn.next_run.Price is None +# +# midnight_price = LatestPrice_Maker( +# from_g_node_alias="d1.isone.ver.keene", +# from_g_node_instance_id=str(uuid.uuid4()), +# price_times1000=22600, +# price_unit=MarketPriceUnit.USDPerMWh, +# market_slot_name=atn_utils.name_from_market_slot(next_slot), +# irl_time_utc=pendulum.from_timestamp(time.time()).to_iso8601_string(), +# message_id=str(uuid.uuid4()), +# ).tuple +# +# midnight_timestep = SimTimestep_Maker( +# from_g_node_alias="d1.time", +# from_g_node_instance_id=str(uuid.uuid4()), +# time_unix_s=midnight, +# timestep_created_ms=int(time.time() * 1000), +# message_id=str(uuid.uuid4()), +# ).tuple +# +# atn.timestep_from_timecoordinator(midnight_timestep) +# assert tc.atn_ready == False +# assert int(atn.time()) == midnight +# assert atn.active_run.Slot.StartUnixS == midnight +# assert atn.active_run.Price is None +# +# atn.latest_price_from_market_maker(midnight_price) +# +# assert atn.active_run.Price == 22.6 +# wait_for(lambda: tc.atn_ready, 2, "timecoordinator got ready from atn") +# assert tc.atn_ready == True +# tc.stop() +# atn.stop() +# +# diff --git a/tests/types/test_atn_params_simpleresistivehydronic.py b/tests/types/test_atn_params_simpleresistivehydronic.py new file mode 100644 index 0000000..046fb2a --- /dev/null +++ b/tests/types/test_atn_params_simpleresistivehydronic.py @@ -0,0 +1,283 @@ +"""Tests atn.params.simpleresistivehydronic type, version 000""" +import json + +import pytest +from gridworks.errors import SchemaError +from pydantic import ValidationError + +from gwatn.enums import DistributionTariff +from gwatn.enums import EnergySupplyType +from gwatn.enums import RecognizedCurrencyUnit +from gwatn.types import AtnParamsSimpleresistivehydronic_Maker as Maker + + +def test_atn_params_simpleresistivehydronic_generated() -> None: + d = { + "GNodeAlias": "d1.isone.ver.keene.holly", + "HomeCity": "MILLINOCKET_ME", + "TimezoneString": "US/Eastern", + "StorageSteps": 100, + "FloSlices": 48, + "SliceDurationMinutes": 60, + "CurrencyUnitGtEnumSymbol": "e57c5143", + "TariffGtEnumSymbol": "2127aba6", + "EnergyTypeGtEnumSymbol": "e9dc99a6", + "StandardOfferPriceDollarsPerMwh": 110, + "DistributionTariffDollarsPerMwh": 113, + "StoreSizeGallons": 240, + "MaxStoreTempF": 210, + "ElementMaxPowerKw": 9.5, + "RequiredSourceWaterTempF": 120, + "FixedPumpGpm": 5.5, + "ReturnWaterFixedDeltaT": 20, + "AnnualHvacKwhTh": 25000, + "AmbientPowerInKw": 1.2, + "HouseWorstCaseTempF": -7, + "RoomTempF": 68, + "TypeName": "atn.params.simpleresistivehydronic", + "Version": "000", + } + + with pytest.raises(SchemaError): + Maker.type_to_tuple(d) + + with pytest.raises(SchemaError): + Maker.type_to_tuple('"not a dict"') + + # Test type_to_tuple + gtype = json.dumps(d) + gtuple = Maker.type_to_tuple(gtype) + + # test type_to_tuple and tuple_to_type maps + assert Maker.type_to_tuple(Maker.tuple_to_type(gtuple)) == gtuple + + # test Maker init + t = Maker( + g_node_alias=gtuple.GNodeAlias, + home_city=gtuple.HomeCity, + timezone_string=gtuple.TimezoneString, + storage_steps=gtuple.StorageSteps, + flo_slices=gtuple.FloSlices, + slice_duration_minutes=gtuple.SliceDurationMinutes, + currency_unit=gtuple.CurrencyUnit, + tariff=gtuple.Tariff, + energy_type=gtuple.EnergyType, + standard_offer_price_dollars_per_mwh=gtuple.StandardOfferPriceDollarsPerMwh, + distribution_tariff_dollars_per_mwh=gtuple.DistributionTariffDollarsPerMwh, + store_size_gallons=gtuple.StoreSizeGallons, + max_store_temp_f=gtuple.MaxStoreTempF, + element_max_power_kw=gtuple.ElementMaxPowerKw, + required_source_water_temp_f=gtuple.RequiredSourceWaterTempF, + fixed_pump_gpm=gtuple.FixedPumpGpm, + return_water_fixed_delta_t=gtuple.ReturnWaterFixedDeltaT, + annual_hvac_kwh_th=gtuple.AnnualHvacKwhTh, + ambient_power_in_kw=gtuple.AmbientPowerInKw, + house_worst_case_temp_f=gtuple.HouseWorstCaseTempF, + room_temp_f=gtuple.RoomTempF, + ).tuple + assert t == gtuple + + ###################################### + # SchemaError raised if missing a required attribute + ###################################### + + d2 = dict(d) + del d2["TypeName"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["GNodeAlias"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["HomeCity"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["TimezoneString"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["StorageSteps"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["FloSlices"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["SliceDurationMinutes"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["CurrencyUnitGtEnumSymbol"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["TariffGtEnumSymbol"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["EnergyTypeGtEnumSymbol"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["StandardOfferPriceDollarsPerMwh"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["DistributionTariffDollarsPerMwh"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["StoreSizeGallons"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["MaxStoreTempF"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["ElementMaxPowerKw"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["RequiredSourceWaterTempF"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["FixedPumpGpm"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["ReturnWaterFixedDeltaT"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["AnnualHvacKwhTh"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["AmbientPowerInKw"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["HouseWorstCaseTempF"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["RoomTempF"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + ###################################### + # Behavior on incorrect types + ###################################### + + d2 = dict(d, StorageSteps="100.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, FloSlices="48.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, SliceDurationMinutes="60.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, CurrencyUnitGtEnumSymbol="hi") + Maker.dict_to_tuple(d2).CurrencyUnit = RecognizedCurrencyUnit.default() + + d2 = dict(d, TariffGtEnumSymbol="hi") + Maker.dict_to_tuple(d2).Tariff = DistributionTariff.default() + + d2 = dict(d, EnergyTypeGtEnumSymbol="hi") + Maker.dict_to_tuple(d2).EnergyType = EnergySupplyType.default() + + d2 = dict(d, StandardOfferPriceDollarsPerMwh="110.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, DistributionTariffDollarsPerMwh="113.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, StoreSizeGallons="240.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, MaxStoreTempF="210.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, ElementMaxPowerKw="this is not a float") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, RequiredSourceWaterTempF="120.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, FixedPumpGpm="this is not a float") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, ReturnWaterFixedDeltaT="20.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, AnnualHvacKwhTh="25000.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, AmbientPowerInKw="this is not a float") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, HouseWorstCaseTempF="-7.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, RoomTempF="68.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + ###################################### + # SchemaError raised if TypeName is incorrect + ###################################### + + d2 = dict(d, TypeName="not the type alias") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + ###################################### + # SchemaError raised if primitive attributes do not have appropriate property_format + ###################################### + + d2 = dict(d, GNodeAlias="a.b-h") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + # End of Test diff --git a/tests/types/test_flo_params_brickstorageheater.py b/tests/types/test_flo_params_brickstorageheater.py index ab1f7b2..46b74a5 100644 --- a/tests/types/test_flo_params_brickstorageheater.py +++ b/tests/types/test_flo_params_brickstorageheater.py @@ -172,6 +172,11 @@ def test_flo_params_brickstorageheater_generated() -> None: with pytest.raises(SchemaError): Maker.dict_to_tuple(d2) + d2 = dict(d) + del d2["C"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + d2 = dict(d) del d2["RealtimeElectricityPrice"] with pytest.raises(SchemaError): @@ -202,6 +207,11 @@ def test_flo_params_brickstorageheater_generated() -> None: with pytest.raises(SchemaError): Maker.dict_to_tuple(d2) + d2 = dict(d) + del d2["DistPriceUid"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + d2 = dict(d) del d2["StartYearUtc"] with pytest.raises(SchemaError): @@ -256,16 +266,6 @@ def test_flo_params_brickstorageheater_generated() -> None: # Optional attributes can be removed from type ###################################### - d2 = dict(d) - if "C" in d2.keys(): - del d2["C"] - Maker.dict_to_tuple(d2) - - d2 = dict(d) - if "DistPriceUid" in d2.keys(): - del d2["DistPriceUid"] - Maker.dict_to_tuple(d2) - d2 = dict(d) if "RegPriceUid" in d2.keys(): del d2["RegPriceUid"] diff --git a/tests/types/test_flo_params_report.py b/tests/types/test_flo_params_report.py deleted file mode 100644 index 4da0ef8..0000000 --- a/tests/types/test_flo_params_report.py +++ /dev/null @@ -1,176 +0,0 @@ -"""Tests flo.params.report type, version 000""" -import json - -import pytest -from gridworks.errors import SchemaError -from pydantic import ValidationError - -from gwatn.types import FloParamsReport_Maker as Maker - - -def test_flo_params_report_generated() -> None: - d = { - "GNodeAlias": "d1.isone.ver.keene.holly", - "GNodeInstanceId": "97eba574-bd20-45b5-bf82-9ba2f492d8f6", - "FloParamsTypeName": "flo.params.brickstorageheater", - "FloParamsTypeVersion": "000", - "ReportGeneratedTimeUnixS": 1577836800, - "IrlTimeUnixS": 1668127823, - "Params": { - "MaxBrickTempC": 190, - "RatedMaxPowerKw": 13.5, - "ROff": 0.08, - "ROn": 0.15, - "RoomTempF": 70, - "CurrencyUnitGtEnumSymbol": "e57c5143", - "TempUnitGtEnumSymbol": "6f16ee63", - "TimezoneString": "US/Eastern", - "HomeCity": "MILLINOCKET_ME", - "IsRegulating": False, - "StorageSteps": 100, - "SliceDurationMinutes": [60], - "PowerRequiredByHouseFromSystemAvgKwList": [3.42], - "C": 200, - "RealtimeElectricityPrice": [10.35], - "OutsideTempF": [-5.1], - "DistributionPrice": [40.0], - "RtElecPriceUid": "bd2ec5c5-40b9-4b61-ad1b-4613370246d6", - "RegulationPrice": [25.3], - "WeatherUid": "3bbcb552-52e3-4b86-84e0-084959f9fc0f", - "DistPriceUid": "b91ef8e7-50d7-4587-bf13-a3af7ecdb83a", - "RegPriceUid": "0499a20e-7b81-47af-a2b4-8f4df0cd1284", - "StartYearUtc": 2020, - "StartMonthUtc": 1, - "StartDayUtc": 1, - "StartHourUtc": 0, - "StartMinuteUtc": 0, - "StartingStoreIdx": 50, - "AmbientPowerInKw": 1.25, - "HouseWorstCaseTempF": -7, - "GNodeAlias": "d1.isone.ver.keene.holly", - "FloParamsUid": "97eba574-bd20-45b5-bf82-9ba2f492d8f6", - "TypeName": "flo.params.brickstorageheater", - "Version": "000", - }, - "TypeName": "flo.params.report", - "Version": "000", - } - - with pytest.raises(SchemaError): - Maker.type_to_tuple(d) - - with pytest.raises(SchemaError): - Maker.type_to_tuple('"not a dict"') - - # Test type_to_tuple - gtype = json.dumps(d) - gtuple = Maker.type_to_tuple(gtype) - - # test type_to_tuple and tuple_to_type maps - assert Maker.type_to_tuple(Maker.tuple_to_type(gtuple)) == gtuple - - # test Maker init - t = Maker( - g_node_alias=gtuple.GNodeAlias, - g_node_instance_id=gtuple.GNodeInstanceId, - flo_params_type_name=gtuple.FloParamsTypeName, - flo_params_type_version=gtuple.FloParamsTypeVersion, - report_generated_time_unix_s=gtuple.ReportGeneratedTimeUnixS, - irl_time_unix_s=gtuple.IrlTimeUnixS, - params=gtuple.Params, - ).tuple - assert t == gtuple - - ###################################### - # SchemaError raised if missing a required attribute - ###################################### - - d2 = dict(d) - del d2["TypeName"] - with pytest.raises(SchemaError): - Maker.dict_to_tuple(d2) - - d2 = dict(d) - del d2["GNodeAlias"] - with pytest.raises(SchemaError): - Maker.dict_to_tuple(d2) - - d2 = dict(d) - del d2["GNodeInstanceId"] - with pytest.raises(SchemaError): - Maker.dict_to_tuple(d2) - - d2 = dict(d) - del d2["FloParamsTypeName"] - with pytest.raises(SchemaError): - Maker.dict_to_tuple(d2) - - d2 = dict(d) - del d2["FloParamsTypeVersion"] - with pytest.raises(SchemaError): - Maker.dict_to_tuple(d2) - - d2 = dict(d) - del d2["ReportGeneratedTimeUnixS"] - with pytest.raises(SchemaError): - Maker.dict_to_tuple(d2) - - d2 = dict(d) - del d2["Params"] - with pytest.raises(SchemaError): - Maker.dict_to_tuple(d2) - - ###################################### - # Optional attributes can be removed from type - ###################################### - - d2 = dict(d) - if "IrlTimeUnixS" in d2.keys(): - del d2["IrlTimeUnixS"] - Maker.dict_to_tuple(d2) - - ###################################### - # Behavior on incorrect types - ###################################### - - d2 = dict(d, ReportGeneratedTimeUnixS="1577836800.1") - with pytest.raises(ValidationError): - Maker.dict_to_tuple(d2) - - d2 = dict(d, IrlTimeUnixS="1668127823.1") - with pytest.raises(ValidationError): - Maker.dict_to_tuple(d2) - - ###################################### - # SchemaError raised if TypeName is incorrect - ###################################### - - d2 = dict(d, TypeName="not the type alias") - with pytest.raises(ValidationError): - Maker.dict_to_tuple(d2) - - ###################################### - # SchemaError raised if primitive attributes do not have appropriate property_format - ###################################### - - d2 = dict(d, GNodeAlias="a.b-h") - with pytest.raises(ValidationError): - Maker.dict_to_tuple(d2) - - d2 = dict(d, GNodeInstanceId="d4be12d5-33ba-4f1f-b9e5") - with pytest.raises(ValidationError): - Maker.dict_to_tuple(d2) - - d2 = dict(d, FloParamsTypeName="a.b-h") - with pytest.raises(ValidationError): - Maker.dict_to_tuple(d2) - - d2 = dict(d, ReportGeneratedTimeUnixS=32503683600) - with pytest.raises(ValidationError): - Maker.dict_to_tuple(d2) - - d2 = dict(d, IrlTimeUnixS=32503683600) - with pytest.raises(ValidationError): - Maker.dict_to_tuple(d2) - - # End of Test diff --git a/tests/types/test_flo_params_simpleresistivehydronic.py b/tests/types/test_flo_params_simpleresistivehydronic.py new file mode 100644 index 0000000..8b63722 --- /dev/null +++ b/tests/types/test_flo_params_simpleresistivehydronic.py @@ -0,0 +1,296 @@ +"""Tests flo.params.simpleresistivehydronic type, version 000""" +import json + +import pytest +from gridworks.errors import SchemaError +from pydantic import ValidationError + +from gwatn.enums import RecognizedCurrencyUnit +from gwatn.types import FloParamsSimpleresistivehydronic_Maker as Maker + + +def test_flo_params_simpleresistivehydronic_generated() -> None: + d = { + "GNodeAlias": "d1.isone.ver.keene.holly", + "FloParamsUid": "97eba574-bd20-45b5-bf82-9ba2f492d8f6", + "HomeCity": "MILLINOCKET_ME", + "TimezoneString": "US/Eastern", + "StartYearUtc": 2020, + "StartMonthUtc": 1, + "StartDayUtc": 1, + "StartHourUtc": 0, + "StartMinuteUtc": 0, + "StoreSizeGallons": 240, + "MaxStoreTempF": 190, + "ElementMaxPowerKw": 9.5, + "RequiredSourceWaterTempF": 120, + "FixedPumpGpm": 4.5, + "ReturnWaterFixedDeltaT": 20, + "SliceDurationMinutes": [60], + "PowerLostFromHouseKwList": [3.42], + "OutsideTempF": [-5.1], + "DistributionPrice": [40.0], + "RealtimeElectricityPrice": [10.35], + "RtElecPriceUid": "bd2ec5c5-40b9-4b61-ad1b-4613370246d6", + "WeatherUid": "3bbcb552-52e3-4b86-84e0-084959f9fc0f", + "DistPriceUid": "b91ef8e7-50d7-4587-bf13-a3af7ecdb83a", + "CurrencyUnitGtEnumSymbol": "e57c5143", + "TypeName": "flo.params.simpleresistivehydronic", + "Version": "000", + } + + with pytest.raises(SchemaError): + Maker.type_to_tuple(d) + + with pytest.raises(SchemaError): + Maker.type_to_tuple('"not a dict"') + + # Test type_to_tuple + gtype = json.dumps(d) + gtuple = Maker.type_to_tuple(gtype) + + # test type_to_tuple and tuple_to_type maps + assert Maker.type_to_tuple(Maker.tuple_to_type(gtuple)) == gtuple + + # test Maker init + t = Maker( + g_node_alias=gtuple.GNodeAlias, + flo_params_uid=gtuple.FloParamsUid, + home_city=gtuple.HomeCity, + timezone_string=gtuple.TimezoneString, + start_year_utc=gtuple.StartYearUtc, + start_month_utc=gtuple.StartMonthUtc, + start_day_utc=gtuple.StartDayUtc, + start_hour_utc=gtuple.StartHourUtc, + start_minute_utc=gtuple.StartMinuteUtc, + store_size_gallons=gtuple.StoreSizeGallons, + max_store_temp_f=gtuple.MaxStoreTempF, + element_max_power_kw=gtuple.ElementMaxPowerKw, + required_source_water_temp_f=gtuple.RequiredSourceWaterTempF, + fixed_pump_gpm=gtuple.FixedPumpGpm, + return_water_fixed_delta_t=gtuple.ReturnWaterFixedDeltaT, + slice_duration_minutes=gtuple.SliceDurationMinutes, + power_lost_from_house_kw_list=gtuple.PowerLostFromHouseKwList, + outside_temp_f=gtuple.OutsideTempF, + distribution_price=gtuple.DistributionPrice, + realtime_electricity_price=gtuple.RealtimeElectricityPrice, + rt_elec_price_uid=gtuple.RtElecPriceUid, + weather_uid=gtuple.WeatherUid, + dist_price_uid=gtuple.DistPriceUid, + currency_unit=gtuple.CurrencyUnit, + ).tuple + assert t == gtuple + + ###################################### + # SchemaError raised if missing a required attribute + ###################################### + + d2 = dict(d) + del d2["TypeName"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["GNodeAlias"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["FloParamsUid"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["HomeCity"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["TimezoneString"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["StartYearUtc"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["StartMonthUtc"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["StartDayUtc"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["StartHourUtc"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["StartMinuteUtc"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["StoreSizeGallons"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["MaxStoreTempF"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["ElementMaxPowerKw"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["RequiredSourceWaterTempF"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["FixedPumpGpm"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["ReturnWaterFixedDeltaT"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["SliceDurationMinutes"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["PowerLostFromHouseKwList"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["OutsideTempF"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["DistributionPrice"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["RealtimeElectricityPrice"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["RtElecPriceUid"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["WeatherUid"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["DistPriceUid"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + d2 = dict(d) + del d2["CurrencyUnitGtEnumSymbol"] + with pytest.raises(SchemaError): + Maker.dict_to_tuple(d2) + + ###################################### + # Behavior on incorrect types + ###################################### + + d2 = dict(d, StartYearUtc="2020.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, StartMonthUtc="1.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, StartDayUtc="1.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, StartHourUtc="0.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, StartMinuteUtc="0.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, StoreSizeGallons="240.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, MaxStoreTempF="190.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, ElementMaxPowerKw="this is not a float") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, RequiredSourceWaterTempF="120.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, FixedPumpGpm="this is not a float") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, ReturnWaterFixedDeltaT="20.1") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, CurrencyUnitGtEnumSymbol="hi") + Maker.dict_to_tuple(d2).CurrencyUnit = RecognizedCurrencyUnit.default() + + ###################################### + # SchemaError raised if TypeName is incorrect + ###################################### + + d2 = dict(d, TypeName="not the type alias") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + ###################################### + # SchemaError raised if primitive attributes do not have appropriate property_format + ###################################### + + d2 = dict(d, GNodeAlias="a.b-h") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, FloParamsUid="d4be12d5-33ba-4f1f-b9e5") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, RtElecPriceUid="d4be12d5-33ba-4f1f-b9e5") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, WeatherUid="d4be12d5-33ba-4f1f-b9e5") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + d2 = dict(d, DistPriceUid="d4be12d5-33ba-4f1f-b9e5") + with pytest.raises(ValidationError): + Maker.dict_to_tuple(d2) + + # End of Test diff --git a/tests/types/test_simplesim_driver_report.py b/tests/types/test_simplesim_driver_report.py index 5276755..2a09ffe 100644 --- a/tests/types/test_simplesim_driver_report.py +++ b/tests/types/test_simplesim_driver_report.py @@ -11,7 +11,7 @@ def test_simplesim_driver_report_generated() -> None: d = { "FromGNodeAlias": "d1.isone.ver.keene.holly", - "FromGNodeInstanecId": "c0cd37c4-d4ae-46d7-baff-af705ea6871a", + "FromGNodeInstanceId": "c0cd37c4-d4ae-46d7-baff-af705ea6871a", "DriverDataTypeName": "simplesim.driver.data.bsh", "DriverData": { "FromGNodeAlias": "d1.isone.ver.keene.holly", @@ -41,7 +41,7 @@ def test_simplesim_driver_report_generated() -> None: # test Maker init t = Maker( from_g_node_alias=gtuple.FromGNodeAlias, - from_g_node_instanec_id=gtuple.FromGNodeInstanecId, + from_g_node_instance_id=gtuple.FromGNodeInstanceId, driver_data_type_name=gtuple.DriverDataTypeName, driver_data=gtuple.DriverData, ).tuple @@ -62,7 +62,7 @@ def test_simplesim_driver_report_generated() -> None: Maker.dict_to_tuple(d2) d2 = dict(d) - del d2["FromGNodeInstanecId"] + del d2["FromGNodeInstanceId"] with pytest.raises(SchemaError): Maker.dict_to_tuple(d2) @@ -96,7 +96,7 @@ def test_simplesim_driver_report_generated() -> None: with pytest.raises(ValidationError): Maker.dict_to_tuple(d2) - d2 = dict(d, FromGNodeInstanecId="d4be12d5-33ba-4f1f-b9e5") + d2 = dict(d, FromGNodeInstanceId="d4be12d5-33ba-4f1f-b9e5") with pytest.raises(ValidationError): Maker.dict_to_tuple(d2) diff --git a/x86.yml b/x86.yml index bf50cef..efeb8bb 100644 --- a/x86.yml +++ b/x86.yml @@ -22,7 +22,7 @@ services: - "dev" hostname: rabbit container_name: atn_rabbit - image: "jessmillar/dev-rabbit-x86:chaos__e58daf6__20230115" + image: "jessmillar/dev-rabbit-x86:chaos__53ea3a0__20230622" ports: - 1885:1885 - 4369:4369