-
Notifications
You must be signed in to change notification settings - Fork 211
/
Copy pathFretBoard.ino
250 lines (204 loc) · 6.04 KB
/
FretBoard.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
/*
FretBoard
Retrieve continuous integration server status from the net and visualise status with a
ws2811-based programmable LED array.
For info and circuit diagrams see https://github.com/tardate/LittleArduinoProjects/tree/master/FretBoard
*/
#include <SPI.h>
#include <Ethernet.h>
#include <TextFinder.h>
#include <FastLED.h>
#include <FlexiTimer2.h>
#include "settings.h"
EthernetClient client;
TextFinder finder( client );
CRGB leds[MAX_PROJECTS]; // array of LEDs
byte build_status[MAX_PROJECTS]; // array of build status values.
/*
build status values are recorded as follows:
xx00 - undefined/off
xx01 - 1 sleeping, checking mods or unknown
xx10 - 2 building
00xx - undefined/off
01xx - 4 fail
10xx - 8 success
i.e.:
5 = failed/sleeping
6 = failed/building
9 = success/sleeping or checking mods
10 = success/building
< 5 = undefined
< 8 = failed
*/
volatile byte total_projects = 0; // actual number of active projects
volatile byte status_updating = 0; // set to total_projects after build status update
volatile boolean led_toggle = true;
volatile signed int led_scale = 0;
volatile signed int led_scale_factor = 1;
void setup() {
#ifdef SERIAL_DEBUG
Serial.begin(115200);
#endif
// start the Ethernet connection:
if (Ethernet.begin(mac) == 0) {
#ifdef SERIAL_DEBUG
Serial.println("Failed to configure Ethernet using DHCP");
#endif
// try to congifure using IP address instead of DHCP:
Ethernet.begin(mac, ip);
}
// give the Ethernet shield a second to initialize:
#ifdef SERIAL_DEBUG
Serial.println("firing up the Ethernet...");
#endif
delay(1000);
// you might need to change this based on your LED strip
FastLED.addLeds<WS2811, LED_DATA_PIN, RGB>(leds, MAX_PROJECTS);
for(int iLed = 0; iLed < MAX_PROJECTS; iLed++) {
leds[iLed] = CRGB::Black;
}
FastLED.clear(true);
FastLED.setBrightness(32);
FlexiTimer2::set(10, redrawLedEffects);
FlexiTimer2::start();
}
/*
loop()
runs the main loop to update build status every BUILD_STATUS_POLL_INTERVAL milliseconds or so
*/
void loop()
{
getBuildStatus();
delay(BUILD_STATUS_POLL_INTERVAL);
}
/*
redrawLedEffects()
invoked by timer to update LED display
*/
void redrawLedEffects() {
if(status_updating > 0) {
// cycles a blue LED around all the active projects after an update received
status_updating--;
for(int iLed = 0; iLed < MAX_PROJECTS; iLed++) {
if(iLed==status_updating) {
leds[iLed] = CRGB::Blue;
} else {
leds[iLed] = CRGB::Black;
}
}
FastLED.show(128);
} else {
for(int iLed = 0; iLed < MAX_PROJECTS; iLed++) {
// map the build status to LED color
switch(build_status[iLed]) {
case 5 : // failed/sleeping
leds[iLed] = CRGB::Red;
break;
case 6 : // failed/building
leds[iLed] = led_toggle ? CRGB::OrangeRed : CRGB::Black;
break;
case 9 : // success/sleeping or checking mods
leds[iLed] = CRGB::Green;
break;
case 10 : //success/building
leds[iLed] = led_toggle ? CRGB::PaleGreen : CRGB::Black;
break;
default: // undefined
leds[iLed] = CRGB::Black;
}
}
FastLED.show(led_scale);
led_toggle = !led_toggle;
led_scale += led_scale_factor;
if(led_scale>=128) led_scale_factor = -1;
if(led_scale<=0) led_scale_factor = 1;
}
}
/*
getBuildStatus
sends web request for build status and parses the result.
*/
void getBuildStatus() {
char name[MAX_STATUS_PART_LENGTH];
char build_activity_name[MAX_STATUS_PART_LENGTH];
char build_status_name[MAX_STATUS_PART_LENGTH];
byte build_id = 0;
byte status;
#ifdef SERIAL_DEBUG
String debug = "name:";
#endif
if(sendRequest()) {
if(finder.find("\r\n\r\n")) {
while(finder.getString(",", name, MAX_STATUS_PART_LENGTH) > 0) {
if(finder.getString(",", build_activity_name, MAX_STATUS_PART_LENGTH) > 0) {
if(finder.getString("\n", build_status_name, MAX_STATUS_PART_LENGTH) > 0) {
if(build_id < MAX_PROJECTS) {
// parse the build status value
switch(build_status_name[0]) {
case 'S': // Success
status = 8;
break;
case 'F': // Failure
status = 4;
break;
default:
status = 0;
}
// parse the build activity value
switch(build_activity_name[0]) {
case 'B': // Building
status = status ^ 2;
break;
case 'S': // Sleeping
case 'C': // CheckingModifications
case 'U': // Unknown
status = status ^ 1;
break;
}
build_status[build_id] = status;
#ifdef SERIAL_DEBUG
Serial.println(debug + name + " activity:" + build_activity_name + " status:" + build_status_name + " build_id:" + build_id + " build_status[]:" + build_status[build_id]);
#endif
} else {
#ifdef SERIAL_DEBUG
Serial.println(debug + name + " activity:" + build_activity_name + " status:" + build_status_name + " -- ignore because build_id " + build_id + " exceeds limits");
#endif
}
build_id += 1;
}
}
}
}
// update the total project count
total_projects = build_id;
client.stop();
client.flush();
status_updating = MAX_PROJECTS;
}
}
/*
sendRequest
sends the HTTP request for build status
*/
boolean sendRequest() {
if (client.connect(hostname, 80)) {
#ifdef SERIAL_DEBUG
Serial.println("connected");
#endif
// Make a HTTP request:
client.print("GET ");
client.print(request_path);
client.println(" HTTP/1.0");
client.print("Host: ");
client.println(hostname);
client.println("Connection: close");
client.println();
return true;
}
else {
#ifdef SERIAL_DEBUG
Serial.println("connection failed");
#endif
return false;
}
}