-
Notifications
You must be signed in to change notification settings - Fork 3
/
WiFi_Wx_Station.ino
231 lines (197 loc) · 8.14 KB
/
WiFi_Wx_Station.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
/****************************************************************
Arduino Weather Station (WiFi)
Sparkfun Wifi shield on an Arduino Uno: Weather station that sends data to wunderground.com
By Dan Fein
Sparkfun WiFi library by Shawn Hymel, https://github.com/sparkfun/SFE_CC3000_Library
Sensors from Adafruit, see their product pages for those libraries.
Notes:
This project has options to conserve battery power. To save power use lpDelay to rest for awhile, wake, send data and rest again.
You will need to sign up for a free account at wunderground.com, to get your pass
When you register a station you will get an ID (If you send data you get ad free membership free)
Sign up at http://www.wunderground.com/personal-weather-station/signup.asp
Wunderground wants UTC Zulu, not local time, if your RTC is local, offset it in code.
Wunderground Upload guidelines: http://wiki.wunderground.com/index.php/PWS_-_Upload_Protocol
This code assumes your wifi credentials are already saved in the CC3000's non-volitile memory
****************************************************************/
int DEBUG = 0; // DEBUG counter; if set to 1, will write values back via serial
#include <Wire.h>
#include <SPI.h>
#include <SFE_CC3000.h> //Wifi
#include <SFE_CC3000_Client.h> //Wifi
#include "DHT.h" //Humidity
#include <Adafruit_BMP085.h> //Barometer
#include "Adafruit_SI1145.h" //UV sensor
//#include "RTClib.h" //Real time clock -- If used
//#include <io.h> //for the sleepies -- Needed if using lpdelay (sleeping when using battery)
// Pins
#define CC3000_INT 2 // Needs to be an interrupt pin (D2/D3)
#define CC3000_EN 7 // Can be any digital pin
#define CC3000_CS 10 // Preferred is pin 10 on Uno
#define DHTPIN 3 // DHT 22 (AM2302)
// The UV sensor and Barometer are on i2C pins
// Constants
#define DHTTYPE DHT22 // DHT 22 (AM2302)
char SERVER[] = "rtupdate.wunderground.com"; // Realtime update server - RapidFire
//char SERVER [] = "weatherstation.wunderground.com"; //standard server
char WEBPAGE [] = "GET /weatherstation/updateweatherstation.php?";
char ID [] = "xxxx";
char PASSWORD [] = "xxxx";
// Global Variables
SFE_CC3000 wifi = SFE_CC3000(CC3000_INT, CC3000_EN, CC3000_CS);
SFE_CC3000_Client client = SFE_CC3000_Client(wifi);
//RTC_DS1307 rtc; // Hardware RTC time -- If used
DHT dht(DHTPIN, DHTTYPE); // DHT 22 (AM2302)
Adafruit_BMP085 bmp; // BMP Pressure Sensor
Adafruit_SI1145 uv = Adafruit_SI1145(); // UV Sensor
unsigned int connections = 0; // number of connections
unsigned int timeout = 30000; // Milliseconds -- 1000 = 1 Second
//Setup runs once at power on or reset
void setup(void){
//Turn everything on
Serial.begin(38400);
Wire.begin(); //For I2C
//rtc.begin(); //Hardware rtc
bmp.begin(); //Pressure sensor
dht.begin(); //Humidity Sensor
// Turn the Wifi on
Serial.print(F("\nInitializing..."));
if ( wifi.init() ) {
Serial.println("WiFi is go");
} else {
Serial.println("Something went wrong during WiFi setup");
}
// Connect to WiFi network stored in non-volatile memory
Serial.println("Connecting to network stored in profile...");
if ( !wifi.fastConnect(timeout) ) {
Serial.println("Error: Could not connect to network");
}
}//End of Setup
//Loop runs continuously
void loop(void){
//Lets see what time the RTC is set at! -- If RTC is used
//DateTime now = rtc.now();
//Get sensor data
float tempc = bmp.readTemperature(); //Can read temp from bmp or dht sensors
float tempf = (tempc * 9.0)/ 5.0 + 32.0; //was dht.readTemperature, need to convert native C to F
float humidity = dht.readHumidity();
float baromin = bmp.readPressure()* 0.0002953;// Calc for converting Pa to inHg (wunderground)
float dewptf = (dewPoint(tempf, dht.readHumidity())); //Dew point calc(wunderground) //replaced dht.readtemp with converted temp
float UVindex = uv.readUV();
// the index is multiplied by 100 so to get the integer index, divide by 100!
UVindex /= 100.0;
if (DEBUG) {
// Debug, or you can sit up all night watching it.
Serial.println("+++++++++++++++++++++++++");
/*
//If you are using Real Time Clock (RTC)
Serial.println("RTC TIME ");
Serial.print("&dateutc=");
Serial.print(now.year());
Serial.print("-");
Serial.print(now.month());
Serial.print("-");
Serial.print(now.day());
Serial.println("+");
Serial.print(now.hour());
Serial.print(":");
Serial.print(now.minute());
Serial.print(":");
Serial.println(now.second());
*/
Serial.print("temp= ");
Serial.print(tempf);
Serial.println(" *F");
Serial.print("baro= ");
Serial.print(baromin);
Serial.println(" inHg");
Serial.print("dew point= ");
Serial.println(dewptf);
Serial.print("humidity= ");
Serial.println(humidity);
Serial.print("UV: ");
Serial.println(UVindex);
}//End debug loop
//Send data to Weather Underground
if (client.connect(SERVER, 80)) {
if (DEBUG) {
Serial.println("Sending DATA ");
}
// Ship it!
client.print(WEBPAGE);
client.print("ID=");
client.print(ID);
client.print("&PASSWORD=");
client.print(PASSWORD);
client.print("&dateutc=");
client.print("now"); //can use instead of RTC if sending in real time
/*
//If you are using Real Time Clock (RTC)
client.print(now.year());
client.print("-");
client.print(now.month());
client.print("-");
client.print(now.day());
client.print("+");
client.print(now.hour()+8);// YOU MUST Add 8 hours -for pacific time- to get back to UTC or Wunderground wont show RAPID FIRE
client.print("%3A");
client.print(now.minute());
client.print("%3A");
client.print(now.second());
*/
client.print("&tempf=");
client.print(tempf);
client.print("&baromin=");
client.print(baromin);
client.print("&dewptf=");
client.print(dewptf);
client.print("&humidity=");
client.print(humidity);
client.print("&uv=");
client.print(UVindex);
//client.print("&action=updateraw");//Standard update
client.print("&softwaretype=Arduino%20UNO%20version1&action=updateraw&realtime=1&rtfreq=2.5");//Rapid Fire
client.print(" HTTP/1.0\r\n");
client.print("Accept: text/html\r\n");
client.print("Host: ");
client.print(SERVER);
client.print("\r\n\r\n");
client.println();
if (DEBUG) {
Serial.println("Upload complete");
}
}//End send loop
else {
if (DEBUG) { Serial.println(F("Connection failed")); }
return;
}
delay(2500); // --If plugged in send every 2.5 seconds
//lpDelay(1200); // Low Power Delay. Value of 4=1sec, 40=10sec, 1200=5min --If battery, send every 5 min.
}//End loop
/****************************************************************
Function Fun Time
****************************************************************/
double dewPoint(double tempf, double humidity)
{
double A0= 373.15/(273.15 + tempf);
double SUM = -7.90298 * (A0-1);
SUM += 5.02808 * log10(A0);
SUM += -1.3816e-7 * (pow(10, (11.344*(1-1/A0)))-1) ;
SUM += 8.1328e-3 * (pow(10,(-3.49149*(A0-1)))-1) ;
SUM += log10(1013.246);
double VP = pow(10, SUM-3) * humidity;
double T = log(VP/0.61078);
return (241.88 * T) / (17.558-T);
}
// Low Power Delay. Drops the system clock to its lowest setting and sleeps for 256*quarterSeconds milliseconds.
// ie: value of 4=1sec 20=5sec
int lpDelay(int quarterSeconds) {
int oldClkPr = CLKPR; // save old system clock prescale
CLKPR = 0x80; // Tell the AtMega we want to change the system clock
CLKPR = 0x08; // 1/256 prescaler = 60KHz for a 16MHz crystal
delay(quarterSeconds); // since the clock is slowed way down, delay(n) now acts like delay(n*256)
CLKPR = 0x80; // Tell the AtMega we want to change the system clock
CLKPR = oldClkPr; // Restore old system clock prescale
}
/****************************************************************
End of Program
****************************************************************/