keihan-tracker 2.2.0__tar.gz
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- keihan_tracker-2.2.0/PKG-INFO +23 -0
- keihan_tracker-2.2.0/README.md +435 -0
- keihan_tracker-2.2.0/keihan_tracker/__init__.py +5 -0
- keihan_tracker-2.2.0/keihan_tracker/bus/__init__.py +1 -0
- keihan_tracker-2.2.0/keihan_tracker/bus/schemes.py +58 -0
- keihan_tracker-2.2.0/keihan_tracker/bus/tracker.py +13 -0
- keihan_tracker-2.2.0/keihan_tracker/delay_tracker.py +153 -0
- keihan_tracker-2.2.0/keihan_tracker/keihan_train/__init__.py +1 -0
- keihan_tracker-2.2.0/keihan_tracker/keihan_train/position_calculation.py +129 -0
- keihan_tracker-2.2.0/keihan_tracker/keihan_train/schemes.py +214 -0
- keihan_tracker-2.2.0/keihan_tracker/keihan_train/stations_map.py +8 -0
- keihan_tracker-2.2.0/keihan_tracker/keihan_train/tracker.py +893 -0
- keihan_tracker-2.2.0/keihan_tracker.egg-info/PKG-INFO +23 -0
- keihan_tracker-2.2.0/keihan_tracker.egg-info/SOURCES.txt +17 -0
- keihan_tracker-2.2.0/keihan_tracker.egg-info/dependency_links.txt +1 -0
- keihan_tracker-2.2.0/keihan_tracker.egg-info/requires.txt +5 -0
- keihan_tracker-2.2.0/keihan_tracker.egg-info/top_level.txt +1 -0
- keihan_tracker-2.2.0/setup.cfg +4 -0
- keihan_tracker-2.2.0/setup.py +29 -0
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
Metadata-Version: 2.4
|
|
2
|
+
Name: keihan_tracker
|
|
3
|
+
Version: 2.2.0
|
|
4
|
+
Summary: 京阪電車・バスの非公式接近情報API解析ライブラリ
|
|
5
|
+
Home-page: https://github.com/dk-butsuri/keihan_tracker
|
|
6
|
+
Download-URL: https://github.com/dk-butsuri/keihan_tracker
|
|
7
|
+
Author: dk-butsuri
|
|
8
|
+
Maintainer: dk-butsuri
|
|
9
|
+
License: MIT License
|
|
10
|
+
Requires-Python: >=3.10
|
|
11
|
+
Requires-Dist: pydantic
|
|
12
|
+
Requires-Dist: httpx
|
|
13
|
+
Requires-Dist: tabulate
|
|
14
|
+
Requires-Dist: beautifulsoup4
|
|
15
|
+
Requires-Dist: tzdata
|
|
16
|
+
Dynamic: author
|
|
17
|
+
Dynamic: download-url
|
|
18
|
+
Dynamic: home-page
|
|
19
|
+
Dynamic: license
|
|
20
|
+
Dynamic: maintainer
|
|
21
|
+
Dynamic: requires-dist
|
|
22
|
+
Dynamic: requires-python
|
|
23
|
+
Dynamic: summary
|
|
@@ -0,0 +1,435 @@
|
|
|
1
|
+
# keihan_tracker
|
|
2
|
+
|
|
3
|
+
京阪電車 リアルタイム列車位置情報API 非公式Pythonライブラリ
|
|
4
|
+
|
|
5
|
+
> ⚠️ **注意: 本ライブラリは京阪電車の非公式APIラッパーです。不具合やAPI仕様の変更などの可能性があります。利用は私的かつ自己責任でお願いします。**
|
|
6
|
+
|
|
7
|
+
---
|
|
8
|
+
|
|
9
|
+
## 概要
|
|
10
|
+
京阪電車公式APIのJSONデータをPydanticモデルでバリデートし、Pythonクラスから駅・路線・列車・ダイヤ情報を直感的に操作できるライブラリです。
|
|
11
|
+
|
|
12
|
+
「現在どこを走っているか」だけでなく、「これからどのような列車が来るか」というダイヤ情報も統合して扱えるため、駅の発車標のようなアプリケーションも簡単に作成できます。
|
|
13
|
+
|
|
14
|
+
### 特徴
|
|
15
|
+
* **ダイヤ・位置情報の統合**: 走行中の列車(`ActiveTrainData`)と、これから走る予定の列車(`TrainData`)を統一的に扱えます。
|
|
16
|
+
* **種別・方向の自動推定**: 予定データ(ダイヤ)には本来含まれていない「列車種別」や「進行方向」を、停車駅リストから自動的に推定します。
|
|
17
|
+
* **相互リンク構造**: 「この駅に来る列車」⇔「この列車の次の駅」といった情報を双方向に参照可能。
|
|
18
|
+
* **型安全**: 公式APIのJSONをPydanticで厳密に定義。エディタの補完が効きます。
|
|
19
|
+
* **高度な位置特定**: 路線図上のグリッド座標から「走行中」か「停車中」かを独自ロジックで判定。
|
|
20
|
+
* **バス・遅延情報**: 電車だけでなく、京阪バスの接近情報や関西エリア全体の運行情報も取得可能。
|
|
21
|
+
* **レートリミット内蔵**: サーバー負荷を考慮し、15秒/reqのレート制限を内蔵。意図しない高頻度取得を防ぎます。
|
|
22
|
+
|
|
23
|
+
## データ構造と関係性
|
|
24
|
+
本ライブラリは、**全体管理クラス**、**駅クラス**、**列車クラス**が相互に繋がっています。
|
|
25
|
+
|
|
26
|
+
```text
|
|
27
|
+
[KHTracker] (全体管理)
|
|
28
|
+
│
|
|
29
|
+
├── .stations (全駅: dict)
|
|
30
|
+
│ └── [StationData] (駅)
|
|
31
|
+
│ ├── .upcoming_trains ──→ 停車中・今後この駅に停車する [TrainData] のリスト
|
|
32
|
+
│ └── .arriving_trains ──→ 停車中・次にこの駅に停車する [TrainData]のリスト
|
|
33
|
+
│
|
|
34
|
+
└── .trains (全列車: dict)
|
|
35
|
+
└── [TrainData|ActiveTrainData] (列車)
|
|
36
|
+
├── .next_stop_station ──→ 次に止まる [StationData]
|
|
37
|
+
├── .stop_stations ──→ 停車駅リスト
|
|
38
|
+
├── .train_type ──→ [推定] 停車駅に基づく種別
|
|
39
|
+
├── .direction ──→ [推定] 始発・終着に基づく方向
|
|
40
|
+
└── その他多数...
|
|
41
|
+
```
|
|
42
|
+
|
|
43
|
+
## インストール
|
|
44
|
+
|
|
45
|
+
```bash
|
|
46
|
+
pip install git+https://github.com/dk-butsuri/keihan_tracker.git
|
|
47
|
+
```
|
|
48
|
+
|
|
49
|
+
## 依存パッケージ
|
|
50
|
+
```pip install```時にインストールされます。
|
|
51
|
+
- Python 3.10+
|
|
52
|
+
- pydantic
|
|
53
|
+
- httpx
|
|
54
|
+
- tabulate
|
|
55
|
+
- beautifulsoup4 (Yahoo!遅延情報取得に使用)
|
|
56
|
+
- tzdata (タイムゾーン情報)
|
|
57
|
+
|
|
58
|
+
## 使い方
|
|
59
|
+
|
|
60
|
+
### 1. 基本的なデータ取得(電車)
|
|
61
|
+
一度だけデータを取得して表示する例です。
|
|
62
|
+
|
|
63
|
+
```python
|
|
64
|
+
import asyncio
|
|
65
|
+
from keihan_tracker import KHTracker, TrainType
|
|
66
|
+
|
|
67
|
+
async def main():
|
|
68
|
+
tracker = KHTracker()
|
|
69
|
+
|
|
70
|
+
# APIから最新データを取得・更新
|
|
71
|
+
# 駅情報などの静的データは初回のみ取得、列車位置は毎回更新されます
|
|
72
|
+
# 注意: これを呼ばないとデータは空のままです!
|
|
73
|
+
await tracker.fetch_pos()
|
|
74
|
+
|
|
75
|
+
# --- 特定の駅から情報を見る ---
|
|
76
|
+
# 駅番号はKH番号の数字部分です (例: KH01 淀屋橋 -> 1, KH40 三条 -> 40)
|
|
77
|
+
station = tracker.stations[17]
|
|
78
|
+
print(f"=== {station.station_name.ja}駅 ===")
|
|
79
|
+
|
|
80
|
+
# 次の列車を取得
|
|
81
|
+
next_train, arrival_time = station.upcoming_trains[0]
|
|
82
|
+
|
|
83
|
+
if arrival_time.time is not None:
|
|
84
|
+
time = arrival_time.time.strftime("%H時%M分 到着予定")
|
|
85
|
+
else:
|
|
86
|
+
time = ""
|
|
87
|
+
|
|
88
|
+
print(time)
|
|
89
|
+
print(next_train)
|
|
90
|
+
|
|
91
|
+
asyncio.run(main())
|
|
92
|
+
```
|
|
93
|
+
|
|
94
|
+
### 2. リアルタイム監視(ポーリング)
|
|
95
|
+
定期的に情報を更新して、列車の動きを監視するパターンの例です。
|
|
96
|
+
`fetch_pos()` をループ内で呼び出し続けてください。
|
|
97
|
+
|
|
98
|
+
```python
|
|
99
|
+
import asyncio
|
|
100
|
+
from keihan_tracker import KHTracker
|
|
101
|
+
|
|
102
|
+
async def watch_loop():
|
|
103
|
+
tracker = KHTracker()
|
|
104
|
+
|
|
105
|
+
while True:
|
|
106
|
+
# データを更新
|
|
107
|
+
await tracker.fetch_pos()
|
|
108
|
+
|
|
109
|
+
# 例: 現在最も遅れている列車を表示
|
|
110
|
+
worst_train = tracker.max_delay_train
|
|
111
|
+
max_delay = tracker.max_delay_minutes
|
|
112
|
+
|
|
113
|
+
print(f"\r現在走行中の列車数: {len(tracker.trains)} | 最大遅延: {max_delay}分")
|
|
114
|
+
|
|
115
|
+
if worst_train:
|
|
116
|
+
print(f"⚠️ {worst_train.train_number}号 ({worst_train.train_type.value}) が {max_delay}分 遅延しています!")
|
|
117
|
+
|
|
118
|
+
# 60秒待機 (京阪側の更新頻度は1分間隔)
|
|
119
|
+
await asyncio.sleep(60)
|
|
120
|
+
|
|
121
|
+
if __name__ == "__main__":
|
|
122
|
+
asyncio.run(watch_loop())
|
|
123
|
+
```
|
|
124
|
+
|
|
125
|
+
### 3. 遅延を加味した停車駅到着時刻の推定
|
|
126
|
+
|
|
127
|
+
`delay_minutes` と `get_stop_time()` を組み合わせるだけでは、列車が「単に遅延して走っている」のか「途中駅で足止めされている」のかを区別できません。大幅遅延時は `stopping_time` と `is_at_start_station` を使うことで、推定到着時刻の信頼度を判定できます。
|
|
128
|
+
遅延分数を停車時刻表に表示する際は、どの駅で何分間停車しているか表示すると良いでしょう。
|
|
129
|
+
|
|
130
|
+
```python
|
|
131
|
+
from datetime import timedelta
|
|
132
|
+
from keihan_tracker import KHTracker, ActiveTrainData, StationData
|
|
133
|
+
|
|
134
|
+
STUCK_THRESHOLD = timedelta(minutes=15) # この時間以上停車なら足止めと判定
|
|
135
|
+
|
|
136
|
+
def estimate_arrival(train: ActiveTrainData, station: StationData):
|
|
137
|
+
"""
|
|
138
|
+
遅延を加味した推定到着時刻を返す。
|
|
139
|
+
足止めと判定した場合は None を返す(推定不可)。
|
|
140
|
+
"""
|
|
141
|
+
scheduled = train.get_stop_time(station)
|
|
142
|
+
if scheduled is None:
|
|
143
|
+
return None
|
|
144
|
+
|
|
145
|
+
estimated = scheduled + timedelta(minutes=train.delay_minutes)
|
|
146
|
+
|
|
147
|
+
# 始発駅での長時間停車は正常(発車待ち)なので除外
|
|
148
|
+
if train.is_at_start_station:
|
|
149
|
+
return estimated
|
|
150
|
+
|
|
151
|
+
# 閾値以上停車中なら足止めの可能性があり推定信頼度が低い
|
|
152
|
+
if train.stopping_time > STUCK_THRESHOLD:
|
|
153
|
+
return None # 推定不可
|
|
154
|
+
|
|
155
|
+
return estimated
|
|
156
|
+
```
|
|
157
|
+
|
|
158
|
+
### 4. 京阪バス接近情報の取得
|
|
159
|
+
京阪グループ「BUS NAVI」から接近情報を取得します。
|
|
160
|
+
|
|
161
|
+
**注:利用規約では「商業的二次利用」が禁止されており、私的利用に限ります。**
|
|
162
|
+
また使用に際しては、[京阪グループBUS NAVI利用規約](https://busnavi.keihanbus.jp/pc/termsofuse)を必ずご確認ください。
|
|
163
|
+
|
|
164
|
+
```python
|
|
165
|
+
import asyncio
|
|
166
|
+
from keihan_tracker.bus.tracker import get_khbus_info
|
|
167
|
+
|
|
168
|
+
async def main():
|
|
169
|
+
# バス停名と番号を指定(番号はのりばや方向によって異なります)
|
|
170
|
+
bus_info = await get_khbus_info("京阪香里園", 1)
|
|
171
|
+
|
|
172
|
+
for bus in bus_info.body.busstates:
|
|
173
|
+
prms = bus.busstateprms
|
|
174
|
+
print(f"[{prms.route}] {prms.destination} 行き")
|
|
175
|
+
print(f" 状況: {prms.status} (予定: {prms.timetable})")
|
|
176
|
+
print(f" 現在地: 緯度{prms.lat}, 経度{prms.lon}")
|
|
177
|
+
print("\n")
|
|
178
|
+
|
|
179
|
+
if __name__ == "__main__":
|
|
180
|
+
asyncio.run(main())
|
|
181
|
+
```
|
|
182
|
+
|
|
183
|
+
### 5. 運行情報(遅延情報)の取得
|
|
184
|
+
遅延情報の取得には2つの方法があります。
|
|
185
|
+
* Yahoo!路線情報 (スクレイピング)
|
|
186
|
+
* 駅すぱあと 運行情報API(要契約)
|
|
187
|
+
#### A. Yahoo!路線情報 (スクレイピング)
|
|
188
|
+
関西エリア(デフォルト)の運行情報を取得します。
|
|
189
|
+
|
|
190
|
+
**注:Yahoo!路線情報の運行情報は[二次利用を禁止しており](https://support.yahoo-net.jp/PccTransit/s/article/H000007493)、私的利用に限ります。**
|
|
191
|
+
利用に際しては、[LINEヤフー共通利用規約](https://www.lycorp.co.jp/ja/company/terms/)も必ずご覧ください。
|
|
192
|
+
|
|
193
|
+
```python
|
|
194
|
+
import asyncio
|
|
195
|
+
from keihan_tracker.delay_tracker import get_yahoo_delay
|
|
196
|
+
|
|
197
|
+
async def main():
|
|
198
|
+
# エリアコードを指定可能(デフォルト6=近畿)
|
|
199
|
+
delays = await get_yahoo_delay()
|
|
200
|
+
|
|
201
|
+
print("【現在の運行情報 (Yahoo!)】")
|
|
202
|
+
if not delays:
|
|
203
|
+
print("現在、目立った遅延情報はありません。")
|
|
204
|
+
|
|
205
|
+
for info in delays:
|
|
206
|
+
print(f"・{info.LineName}: {info.status}")
|
|
207
|
+
print(f" {info.detail}\n")
|
|
208
|
+
|
|
209
|
+
if __name__ == "__main__":
|
|
210
|
+
asyncio.run(main())
|
|
211
|
+
```
|
|
212
|
+
|
|
213
|
+
#### B. 駅すぱあと API
|
|
214
|
+
駅すぱあとのAPIキーをお持ちの場合は、こちらを利用することでより安定して情報を取得できます。
|
|
215
|
+
レスキューナウから提供されている信頼性の高い運行情報を使用できます。
|
|
216
|
+
|
|
217
|
+
```python
|
|
218
|
+
import asyncio
|
|
219
|
+
from keihan_tracker.delay_tracker import get_ekispert_delay
|
|
220
|
+
|
|
221
|
+
async def main():
|
|
222
|
+
# APIキーを指定して取得
|
|
223
|
+
# prefs引数で都道府県コードを指定可能(デフォルトは京都・大阪・兵庫)
|
|
224
|
+
delays = await get_ekispert_delay(api_key="YOUR_API_KEY")
|
|
225
|
+
|
|
226
|
+
print("【現在の運行情報 (駅すぱあと)】")
|
|
227
|
+
for info in delays:
|
|
228
|
+
print(f"・{info.LineName}: {info.status}")
|
|
229
|
+
print(f" {info.detail}\n")
|
|
230
|
+
|
|
231
|
+
if __name__ == "__main__":
|
|
232
|
+
asyncio.run(main())
|
|
233
|
+
```
|
|
234
|
+
|
|
235
|
+
## 知っておくべき仕様・注意点 (ハマりポイント)
|
|
236
|
+
|
|
237
|
+
このライブラリを使用する際の注意点を以下に挙げます。
|
|
238
|
+
|
|
239
|
+
1. **非同期処理が必須 (`async`/`await`)**
|
|
240
|
+
* 本ライブラリは `httpx` を使用した非同期設計です。`tracker.fetch_pos()` などのメソッドは必ず `await` する必要があります。
|
|
241
|
+
* 通常の `def main():` ではなく、`async def main():` と書き、`asyncio.run(main())` で実行してください。
|
|
242
|
+
|
|
243
|
+
2. **`fetch_pos()` を呼ばないとデータは空**
|
|
244
|
+
* `KHTracker()` をインスタンス化した直後は、駅データも列車データも空です。
|
|
245
|
+
* 必ず最初に `await tracker.fetch_pos()` を呼び出してください。また、列車位置は自動更新されないため、最新位置を知るには定期的(30秒から数分間隔が目安)にこのメソッドを呼ぶ必要があります。
|
|
246
|
+
|
|
247
|
+
3. **「走行中」と「予定」の列車クラスの違い**
|
|
248
|
+
* **ActiveTrainData**: 現在APIに位置情報が存在する列車です。`is_stopping` (停車中か) や `location_col` (座標) などのリアルタイムなプロパティを持ちます。
|
|
249
|
+
* **TrainData**: ダイヤ上の予定データ、または運行終了したデータです。これらには現在位置情報がないため、`is_stopping` などのプロパティにアクセスするとエラーにはなりませんが、常にFalseなどのデフォルト値であったり、意味を持たない場合があります。
|
|
250
|
+
* `tracker.trains` には両方が混在しています。区別するには `isinstance(train, ActiveTrainData)` を使用してください。
|
|
251
|
+
* **臨時列車 (`is_special=True`) は必ず `ActiveTrainData`** です。走行が確認された時点で `ActiveTrainData` として登録され、同時にダイヤ情報も登録されます。走行が確認されるまではダイヤ登録をスキップするため、当日運転しない臨時列車のゴースト登録を防いでいます。(2026GWで運行された臨時列車でゴースト現象を確認済み)
|
|
252
|
+
|
|
253
|
+
4. **駅番号は整数 (int)**
|
|
254
|
+
* 駅番号(ナンバリング)は `KH01` のような文字列ではなく、数字部分の整数 `1` として扱います。辞書のキーも整数です。
|
|
255
|
+
|
|
256
|
+
5. **時刻は日本標準時 (JST)**
|
|
257
|
+
* ライブラリ内で扱われる `datetime` オブジェクトには、すべてタイムゾーン情報(`Asia/Tokyo`)が付与されています。
|
|
258
|
+
* 現在時刻と比較する場合は、`datetime.now()` ではなく `datetime.now(ZoneInfo("Asia/Tokyo"))` などと比較しないとエラーになります。
|
|
259
|
+
|
|
260
|
+
## クラスリファレンス
|
|
261
|
+
|
|
262
|
+
### KHTracker (電車)
|
|
263
|
+
ライブラリのルートとなる管理クラス。
|
|
264
|
+
|
|
265
|
+
```python
|
|
266
|
+
KHTracker(rate_limit: float = 15)
|
|
267
|
+
```
|
|
268
|
+
`rate_limit`: `fetch_pos()` の最小呼び出し間隔(秒)。デフォルトは15秒。これよりも高頻度で実行すると、取得処理がスキップされる。
|
|
269
|
+
|
|
270
|
+
* `stations: dict[int, StationData]`: 駅データ。キーは駅番号の整数値(KH01なら1)。
|
|
271
|
+
* `trains: dict[int, TrainData | ActiveTrainData]`: 全列車データ(走行中・予定・終了含む)。キーは内部管理番号(WDF)。
|
|
272
|
+
* `active_trains: dict[int, ActiveTrainData]`: 現在走行中の列車データのみを抽出した辞書
|
|
273
|
+
* `date: datetime.date`: 現在の営業日(24-5時の深夜帯は前日扱い)
|
|
274
|
+
* `max_delay_train: Optional[ActiveTrainData]`: 最も遅延している列車(0分の場合はNoneを返す)
|
|
275
|
+
* `max_delay_minutes: int`: 現在の最大遅延分数
|
|
276
|
+
* `last_fetch_pos_datetime: Optional[datetime]`: 最後に `fetch_pos()` を実行した時刻
|
|
277
|
+
* `rate_limit_interval: float`: 現在設定されているレートリミット間隔(秒)
|
|
278
|
+
|
|
279
|
+
#### 主要メソッド
|
|
280
|
+
* `async fetch_pos()`: **[重要]** 最新の列車位置・遅延情報をAPIから取得し、インスタンス内のデータを更新します。
|
|
281
|
+
* `async regist_dia(download: bool)`: ダイヤ情報を更新します。通常は `fetch_pos()` から自動的に呼び出されるため、実行する必要はありません。
|
|
282
|
+
* `find_trains(...)`: 条件に合致する列車をリストで返します。全引数はオプションで、省略した項目は絞り込み対象外となります。
|
|
283
|
+
|
|
284
|
+
`find_trains(...)` の引数:
|
|
285
|
+
|
|
286
|
+
| 引数 | 型 | 説明 |
|
|
287
|
+
|:---|:---|:---|
|
|
288
|
+
| `status` | `"active" \| "scheduled" \| "completed"` | 運行状態。`status` 参照 |
|
|
289
|
+
| `train_type` | `TrainType` | 列車種別 |
|
|
290
|
+
| `direction` | `"up" \| "down"` | 進行方向(ActiveTrainData のみ有効) |
|
|
291
|
+
| `is_special` | `bool` | 臨時列車かどうか(ActiveTrainData のみ有効) |
|
|
292
|
+
| `train_number` | `str` | 列車番号(例: `"1051"`)(ActiveTrainData のみ有効) |
|
|
293
|
+
| `has_premiumcar` | `bool` | プレミアムカー連結の有無 |
|
|
294
|
+
| `destination` | `StationData` | 行先駅 |
|
|
295
|
+
| `next_stop_station` | `StationData` | 次の停車駅(ActiveTrainData のみ有効) |
|
|
296
|
+
| `min_delay` | `int` | この分数を**超える**遅延のみ対象 |
|
|
297
|
+
| `max_delay` | `int` | この分数**未満**の遅延のみ対象 |
|
|
298
|
+
| `is_stopping` | `bool` | 停車中かどうか(ActiveTrainData のみ有効) |
|
|
299
|
+
|
|
300
|
+
### 列車データの構造 (TrainData と ActiveTrainData)
|
|
301
|
+
|
|
302
|
+
本ライブラリでは、列車の状態によって2つのクラスが使われます。
|
|
303
|
+
`ActiveTrainData` は `TrainData` を継承しており、**「TrainDataの全情報 + リアルタイム位置情報」** を持っています。
|
|
304
|
+
|
|
305
|
+
判定する際は`isinstance()`関数を用いると、型厳密に、そしてエディターで入力候補が使用できます。
|
|
306
|
+
|
|
307
|
+
| 項目 | TrainData (予定・終了) | ActiveTrainData (走行中) | 備考 |
|
|
308
|
+
| :--- | :--- | :--- | :--- |
|
|
309
|
+
| **基本情報** | ◎ | ◎ | ID, 編成, 行先など |
|
|
310
|
+
| **ダイヤ情報** | ◎ | ◎ | 停車駅リスト, 到着時刻 |
|
|
311
|
+
| **列車種別** | △ **[推定]** | ◎ | 予定/終了の場合、停車駅から推定、ライナーは識別不可 |
|
|
312
|
+
| **進行方向** | △ **[推定]** | ◎ | 予定/終了の場合、始発・終着から推定 |
|
|
313
|
+
| **現在位置** | × | **○** **[一部推定]** | 座標, 停車中判定 |
|
|
314
|
+
| **遅延情報** | × | **◎** | 遅延分数 |
|
|
315
|
+
|
|
316
|
+
#### TrainData (基底クラス)
|
|
317
|
+
全ての列車のベースとなるクラスです。主にダイヤ情報(静的な予定)を保持します。
|
|
318
|
+
* `master`: KHTrackerインスタンス
|
|
319
|
+
* `wdfBlockNo: int`: 列車管理番号
|
|
320
|
+
* `date: datetime.date`: この列車が属する営業日。0〜5時の深夜帯は前日扱いとハードコーディングされています。
|
|
321
|
+
* `destination: StationData`: 行先駅
|
|
322
|
+
* `start_station: StationData`: 始発駅
|
|
323
|
+
* `stop_stations: list[StopStationData]`: 全停車駅のリスト
|
|
324
|
+
* `route_stations: list[StopStationData]`: 停車・通過駅のリスト(一部の通過駅のみが含まれる)
|
|
325
|
+
* `has_premiumcar: Optional[bool]`: プレミアムカーがあるか
|
|
326
|
+
* `delay_minutes: int` 遅延分数(`TrainData` では常に0)
|
|
327
|
+
* `train_formation: Optional[int]` 列車編成(3003など)
|
|
328
|
+
* `train_type: TrainType`: **[推定]** 停車駅パターンから推定された種別
|
|
329
|
+
* `direction: Literal["up","down"]`: **[推定]** 始発・終着から推定された方向
|
|
330
|
+
* `line: LineLiteral`: **[推定]** 列車の走行路線("京阪本線・鴨東線", "宇治線" など)
|
|
331
|
+
* `status: Literal["active","scheduled","completed"]`: 運行状態。**`"active"` のみ正確**。`"scheduled"` / `"completed"` はダイヤ上の予定時刻から推定するため精度は保証されません。`find_trains(status="active")` などと組み合わせて使います。
|
|
332
|
+
* `is_completed: bool`: 運行完了フラグ。`status` 同様、精度は保証されません。
|
|
333
|
+
|
|
334
|
+
なお、`ActiveTrainData` が運行終了後に `TrainData` に降格した際、`direction` と `train_type` は引き継がれます。
|
|
335
|
+
|
|
336
|
+
#### ActiveTrainData (継承クラス)
|
|
337
|
+
現在走行中の列車です。`TrainData` に加え、以下の**リアルタイム情報**を持ちます。
|
|
338
|
+
* `is_stopping: bool`: 現在、駅に停車中かどうか
|
|
339
|
+
* `next_stop_station: StationData`: 次に停車する駅(**停車中の場合はその駅**)
|
|
340
|
+
* `next_station: StationData`: 次に停車・**通過**する駅
|
|
341
|
+
* `delay_minutes: int`: 遅延分数(定刻なら0)
|
|
342
|
+
* `delay_text: MultiLang`: 遅延テキスト(「約5分」など各言語で)
|
|
343
|
+
* `train_number: str`: 列車番号 (例: "1051"(号)など)
|
|
344
|
+
* `cars: int`: 車両数
|
|
345
|
+
* `location_col`, `location_row`: zaisen上のグリッド座標
|
|
346
|
+
* `is_special: bool`: 臨時列車かどうか
|
|
347
|
+
* `is_at_start_station: bool`: 現在、始発駅に停車中かどうか
|
|
348
|
+
* `stopping_time: datetime.timedelta`: 現在の駅に停車している時間。走行中は `timedelta(0)` を返す
|
|
349
|
+
* `lastpass_station: Optional[StationData]`: ⚠️ **使用非推奨**。基本的に停車中 or 次に停車する駅。公式zaisenページの列車詳細における停車時刻表示の開始基準として使われる内部値であり、「最後に通過した駅」として厳密に管理されているわけではありません。例えば萱島~京橋を走る各駅列車でも常に萱島が設定されるなど、路線・区間によって信頼性の低い値が返ります。
|
|
350
|
+
|
|
351
|
+
### StationData
|
|
352
|
+
駅を表すクラス。
|
|
353
|
+
|
|
354
|
+
* `station_number: int`: 駅番号 (例: 1)
|
|
355
|
+
* `station_name: MultiLang`: 駅名(.ja, .en 等)
|
|
356
|
+
* `line: set[LineLiteral]`: 所属路線(例: `{"京阪本線・鴨東線", "中之島線"}`)
|
|
357
|
+
* `transfer: StationConnections`: 乗換情報。JR・地下鉄・モノレールなどへの乗り入れ路線を保持します。
|
|
358
|
+
* `arriving_trains: list[ActiveTrainData]`: 停車中、もしくは**次に**停車する走行中列車のリスト
|
|
359
|
+
* `upcoming_trains: list[tuple[TrainData | ActiveTrainData, StopStationData]]`: 停車中、もしくは**今後**停車する全列車とその到着時刻のリスト(時刻順)。
|
|
360
|
+
* `trains: list[tuple[TrainData | ActiveTrainData, StopStationData]]`: **過去・現在・未来すべて**の停車列車リスト(時刻順)。`upcoming_trains` が「これから停車する列車」のみを返すのに対し、こちらはすでに通過済みの列車も含みます。発車標ではなく運行履歴や全停車情報が必要な場合に使用します。
|
|
361
|
+
|
|
362
|
+
### StopStationData
|
|
363
|
+
列車の停車・通過駅を表すクラス。train.stop_stations や station.upcoming_trains の戻り値に含まれます。
|
|
364
|
+
* station: StationData: 駅
|
|
365
|
+
* time: Optional[datetime]: 到着/出発時刻(始発・終着・通過駅などでNoneの場合あり)
|
|
366
|
+
* is_stop: bool: 停車するかどうか(通過駅ならFalse)
|
|
367
|
+
* is_start: bool: この駅が始発駅かどうか
|
|
368
|
+
* is_final: bool: この駅が終着駅かどうか
|
|
369
|
+
|
|
370
|
+
### BusLocationResponse / BusStatePrms (バス)
|
|
371
|
+
バス接近情報のレスポンスモデル。
|
|
372
|
+
|
|
373
|
+
* `BusLocationResponse.body.busstates`: `BusState` のリスト
|
|
374
|
+
* `BusState.busstateprms`: 詳細情報オブジェクト
|
|
375
|
+
* `route: str`: 系統番号 (例: "[22]")
|
|
376
|
+
* `destination: str`: 行先
|
|
377
|
+
* `status: str`: 運行状況テキスト
|
|
378
|
+
* `timetable: str`: 到着予定時刻など
|
|
379
|
+
* `lat`, `lon`: バスの現在座標
|
|
380
|
+
|
|
381
|
+
### DelayLine (遅延情報)
|
|
382
|
+
Yahoo!路線情報または駅すぱあとから取得した遅延情報モデル。
|
|
383
|
+
|
|
384
|
+
* `LineName: str`: 路線名 (例: "京阪本線")
|
|
385
|
+
* `status: str`: 状態概要 (例: "遅延")
|
|
386
|
+
* `detail: str`: 詳細な遅延理由や状況のテキスト
|
|
387
|
+
* `AnnouncedTime: datetime`: 運行情報の発表時刻
|
|
388
|
+
|
|
389
|
+
### MultiLang
|
|
390
|
+
多言語対応の文字列型。`station_name` や `delay_text` などで使われます。
|
|
391
|
+
|
|
392
|
+
* `ja: str`: 日本語
|
|
393
|
+
* `en: str`: 英語
|
|
394
|
+
* `cn: str`: 簡体字中国語
|
|
395
|
+
* `tw: str`: 繁体字中国語
|
|
396
|
+
* `kr: str`: 韓国語
|
|
397
|
+
|
|
398
|
+
### StationConnections
|
|
399
|
+
駅の乗換情報を保持するクラス。`StationData.transfer` から参照できます。各フィールドは `MultiLang_Lines` 型(各言語の路線名リスト)で、乗換先がない場合は `None` です。
|
|
400
|
+
|
|
401
|
+
* `train: Optional[MultiLang_Lines]`: JRなど鉄道の乗換情報
|
|
402
|
+
* `subway: Optional[MultiLang_Lines]`: 地下鉄の乗換情報
|
|
403
|
+
* `monorail: Optional[MultiLang_Lines]`: モノレールの乗換情報
|
|
404
|
+
|
|
405
|
+
### TrainType
|
|
406
|
+
列車種別を表す列挙型(`str` のサブクラス)。`.value` で日本語文字列を取得できます。
|
|
407
|
+
|
|
408
|
+
| 定数名 | 値 |
|
|
409
|
+
|:---|:---|
|
|
410
|
+
| `TrainType.LOCAL` | `"普通"` |
|
|
411
|
+
| `TrainType.SEMI_EXP` | `"区間急行"` |
|
|
412
|
+
| `TrainType.SUB_EXP` | `"準急"` |
|
|
413
|
+
| `TrainType.COMMUTER_SUB_EXP` | `"通勤準急"` |
|
|
414
|
+
| `TrainType.EXPRESS` | `"急行"` |
|
|
415
|
+
| `TrainType.COMMUTER_EXP` | `"通勤急行"` |
|
|
416
|
+
| `TrainType.MIDNIGHT_EXP` | `"深夜急行"` |
|
|
417
|
+
| `TrainType.RAPID_EXP` | `"快速急行"` |
|
|
418
|
+
| `TrainType.COMMUTER_RAPID_EXP` | `"通勤快急"` |
|
|
419
|
+
| `TrainType.LTD_EXP` | `"特急"` |
|
|
420
|
+
| `TrainType.LINER` | `"ライナー"` |
|
|
421
|
+
| `TrainType.RAPID_LTD_EXP` | `"快速特急 洛楽"` |
|
|
422
|
+
| `TrainType.EXTRA_TRAIN` | `"臨時列車"` |
|
|
423
|
+
|
|
424
|
+
> **注**: ダイヤ上のデータからライナーを識別することは仕様上困難なため、`TrainData.train_type` では `LINER` ではなく `RAPID_EXP` または `LTD_EXP` と推定されます。`ActiveTrainData` ではAPIから正確な種別が返ります。
|
|
425
|
+
|
|
426
|
+
## 公式APIエンドポイント
|
|
427
|
+
本ライブラリは以下の公開JSONを利用しています。
|
|
428
|
+
- 駅・路線データ: `select_station.json`, `transferGuideInfo.json`
|
|
429
|
+
- リアルタイム位置: `trainPositionList.json`
|
|
430
|
+
- ダイヤ・時刻: `startTimeList.json`
|
|
431
|
+
|
|
432
|
+
## ライセンス
|
|
433
|
+
MIT License
|
|
434
|
+
|
|
435
|
+
Copyright © 2025 dk-butsuri
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
from .tracker import get_khbus_info
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
from typing import List, Optional
|
|
2
|
+
from pydantic import BaseModel, model_validator
|
|
3
|
+
|
|
4
|
+
|
|
5
|
+
class Head(BaseModel):
|
|
6
|
+
errorcode: int
|
|
7
|
+
objid: str
|
|
8
|
+
|
|
9
|
+
class BusStatePrms(BaseModel):
|
|
10
|
+
order: int # 接近順位
|
|
11
|
+
lat: float # 緯度
|
|
12
|
+
lon: float # 経度
|
|
13
|
+
heading: int # 方位
|
|
14
|
+
guid: str # GUID
|
|
15
|
+
stop_id: int # 停留所ID?
|
|
16
|
+
platform_index: int # のりばインデックス?
|
|
17
|
+
route: str # 系統番号(例: [22])
|
|
18
|
+
destination: str # 行先
|
|
19
|
+
via: str # 経由
|
|
20
|
+
status: str # 状態(到着済など)
|
|
21
|
+
timetable: str # 時刻文字列(例: 17:30 到着予定)
|
|
22
|
+
|
|
23
|
+
@classmethod
|
|
24
|
+
def from_string(cls, raw: str):
|
|
25
|
+
parts = raw.split(":")
|
|
26
|
+
return cls(
|
|
27
|
+
order=int(parts[0]),
|
|
28
|
+
lat=float(parts[1]),
|
|
29
|
+
lon=float(parts[2]),
|
|
30
|
+
heading=int(parts[3]),
|
|
31
|
+
guid=parts[4],
|
|
32
|
+
stop_id=int(parts[5]),
|
|
33
|
+
platform_index=int(parts[6]),
|
|
34
|
+
route=parts[7],
|
|
35
|
+
destination=parts[8],
|
|
36
|
+
via=parts[9],
|
|
37
|
+
status=parts[10],
|
|
38
|
+
timetable=parts[11].replace("__________", ":"),
|
|
39
|
+
)
|
|
40
|
+
|
|
41
|
+
class BusState(BaseModel):
|
|
42
|
+
html: Optional[str] = None
|
|
43
|
+
html_sp: Optional[str] = None
|
|
44
|
+
busstateprms: BusStatePrms
|
|
45
|
+
|
|
46
|
+
@model_validator(mode="before")
|
|
47
|
+
def parse_prms(cls, values):
|
|
48
|
+
if isinstance(values, dict) and "busstateprms" in values and isinstance(values["busstateprms"], str):
|
|
49
|
+
values["busstateprms"] = BusStatePrms.from_string(values["busstateprms"])
|
|
50
|
+
return values
|
|
51
|
+
|
|
52
|
+
class Body(BaseModel):
|
|
53
|
+
datetimeStr: str
|
|
54
|
+
busstates: List[BusState]
|
|
55
|
+
|
|
56
|
+
class BusLocationResponse(BaseModel):
|
|
57
|
+
head: Head
|
|
58
|
+
body: Body
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
from httpx import AsyncClient
|
|
2
|
+
import urllib.parse
|
|
3
|
+
from keihan_tracker.bus.schemes import BusLocationResponse
|
|
4
|
+
|
|
5
|
+
UPDATE_URL = "https://busnavi.keihanbus.jp/pc/busstateupd"
|
|
6
|
+
|
|
7
|
+
async def get_khbus_info(stop_name:str, stop_num:int=1):
|
|
8
|
+
async with AsyncClient() as client:
|
|
9
|
+
dgmpl = f"{stop_name}:{stop_num}::"
|
|
10
|
+
result = await client.post(UPDATE_URL,data={"dgmpl":urllib.parse.quote(dgmpl), "sort4":"0"})
|
|
11
|
+
result.raise_for_status()
|
|
12
|
+
res = BusLocationResponse.model_validate_json(result.text)
|
|
13
|
+
return res
|