yfinance-exporter 1.0.0__tar.gz
Sign up to get free protection for your applications and to get access to all the features.
@@ -0,0 +1,61 @@
|
|
1
|
+
Metadata-Version: 2.1
|
2
|
+
Name: yfinance-exporter
|
3
|
+
Version: 1.0.0
|
4
|
+
Summary:
|
5
|
+
Author: François Schmidts
|
6
|
+
Author-email: francois@schmidts.fr
|
7
|
+
Requires-Python: >=3.12,<4.0
|
8
|
+
Classifier: Programming Language :: Python :: 3
|
9
|
+
Classifier: Programming Language :: Python :: 3.12
|
10
|
+
Requires-Dist: prometheus-client (>=0.21.1,<0.22.0)
|
11
|
+
Requires-Dist: the-conf (>=1.0.4,<2.0.0)
|
12
|
+
Requires-Dist: yfinance (>=0.2.51,<0.3.0)
|
13
|
+
Description-Content-Type: text/markdown
|
14
|
+
|
15
|
+
[data:image/s3,"s3://crabby-images/7854e/7854e1e39438f8ed4f3f32b79a2ad265d922bb1d" alt="PyPI - Version"](https://pypi.org/project/yfinance-exporter/) [data:image/s3,"s3://crabby-images/246b8/246b8c606657690d9f5efbdac1f2004695df023f" alt="Docker Image Version"](https://hub.docker.com/r/jaesivsm/yfinance-exporter/tags)
|
16
|
+
|
17
|
+
# YFinance Exporter
|
18
|
+
|
19
|
+
|
20
|
+
## Metrics served
|
21
|
+
|
22
|
+
|
23
|
+
```json
|
24
|
+
{
|
25
|
+
"stocks": [
|
26
|
+
{"isin": "FR0000120073", "name": "AIR LIQUIDE", "ycode": "AI.PA"}
|
27
|
+
]
|
28
|
+
}
|
29
|
+
```
|
30
|
+
|
31
|
+
## Running it
|
32
|
+
|
33
|
+
For the next few bits of code, we'll suppose you have a working configuration above in `~/.config/yfinance-exporter.json`.
|
34
|
+
|
35
|
+
### ... with python:
|
36
|
+
|
37
|
+
```shell
|
38
|
+
pip install yfinance-exporter
|
39
|
+
python -m yfinance_exporter
|
40
|
+
```
|
41
|
+
|
42
|
+
### ... with docker:
|
43
|
+
|
44
|
+
```shell
|
45
|
+
docker run -v ~/.config/:/etc/yfinance-exporter/:ro -p 9100:9100 yfinance-exporter:main
|
46
|
+
```
|
47
|
+
|
48
|
+
You'll then be able retrieve some values:
|
49
|
+
|
50
|
+
```shell
|
51
|
+
curl localhost:9100/metrics
|
52
|
+
|
53
|
+
# HELP yfinance_exporter
|
54
|
+
# TYPE yfinance_exporter gauge
|
55
|
+
yfinance_exporter{status="loop"} 1.0
|
56
|
+
yfinance_exporter{status="loop-time"} 10.179875
|
57
|
+
yfinance_exporter{status="ok-stock"} 45.0
|
58
|
+
yfinance_exporter{status="ko-stock"} 2.0
|
59
|
+
[...]
|
60
|
+
```
|
61
|
+
|
@@ -0,0 +1,46 @@
|
|
1
|
+
[data:image/s3,"s3://crabby-images/7854e/7854e1e39438f8ed4f3f32b79a2ad265d922bb1d" alt="PyPI - Version"](https://pypi.org/project/yfinance-exporter/) [data:image/s3,"s3://crabby-images/246b8/246b8c606657690d9f5efbdac1f2004695df023f" alt="Docker Image Version"](https://hub.docker.com/r/jaesivsm/yfinance-exporter/tags)
|
2
|
+
|
3
|
+
# YFinance Exporter
|
4
|
+
|
5
|
+
|
6
|
+
## Metrics served
|
7
|
+
|
8
|
+
|
9
|
+
```json
|
10
|
+
{
|
11
|
+
"stocks": [
|
12
|
+
{"isin": "FR0000120073", "name": "AIR LIQUIDE", "ycode": "AI.PA"}
|
13
|
+
]
|
14
|
+
}
|
15
|
+
```
|
16
|
+
|
17
|
+
## Running it
|
18
|
+
|
19
|
+
For the next few bits of code, we'll suppose you have a working configuration above in `~/.config/yfinance-exporter.json`.
|
20
|
+
|
21
|
+
### ... with python:
|
22
|
+
|
23
|
+
```shell
|
24
|
+
pip install yfinance-exporter
|
25
|
+
python -m yfinance_exporter
|
26
|
+
```
|
27
|
+
|
28
|
+
### ... with docker:
|
29
|
+
|
30
|
+
```shell
|
31
|
+
docker run -v ~/.config/:/etc/yfinance-exporter/:ro -p 9100:9100 yfinance-exporter:main
|
32
|
+
```
|
33
|
+
|
34
|
+
You'll then be able retrieve some values:
|
35
|
+
|
36
|
+
```shell
|
37
|
+
curl localhost:9100/metrics
|
38
|
+
|
39
|
+
# HELP yfinance_exporter
|
40
|
+
# TYPE yfinance_exporter gauge
|
41
|
+
yfinance_exporter{status="loop"} 1.0
|
42
|
+
yfinance_exporter{status="loop-time"} 10.179875
|
43
|
+
yfinance_exporter{status="ok-stock"} 45.0
|
44
|
+
yfinance_exporter{status="ko-stock"} 2.0
|
45
|
+
[...]
|
46
|
+
```
|
@@ -0,0 +1,31 @@
|
|
1
|
+
[tool.poetry]
|
2
|
+
name = "yfinance-exporter"
|
3
|
+
version = "1.0.0"
|
4
|
+
description = ""
|
5
|
+
authors = ["François Schmidts <francois@schmidts.fr>"]
|
6
|
+
readme = "README.md"
|
7
|
+
|
8
|
+
[tool.poetry.dependencies]
|
9
|
+
python = "^3.12"
|
10
|
+
yfinance = "^0.2.51"
|
11
|
+
prometheus-client = "^0.21.1"
|
12
|
+
the-conf = "^1.0.4"
|
13
|
+
|
14
|
+
|
15
|
+
[build-system]
|
16
|
+
requires = ["poetry-core"]
|
17
|
+
build-backend = "poetry.core.masonry.api"
|
18
|
+
|
19
|
+
|
20
|
+
[tool.poetry.group.dev.dependencies]
|
21
|
+
black = "^24.10.0"
|
22
|
+
mypy = "^1.13.0"
|
23
|
+
pycodestyle = "^2.12.1"
|
24
|
+
flake8 = "^7.1.1"
|
25
|
+
pylint = "^3.3.2"
|
26
|
+
ipdb = "^0.13.13"
|
27
|
+
types-requests = "^2.32.0.20241016"
|
28
|
+
|
29
|
+
[tool.black]
|
30
|
+
line-length = 79
|
31
|
+
target-version = ["py312"]
|
@@ -0,0 +1,127 @@
|
|
1
|
+
#!/usr/bin/env python3
|
2
|
+
|
3
|
+
import logging
|
4
|
+
|
5
|
+
from prometheus_client import Gauge, start_http_server
|
6
|
+
from datetime import datetime
|
7
|
+
import time
|
8
|
+
from the_conf import TheConf
|
9
|
+
from yfinance import Ticker
|
10
|
+
|
11
|
+
metaconf = {
|
12
|
+
"source_order": ["files"],
|
13
|
+
"config_files": [
|
14
|
+
"~/.config/yfinance-exporter.json",
|
15
|
+
"/etc/yfinance-exporter/yfinance-exporter.json",
|
16
|
+
],
|
17
|
+
"parameters": [
|
18
|
+
{
|
19
|
+
"type": "list",
|
20
|
+
"stocks": [
|
21
|
+
{"name": {"type": str}},
|
22
|
+
{"isin": {"type": str}},
|
23
|
+
{"ycode": {"type": str}},
|
24
|
+
],
|
25
|
+
},
|
26
|
+
{
|
27
|
+
"loop": [
|
28
|
+
{"interval": {"type": int, "default": 240}},
|
29
|
+
]
|
30
|
+
},
|
31
|
+
{
|
32
|
+
"prometheus": [
|
33
|
+
{"port": {"type": int, "default": 9100}},
|
34
|
+
{"namespace": {"type": str, "default": ""}},
|
35
|
+
]
|
36
|
+
},
|
37
|
+
{"logging": [{"level": {"default": "WARNING"}}]},
|
38
|
+
],
|
39
|
+
}
|
40
|
+
conf = TheConf(metaconf)
|
41
|
+
logger = logging.getLogger("yfinance-exporter")
|
42
|
+
try:
|
43
|
+
logger.setLevel(getattr(logging, conf.logging.level))
|
44
|
+
logger.addHandler(logging.StreamHandler())
|
45
|
+
except AttributeError as error:
|
46
|
+
raise AttributeError(
|
47
|
+
f"{conf.logging.level} isn't accepted, only DEBUG, INFO, WARNING, "
|
48
|
+
"ERROR and FATAL are accepted"
|
49
|
+
) from error
|
50
|
+
|
51
|
+
YFINANCE_EXPORTER = Gauge(
|
52
|
+
"yfinance_exporter",
|
53
|
+
"",
|
54
|
+
["status"],
|
55
|
+
namespace=conf.prometheus.namespace,
|
56
|
+
)
|
57
|
+
STOCK = Gauge(
|
58
|
+
"financial_positions",
|
59
|
+
"",
|
60
|
+
[
|
61
|
+
"bank",
|
62
|
+
"account_type",
|
63
|
+
"account_name",
|
64
|
+
"account_id",
|
65
|
+
"line_name",
|
66
|
+
"line_id",
|
67
|
+
"value_type", # par-value, shares-value, gain, gain-percent, quantity
|
68
|
+
],
|
69
|
+
namespace=conf.prometheus.namespace,
|
70
|
+
)
|
71
|
+
|
72
|
+
|
73
|
+
def collect(stock):
|
74
|
+
logger.debug("Collecting for %r", stock.name)
|
75
|
+
labels = [
|
76
|
+
stock.ycode.split(".")[1] if "." in stock.ycode else "",
|
77
|
+
"stocks",
|
78
|
+
"market",
|
79
|
+
"market",
|
80
|
+
stock.name,
|
81
|
+
stock.isin,
|
82
|
+
"par-value",
|
83
|
+
]
|
84
|
+
ticker = Ticker(stock.ycode)
|
85
|
+
try:
|
86
|
+
value = ticker.fast_info["last_price"]
|
87
|
+
except KeyError:
|
88
|
+
value = None
|
89
|
+
if not isinstance(value, (int, float)):
|
90
|
+
try:
|
91
|
+
STOCK.remove(*labels)
|
92
|
+
except KeyError:
|
93
|
+
pass
|
94
|
+
return False
|
95
|
+
STOCK.labels(*labels).set(value)
|
96
|
+
return True
|
97
|
+
|
98
|
+
|
99
|
+
def main():
|
100
|
+
YFINANCE_EXPORTER.labels("loop-count").set(0)
|
101
|
+
while True:
|
102
|
+
start = datetime.now()
|
103
|
+
|
104
|
+
results = {"ok-stock": 0, "ko-stock": 0}
|
105
|
+
for stock in conf.stocks:
|
106
|
+
if collect(stock):
|
107
|
+
results["ok-stock"] += 1
|
108
|
+
else:
|
109
|
+
results["ko-stock"] += 1
|
110
|
+
|
111
|
+
exec_interval = (datetime.now() - start).total_seconds()
|
112
|
+
YFINANCE_EXPORTER.labels("loop-duration-second").set(exec_interval)
|
113
|
+
YFINANCE_EXPORTER.labels("loop-count").inc()
|
114
|
+
for result, count in results.items():
|
115
|
+
YFINANCE_EXPORTER.labels(result).set(count)
|
116
|
+
|
117
|
+
interval = conf.loop.interval - exec_interval
|
118
|
+
if interval > 0:
|
119
|
+
time.sleep(interval)
|
120
|
+
|
121
|
+
|
122
|
+
if __name__ == "__main__":
|
123
|
+
logger.info(
|
124
|
+
"Starting yfinance exporter with %d stocks to watch", len(conf.stocks)
|
125
|
+
)
|
126
|
+
start_http_server(conf.prometheus.port)
|
127
|
+
main()
|