mirror of
https://github.com/d3vyce/teleinfo-exporter.git
synced 2025-04-04 09:10:48 +02:00
Compare commits
6 Commits
Author | SHA1 | Date | |
---|---|---|---|
|
ab4a34bfd7 | ||
bd513a78b7 | |||
b45eda0077 | |||
95a4f288cd | |||
fd6969250d | |||
bda833747d |
64
.github/workflows/build-release.yml
vendored
Normal file
64
.github/workflows/build-release.yml
vendored
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
name: Build Package/Image
|
||||||
|
|
||||||
|
on:
|
||||||
|
release:
|
||||||
|
types: [published]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build-image:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: checkout code
|
||||||
|
uses: actions/checkout@v4.1.1
|
||||||
|
- name: Set up QEMU
|
||||||
|
uses: docker/setup-qemu-action@v3
|
||||||
|
- name: Set up Docker Buildx
|
||||||
|
uses: docker/setup-buildx-action@v3
|
||||||
|
- name: Login to Docker registry
|
||||||
|
uses: docker/login-action@v3
|
||||||
|
with:
|
||||||
|
registry: ghcr.io
|
||||||
|
username: ${{ github.actor }}
|
||||||
|
password: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
- name: Build and push
|
||||||
|
uses: docker/build-push-action@v5.1.0
|
||||||
|
with:
|
||||||
|
context: .
|
||||||
|
platforms: linux/amd64,linux/arm64
|
||||||
|
push: true
|
||||||
|
tags: ghcr.io/${{ github.repository }}:latest,ghcr.io/${{ github.repository }}:${{ github.ref_name }}
|
||||||
|
build-package:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: checkout code
|
||||||
|
uses: actions/checkout@v4.1.1
|
||||||
|
- name: Set up Python
|
||||||
|
uses: actions/setup-python@v5
|
||||||
|
- name: Install dependencies
|
||||||
|
run: |
|
||||||
|
python3 -m pip install --upgrade pip
|
||||||
|
python3 -m pip install --upgrade build
|
||||||
|
- name: Build
|
||||||
|
run: python3 -m build
|
||||||
|
- uses: actions/upload-artifact@v4.3.1
|
||||||
|
with:
|
||||||
|
path: ./dist
|
||||||
|
name: dist
|
||||||
|
pypi-publish:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
needs: build-package
|
||||||
|
environment:
|
||||||
|
name: pypi
|
||||||
|
url: https://pypi.org/p/teleinfo_exporter/
|
||||||
|
permissions:
|
||||||
|
id-token: write
|
||||||
|
steps:
|
||||||
|
- uses: actions/download-artifact@v4.1.2
|
||||||
|
with:
|
||||||
|
path: ./dist
|
||||||
|
name: dist
|
||||||
|
- name: Publish package distributions to PyPI
|
||||||
|
uses: pypa/gh-action-pypi-publish@release/v1
|
||||||
|
with:
|
||||||
|
user: __token__
|
||||||
|
password: ${{ secrets.PYPI_TOKEN }}
|
29
.github/workflows/docker-build-version.yml
vendored
29
.github/workflows/docker-build-version.yml
vendored
@ -1,29 +0,0 @@
|
|||||||
name: Build and Push Docker Image
|
|
||||||
|
|
||||||
on:
|
|
||||||
release:
|
|
||||||
types: [published]
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
build-docker:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: checkout code
|
|
||||||
uses: actions/checkout@v4.1.1
|
|
||||||
- name: Set up QEMU
|
|
||||||
uses: docker/setup-qemu-action@v3
|
|
||||||
- name: Set up Docker Buildx
|
|
||||||
uses: docker/setup-buildx-action@v3
|
|
||||||
- name: Login to Docker registry
|
|
||||||
uses: docker/login-action@v3
|
|
||||||
with:
|
|
||||||
registry: ghcr.io
|
|
||||||
username: ${{ github.actor }}
|
|
||||||
password: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
- name: Build and push
|
|
||||||
uses: docker/build-push-action@v5.1.0
|
|
||||||
with:
|
|
||||||
context: .
|
|
||||||
platforms: linux/amd64,linux/arm64
|
|
||||||
push: true
|
|
||||||
tags: ghcr.io/${{ github.repository }}:latest,ghcr.io/${{ github.repository }}:${{ github.ref_name }}
|
|
39
.github/workflows/pypi-build-version.yml
vendored
39
.github/workflows/pypi-build-version.yml
vendored
@ -1,39 +0,0 @@
|
|||||||
name: Build Pypi Package
|
|
||||||
|
|
||||||
on:
|
|
||||||
release:
|
|
||||||
types: [published]
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
build:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: checkout code
|
|
||||||
uses: actions/checkout@v4.1.1
|
|
||||||
- name: Set up Python
|
|
||||||
uses: actions/setup-python@v5
|
|
||||||
- name: Install dependencies
|
|
||||||
run: |
|
|
||||||
python3 -m pip install --upgrade pip
|
|
||||||
python3 -m pip install --upgrade build
|
|
||||||
- name: Build
|
|
||||||
run: python3 -m build
|
|
||||||
- uses: actions/upload-artifact@v4.3.1
|
|
||||||
with:
|
|
||||||
path: ./dist
|
|
||||||
name: dist
|
|
||||||
pypi-publish:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
needs: build
|
|
||||||
environment:
|
|
||||||
name: pypi
|
|
||||||
url: https://pypi.org/p/teleinfo_exporter/
|
|
||||||
permissions:
|
|
||||||
id-token: write
|
|
||||||
steps:
|
|
||||||
- uses: actions/download-artifact@v4.1.2
|
|
||||||
with:
|
|
||||||
path: ./dist
|
|
||||||
name: dist
|
|
||||||
- name: Publish package distributions to PyPI
|
|
||||||
uses: pypa/gh-action-pypi-publish@release/v1
|
|
@ -1,8 +1,12 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
|
## 1.2.0 - 04-02-2024
|
||||||
|
- fix: catch UnicodeDecodeError when loading json from broker
|
||||||
|
- fix: use .get() to avoid KeyError
|
||||||
|
|
||||||
## 1.1.0 - 02-11-2024
|
## 1.1.0 - 02-11-2024
|
||||||
add: Grafana dashboard to README
|
- add: Grafana dashboard to README
|
||||||
fix: MQTT brocker disconnection when client connection timeout
|
- fix: MQTT broker disconnection when client connection timeout
|
||||||
|
|
||||||
## 1.0.0 - 12-19-2023
|
## 1.0.0 - 12-19-2023
|
||||||
Initial release
|
Initial release
|
||||||
|
@ -3,10 +3,10 @@
|
|||||||

|

|
||||||
|
|
||||||
Simple prometheus exporter for Linky teleinfo.
|
Simple prometheus exporter for Linky teleinfo.
|
||||||
Teleinfo Tasmota project :
|
|
||||||
https://github.com/NicolasBernaerts/tasmota/tree/master/teleinfo
|
[Teleinfo Tasmota project](https://github.com/NicolasBernaerts/tasmota/tree/master/teleinfo)
|
||||||
Grafana Dashboard:
|
|
||||||
https://grafana.com/grafana/dashboards/20182-linky-teleinfo/
|
[Grafana Dashboard](https://grafana.com/grafana/dashboards/20182-linky-teleinfo/)
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
### Pip
|
### Pip
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
[project]
|
[project]
|
||||||
name = "teleinfo-exporter"
|
name = "teleinfo-exporter"
|
||||||
version = "1.1.0"
|
version = "1.2.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"bcrypt ~= 4.1",
|
"bcrypt ~= 4.1",
|
||||||
"configargparse ~= 1.7",
|
"configargparse ~= 1.7",
|
||||||
|
@ -101,51 +101,61 @@ def on_connect(client, userdata, flags, rc): # pylint: disable=unused-argument
|
|||||||
|
|
||||||
|
|
||||||
def on_message(client, userdata, message): # pylint: disable=unused-argument
|
def on_message(client, userdata, message): # pylint: disable=unused-argument
|
||||||
message = json.loads(message.payload.decode("utf-8"))
|
try:
|
||||||
|
message = json.loads(message.payload.decode("utf-8"))
|
||||||
|
except UnicodeDecodeError:
|
||||||
|
pass
|
||||||
|
|
||||||
if "ENERGY" in message:
|
if "ENERGY" in message:
|
||||||
teleinfo_total.set(message["ENERGY"]["Total"])
|
teleinfo_total.set(message.get("ENERGY", {}).get("Total", 0))
|
||||||
teleinfo_yesterday.set(message["ENERGY"]["Yesterday"])
|
teleinfo_yesterday.set(message.get("ENERGY", {}).get("Yesterday", 0))
|
||||||
teleinfo_today.set(message["ENERGY"]["Today"])
|
teleinfo_today.set(message.get("ENERGY", {}).get("Today", 0))
|
||||||
teleinfo_power.set(message["ENERGY"]["Power"])
|
teleinfo_power.set(message.get("ENERGY", {}).get("Power", 0))
|
||||||
teleinfo_apparent_power.set(message["ENERGY"]["ApparentPower"])
|
teleinfo_apparent_power.set(message.get("ENERGY", {}).get("ApparentPower", 0))
|
||||||
teleinfo_reactive_power.set(message["ENERGY"]["ReactivePower"])
|
teleinfo_reactive_power.set(message.get("ENERGY", {}).get("ReactivePower", 0))
|
||||||
teleinfo_power_factor.set(message["ENERGY"]["Factor"])
|
teleinfo_power_factor.set(message.get("ENERGY", {}).get("Factor", 0))
|
||||||
teleinfo_voltage.set(message["ENERGY"]["Voltage"])
|
teleinfo_voltage.set(message.get("ENERGY", {}).get("Voltage", 0))
|
||||||
teleinfo_current.set(message["ENERGY"]["Current"])
|
teleinfo_current.set(message.get("ENERGY", {}).get("Current", 0))
|
||||||
elif "METER" in message:
|
elif "METER" in message:
|
||||||
teleinfo_phases_count.set(message["METER"]["PH"])
|
teleinfo_phases_count.set(message.get("METER", {}).get("PH", 0))
|
||||||
teleinfo_max_current_per_phase.set(message["METER"]["ISUB"])
|
teleinfo_max_current_per_phase.set(message.get("METER", {}).get("ISUB", 0))
|
||||||
teleinfo_max_power_per_phase.set(message["METER"]["PSUB"])
|
teleinfo_max_power_per_phase.set(message.get("METER", {}).get("PSUB", 0))
|
||||||
teleinfo_max_power_per_phase_with_overload.set(message["METER"]["PMAX"])
|
teleinfo_max_power_per_phase_with_overload.set(
|
||||||
teleinfo_total_apparent_power.set(message["METER"]["P"])
|
message.get("METER", {}).get("PMAX", 0)
|
||||||
teleinfo_total_active_power.set(message["METER"]["W"])
|
)
|
||||||
teleinfo_total_current.set(message["METER"]["I"])
|
teleinfo_total_apparent_power.set(message.get("METER", {}).get("P", 0))
|
||||||
|
teleinfo_total_active_power.set(message.get("METER", {}).get("W", 0))
|
||||||
|
teleinfo_total_current.set(message.get("METER", {}).get("I", 0))
|
||||||
for i in range(1, 4):
|
for i in range(1, 4):
|
||||||
if not message["METER"].get(f"U{i}"):
|
if not message.get("METER", {}).get(f"U{i}"):
|
||||||
break
|
break
|
||||||
teleinfo_instant_voltage_per_phase.labels(f"U{i}").set(
|
teleinfo_instant_voltage_per_phase.labels(f"U{i}").set(
|
||||||
message["METER"][f"U{i}"]
|
message.get("METER", {}).get(f"U{i}", 0)
|
||||||
)
|
)
|
||||||
teleinfo_instant_apparent_power_per_phase.labels(f"P{i}").set(
|
teleinfo_instant_apparent_power_per_phase.labels(f"P{i}").set(
|
||||||
message["METER"][f"P{i}"]
|
message.get("METER", {}).get(f"P{i}", 0)
|
||||||
)
|
)
|
||||||
teleinfo_instant_active_power_per_phase.labels(f"W{i}").set(
|
teleinfo_instant_active_power_per_phase.labels(f"W{i}").set(
|
||||||
message["METER"][f"W{i}"]
|
message.get("METER", {}).get(f"W{i}", 0)
|
||||||
)
|
)
|
||||||
teleinfo_instant_current_per_phase.labels(f"I{i}").set(
|
teleinfo_instant_current_per_phase.labels(f"I{i}").set(
|
||||||
message["METER"][f"I{i}"]
|
message.get("METER", {}).get(f"I{i}", 0)
|
||||||
)
|
)
|
||||||
teleinfo_power_factor_per_phase.labels(f"C{i}").set(
|
teleinfo_power_factor_per_phase.labels(f"C{i}").set(
|
||||||
message["METER"][f"C{i}"]
|
message.get("METER", {}).get(f"C{i}", 0)
|
||||||
)
|
)
|
||||||
|
|
||||||
elif "PROD" in message:
|
elif "PROD" in message:
|
||||||
teleinfo_production_instant_apparent_power.set(message["PROD"]["VA"])
|
teleinfo_production_instant_apparent_power.set(
|
||||||
teleinfo_production_instant_active_power.set(message["PROD"]["W"])
|
message.get("PROD", {}).get("VA", 0)
|
||||||
teleinfo_production_power_factor.set(message["PROD"]["COS"])
|
)
|
||||||
|
teleinfo_production_instant_active_power.set(
|
||||||
|
message.get("PROD", {}).get("W", 0)
|
||||||
|
)
|
||||||
|
teleinfo_production_power_factor.set(message.get("PROD", {}).get("COS", 0))
|
||||||
elif "TIC" in message:
|
elif "TIC" in message:
|
||||||
teleinfo_contract_number.labels(message["TIC"]["ADCO"]).set(0)
|
teleinfo_contract_number.labels(message.get("TIC", {}).get("ADCO", 0)).set(0)
|
||||||
teleinfo_contract_type.labels(message["TIC"]["OPTARIF"]).set(0)
|
teleinfo_contract_type.labels(message.get("TIC", {}).get("OPTARIF", 0)).set(0)
|
||||||
|
|
||||||
|
|
||||||
def on_disconnect(client, userdata, rc): # pylint: disable=unused-argument
|
def on_disconnect(client, userdata, rc): # pylint: disable=unused-argument
|
||||||
|
Loading…
x
Reference in New Issue
Block a user