-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathesp32-mining-power-controller.ino
366 lines (303 loc) · 10.7 KB
/
esp32-mining-power-controller.ino
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
/*
BH1750FVI と ESP32 の接続:
VCC <-> 3V3
GND <-> GND
SDA <-> GPIO21/SDA
SCL <-> GPIO22/SCL
*/
#include <inttypes.h>
#include <WiFi.h>
#include <BH1750.h>
#include <Ambient.h>
#include <HTTPClient.h>
#include <ArduinoJson.h>
/* Settings -------------------------------- */
// 休日にマイニングするかどうか(電気代が休日割引の場合にtrue)
const boolean miningAtHoliday = true;
// 夜間にマイニングするかどうか(電気代が夜間割引の場合にtrue)
const boolean miningAtNight = true;
// 夜間の時間帯
const int nightTimeStart = 21;
const int nightTimeEnd = 9;
// OCプロファイルの名前
const String HIGH_OC_PROFILE_NAME = "HIGH";
const String LOW_OC_PROFILE_NAME = "LOW";
// Power Limit制御の閾値とヒステリシス (Lux)
// (照度に応じたPower Limit制御をしたくない場合は、ヒステリシスに高い数値(32767など)を設定してください)
const int PL_THRETHOLD = 2800;
const int PL_HYSTERESIS = 300;
// PC電源制御の閾値とヒステリシス (Lux)
// (照度に応じたPC電源ON/OFF制御をしたくない場合は、ヒステリシスに高い数値(32768など)を設定してください)
const int PC_POWER_THRETHOLD = 1800;
const int PC_POWER_HYSTERESIS = 300;
// Ambientのチャネル接続設定
const int AMBIENT_CHANNEL_ID = 99999;
const char AMBIENT_WRITE_KEY[] = "****************";
// WIFIの接続設定
const char WIFI_SSID[] = "************";
const char WIFI_PASSWORD[] = "**********";
const IPAddress WIFI_IP(192, 168, 0, 10);
const IPAddress WIFI_GATEWAY(192, 168, 0, 254);
const IPAddress WIFI_SUBNET(255, 255, 255, 0);
const IPAddress WIFI_DNS(8, 8, 8, 8);
// Hive APIの接続設定 (https://hiveon.com/forum/t/hive-api-v2/4490)
// [farm-id]: ファームのID
// [worker-id]: ワーカーのID
// [personal-token]: パーソナルトークン(アカウント設定から生成可能)
const char GET_OC_PROFILES_URL[] = "https://api2.hiveos.farm/api/v2/farms/[farm-id]/oc";
const char SET_WORKER_OC_URL[] = "https://api2.hiveos.farm/api/v2/farms/[farm-id]/workers/[worker-id]";
const char PERSONAL_TOKEN[] = "Bearer [personal-token]";
// 休日判定API (https://s-proj.com/utils/holiday.html)
const char GHECK_HOLIDAY_URL[] = "https://s-proj.com/utils/checkHoliday.php?date=%04d%02d%02d";
/* ---------------------------------------- */
// PC電源状態の入力ピン
const int PC_POWER_STATUS_PIN = 18;
// PC電源スイッチの出力ピン
const int PC_POWER_SW_PIN = 19;
// Hive OSのOCプロファイル情報
struct OcProfile {
int id;
String name;
int ambientColor;
};
OcProfile UNKNOWN_OC = { 0, "NONE", 0 };
// WIFIクライアント
WiFiClient client;
// BH1750FVI照度センサー
BH1750 lightSensor;
// Ambientクラウドサービス
Ambient ambient;
// HTTPクライアント
HTTPClient httpClient;
// OCプロファイル
OcProfile lowOcProfile;
OcProfile highOcProfile;
// 現在のOCプロファイル
OcProfile* currentOcProfile = &UNKNOWN_OC;
// 時刻と休日フラグ
tm now;
bool holiday;
// 照度の3分間積算値
float threeMinDataSummary = 0.0;
int threeMinDataCount = 0;
// 照度の15分間積算値
float fifteenMinDataSummary = 0.0;
int fifteenMinDataCount = 0;
/**
* 初期化処理
*/
void setup() {
// シリアル初期化
Serial.begin(9600);
while (!Serial);
Serial.println("Start");
// GPIO初期化
pinMode(PC_POWER_STATUS_PIN, INPUT);
gpio_set_pull_mode(GPIO_NUM_18, GPIO_PULLDOWN_ONLY);
pinMode(PC_POWER_SW_PIN, OUTPUT);
// 照度センサー初期化
Wire.begin();
lightSensor.begin(BH1750::ONE_TIME_HIGH_RES_MODE);
// Wifi接続
WiFi.config(WIFI_IP, WIFI_GATEWAY, WIFI_SUBNET, WIFI_DNS);
WiFi.begin(WIFI_SSID, WIFI_PASSWORD);
Serial.print("WiFi connecting");
while (WiFi.status() != WL_CONNECTED) {
Serial.print(".");
delay(100);
}
Serial.println(" connected");
// 時刻同期(NTP)
configTime(9 * 3600L, 0, "ntp.nict.jp", "time.google.com", "ntp.jst.mfeed.ad.jp");
// Ambient初期化
ambient.begin(AMBIENT_CHANNEL_ID, AMBIENT_WRITE_KEY, &client);
// OCプロファイル取得
getOcProfiles();
Serial.print("LOW OC Profile: id=");
Serial.print(lowOcProfile.id);
Serial.print(", name=");
Serial.println(lowOcProfile.name);
Serial.print("HIGH OC Profile: id=");
Serial.print(highOcProfile.id);
Serial.print(", name=");
Serial.println(highOcProfile.name);
delay(1000);
}
/**
* 3秒毎のループ処理
*/
void loop() {
// 現在時刻取得
int lastYear = now.tm_year;
int lastMonth = now.tm_mon;
int lastDay = now.tm_mday;
getLocalTime(&now);
if (now.tm_year != lastYear || now.tm_mon != lastMonth || now.tm_mday != lastDay) {
// 日付が変わったら、休日判定を行う
holiday = isHoliday(&now);
Serial.print("Date: ");
Serial.print(now.tm_year + 1900);
Serial.print("/");
Serial.print(now.tm_mon + 1);
Serial.print("/");
Serial.print(now.tm_mday);
Serial.println(holiday ? " holiday" : " weekday");
}
// 夜間判定
bool nightTime = (now.tm_hour >= nightTimeStart || now.tm_hour <= nightTimeEnd);
// 照度計測
lightSensor.configure(BH1750::ONE_TIME_HIGH_RES_MODE);
while (!lightSensor.measurementReady(true)) {
delay(10);
}
float lux = lightSensor.readLightLevel();
threeMinDataSummary += lux;
threeMinDataCount++;
// PC電源状態取得
int pcPowerStatus = digitalRead(PC_POWER_STATUS_PIN);
Serial.print("current: lux=");
Serial.print(lux);
Serial.print(", pc_power=");
Serial.print(pcPowerStatus == HIGH ? "ON" : "OFF");
Serial.print(", holiday=");
Serial.print(holiday ? "true": "false");
Serial.print(", nightTime=");
Serial.println(nightTime ? "true": "false");
// 60ループ毎(約3分毎)にAmbient送信
if (threeMinDataCount >= 60) {
float average = threeMinDataSummary / threeMinDataCount;
sendAmbient(average, currentOcProfile, pcPowerStatus);
fifteenMinDataSummary += average;
fifteenMinDataCount++;
threeMinDataSummary = 0.0;
threeMinDataCount = 0;
}
// 300ループ毎(約15分毎)にPC電源制御とPower Limit制御
if (fifteenMinDataCount >= 5) {
bool holidayMining = miningAtHoliday && holiday;
bool nightMining = miningAtNight && nightTime;
float average = fifteenMinDataSummary / fifteenMinDataCount;
// 照度が(閾値+ヒステリシス)を上回っていたらPower Limit上げ
if (average > (PL_THRETHOLD + PL_HYSTERESIS) && highOcProfile.name.compareTo(currentOcProfile->name) != 0) {
changeOcProfile(&highOcProfile);
// 照度が(閾値-ヒステリシス)を下回っていたらPower Limit下げ
} else if (average < (PL_THRETHOLD - PL_HYSTERESIS) && lowOcProfile.name.compareTo(currentOcProfile->name) != 0) {
changeOcProfile(&lowOcProfile);
}
// 照度が(閾値+ヒステリシス)を上回っているか、休日/夜間マイニング有効ならPC電源オン
if ((average > (PC_POWER_THRETHOLD + PC_POWER_HYSTERESIS) || holidayMining || nightMining)) {
if (pcPowerStatus == LOW) {
Serial.println("PC power on");
pushPcPowerButton();
}
// 照度が(閾値-ヒステリシス)を下回っていたらPC電源オフ
} else if (average < (PC_POWER_THRETHOLD - PC_POWER_HYSTERESIS)) {
if (pcPowerStatus == HIGH) {
Serial.println("PC power off");
pushPcPowerButton();
}
}
fifteenMinDataSummary = 0.0;
fifteenMinDataCount = 0;
}
delay(3000);
}
/**
* 休日判定
*/
bool isHoliday(tm* timeInfo) {
char url[128];
sprintf(url, GHECK_HOLIDAY_URL, timeInfo->tm_year + 1900, timeInfo->tm_mon + 1, timeInfo->tm_mday);
Serial.print("request: [GET] ");
Serial.println(url);
bool result = false;
httpClient.begin(url);
int resCode = httpClient.GET();
if (resCode == HTTP_CODE_OK) {
String resData = httpClient.getString();
result = resData.compareTo("holiday") == 0;
} else {
Serial.print("get holiday error: ");
Serial.println(resCode);
}
httpClient.end();
return result;
}
/**
* Ambientへのデータ送信
*/
void sendAmbient(double lux, OcProfile* oc, int pcPowerStatus) {
Serial.print("send ambient: lux=");
Serial.print(lux);
Serial.print(", oc=");
Serial.print(oc->name);
Serial.print(", pc_power=");
Serial.println(pcPowerStatus == HIGH ? "ON" : "OFF");
ambient.set(1, lux);
ambient.set(2, currentOcProfile->ambientColor);
ambient.set(3, pcPowerStatus == HIGH ? 1 : 0);
ambient.send();
}
/**
* HIGH/LOWのOCプロファイルを取得
*/
void getOcProfiles() {
Serial.print("request: [GET] ");
Serial.println(GET_OC_PROFILES_URL);
httpClient.begin(GET_OC_PROFILES_URL);
httpClient.addHeader("Authorization", PERSONAL_TOKEN);
int resCode = httpClient.GET();
if (resCode == HTTP_CODE_OK) {
String resData = httpClient.getString();
Serial.print("response: ");
Serial.println(resData);
DynamicJsonDocument jsonDoc(4096);
deserializeJson(jsonDoc, resData);
JsonArray json = jsonDoc["data"].as<JsonArray>();
for (JsonObject data : json) {
String name = data["name"].as<String>();
if (HIGH_OC_PROFILE_NAME.compareTo(name) == 0) {
highOcProfile = { data["id"].as<int>(), name, 9 }; // 9:赤色
} else if (LOW_OC_PROFILE_NAME.compareTo(name) == 0) {
lowOcProfile = { data["id"].as<int>(), name, 12 }; // 12:緑色
}
}
} else {
Serial.print("get oc profiles error: ");
Serial.println(resCode);
}
httpClient.end();
}
/**
* OCプロファイルを変更
*/
void changeOcProfile(OcProfile* ocProfile) {
Serial.print("request: [PATCH] ");
Serial.println(GET_OC_PROFILES_URL);
char json[50];
sprintf(json, "{\"oc_id\":%d,\"oc_apply_mode\":\"replace\"}", ocProfile->id);
Serial.print("payload: ");
Serial.println(json);
httpClient.begin(SET_WORKER_OC_URL);
httpClient.addHeader("Content-Type", "application/json");
httpClient.addHeader("Authorization", PERSONAL_TOKEN);
int resCode = httpClient.PATCH((uint8_t*) json, strlen(json));
if (resCode == HTTP_CODE_OK) {
Serial.print("change oc profile to ");
Serial.println(ocProfile->name);
} else {
Serial.print("changeOcprofile error: ");
Serial.println(resCode);
}
httpClient.end();
currentOcProfile = ocProfile;
}
/**
* PC電源ON/OFF
*/
void pushPcPowerButton() {
// 0.3秒間電源スイッチを押す
digitalWrite(PC_POWER_SW_PIN, HIGH);
delay(300);
digitalWrite(PC_POWER_SW_PIN, LOW);
}