이번엔 arduino uno를 가지고 좀 복잡한 미니 프로젝트를 했습니다.
ds18b20 온도 센서로부터 값을 읽어오고 동시에 RTC에 저장된 날짜와 시간 데이터도 읽어옵니다.
그리고 나서 두 값을 ESP8266 모듈을 이용하여 MYSQL 서버에 접속하여 값들을 저장합니다.
그리고 마지막으로 웹상에서 그래프를 그려볼것입니다..
mysql에 데이터베이스 및 테이블을 새로 생성합니다.
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 | $ mysql -u root -p Enter password: Welcome to the MySQL monitor. Commands end with ; or \g. Your MySQL connection id is 1576 Server version: 5.5.49-0ubuntu0.14.04.1 (Ubuntu) Copyright (c) 2000, 2016, Oracle and/or its affiliates. All rights reserved. Oracle is a registered trademark of Oracle Corporation and/or its affiliates. Other names may be trademarks of their respective owners. Type 'help;' or '\h' for help. Type '\c' to clear the current input statement. mysql> create database testdb; Query OK, 1 row affected (0.00 sec) mysql> use testdb; Database changed mysql> create table data( -> date date, -> time time, -> temperature double -> ); Query OK, 0 rows affected (0.02 sec) mysql> exit Bye | cs |
/var/www/html 위치에 run.php 파일을 작성합니다. 아래 패스워드입력 부분은 바꾸어 주셔야합니다.
1 2 3 4 5 6 7 8 9 10 11 | <?php $result = mysql_connect("127.0.0.1", "root", "패스워드"); $temperature = $_GET["temp"]; $date = $_GET["date"]; $time = $_GET["time"]; $sqlt = "insert into testdb.data values ($date,$time,$temperature)"; echo "$sqlt"; $ret = mysql_query($sqlt); ?> | cs |
이제 웹브라우저의 주소 창에 아래 주소를 입력해보면 데이터가 테이블에 입력되는 것을 확인 할 수 있습니다.
http://localhost/run.php?temp=26.2&date="2016-06-10"&time="00:52:00"
이번엔 아두이노 코드로 같은 일을 해보았습니다...
센서 및 RTC, ESP8266과 아두이노 연결은 아래 글들을 참고하세요.. 설치 필요한 라이브러리도 나와있습니다.
틀린점은 본 포스팅에서는 DS18B20의 데이터핀이 5번에 연결된다는 점입니다.
또하나 틀린점은 ESP8266을 위한 소프트시리얼로 6번 7번핀이 사용된다는 점입니다.
아래 코드에서는 공유기 접속 정보와 Mysql서버 주소를 변경하셔야 합니다.
[Arduino/Arduino Uno] - Arduino에서 1-wire 온도센서(db18b20)로 온도 측정
[Arduino/Arduino Uno] - arduino UNO에서 DS1302 RTC 모듈 사용하기
[Arduino/Arduino Uno] - arduino에서 esp8266 무선 모듈 테스트
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 367 | #include <Time.h> #include <DS1302RTC.h> #include <OneWire.h> #include <DallasTemperature.h> #include <string.h> //DS18B20 온도 센서의 데이터선인 가운데 핀을 아두이노 5번에 연결합니다. #define ONE_WIRE_BUS 5 //1-wire 디바이스와 통신하기 위한 준비 OneWire oneWire(ONE_WIRE_BUS); // oneWire선언한 것을 sensors 선언시 참조함. DallasTemperature sensors(&oneWire); //다바이스 주소를 저장할 배열 선언 DeviceAddress insideThermometer; // Set pins for DS1302 RTC: CE(reset),IO(dat),CLK DS1302RTC RTC(2, 3, 4); #include <SoftwareSerial.h> #define SSID "apple" //공유기 SSID #define PASS "12345678" //공유기 비번 #define DST_IP "192.168.0.3" //MYSQL 서버 주소 SoftwareSerial dbgSerial(6, 7); // RX, TX 6번,7번핀 void setup() { // Setup Serial connection Serial.begin(9600); //ESP8266 ///////////////////////////////////////////////////////////////////////// Serial.setTimeout(5000); dbgSerial.begin(9600); Serial.println("ESP8266 connect"); boolean connected=false; for(int i=0;i<10;i++) { if(connectWiFi()) { connected = true; break; } } if (!connected){while(1);} delay(5000); dbgSerial.println("AT+CIPMUX=0"); /////////////////////////////////////////////////////////////////////////// //ds18b20 ///////////////////////////////////////////////////////////////////////////////////////////////////////// //1-wire 버스 초기화 sensors.begin(); //발견한 디바이스 갯수 Serial.print("Found "); Serial.print(sensors.getDeviceCount(), DEC); Serial.println(" devices."); // parasite power 모드일 때에는 2핀(GND와 DQ 핀)만 연결하면 됨. Serial.print("Parasite power is: "); if (sensors.isParasitePowerMode()) Serial.println("ON"); else Serial.println("OFF"); //버스에서 첫번째 장치의 주소를 가져온다. if (!sensors.getAddress(insideThermometer, 0)) Serial.println("Unable to find address for Device 0"); //버스에서 발견한 첫번째 장치의 주소 출력 Serial.print("Device 0 Address: "); printAddress(insideThermometer); Serial.println(); //데이터시트에서 확인결과 9~12비트까지 설정 가능 sensors.setResolution(insideThermometer, 10); Serial.print("Device 0 Resolution: "); Serial.print(sensors.getResolution(insideThermometer), DEC); Serial.println(); //////////////////////////////////////////////////////////////////////////////////////////////////////////// //DS1302 RTC /////////////////////////////////////////////////////////////////////////////////////////////////// Serial.println("DS1302RTC Read Test"); Serial.println("-------------------"); // Activate RTC module // digitalWrite(DS1302_GND_PIN, LOW); // pinMode(DS1302_GND_PIN, OUTPUT); // digitalWrite(DS1302_VCC_PIN, HIGH); // pinMode(DS1302_VCC_PIN, OUTPUT); Serial.println("RTC module activated"); Serial.println(); delay(500); if (RTC.haltRTC()) { Serial.println("The DS1302 is stopped. Please run the SetTime"); Serial.println("example to initialize the time and begin running."); Serial.println(); } if (!RTC.writeEN()) { Serial.println("The DS1302 is write protected. This normal."); Serial.println(); } time_t t; tmElements_t tm; delay(5000); //시간을 맞춰야할때 주석을 풀어서 설정합니다.. /* tm.Year = CalendarYrToTm(2016); tm.Month = 6; tm.Day = 9; tm.Hour = 21; tm.Minute = 36; tm.Second = 00; t = makeTime(tm); if(RTC.set(t) == 0) { // Success setTime(t); Serial.print("RTC set to: "); printDateTime(t); Serial.println(""); } else Serial.println("RTC set failed!"); */ ////////////////////////////////////////////////////////////////////////////////////////////// } void loop() { tmElements_t tm; char mdate[255],mtime[255]; //현재 날짜 / 시간을 가져옴 if (! RTC.read(tm)) { sprintf( mdate, "%04d-%02d-%02d", tmYearToCalendar(tm.Year), tm.Month, tm.Day); sprintf(mtime, "%02d:%02d:%02d", tm.Hour, tm.Minute, tm.Second ); } else { Serial.println("DS1302 read error! Please check the circuitry."); Serial.println(); delay(9000); } Serial.print("Requesting temperatures..."); sensors.requestTemperaturesByIndex(0); //첫번째 센서의 온도값 읽어옴 Serial.println("DONE"); //센서에서 읽어온 온도를 출력 float tempC = sensors.getTempC(insideThermometer); Serial.print("Temp C: "); Serial.println(tempC); //이하는 데이터베이스에 넣는 부분 String cmd = "AT+CIPSTART=\"TCP\",\""; cmd += DST_IP; cmd += "\",80"; Serial.println(cmd); dbgSerial.println(cmd); if(dbgSerial.find("Error")) { Serial.println( "TCP connect error" ); return; } char test[20]; String temp(floatToString(test,tempC, 2, 0)); cmd = "GET /run.php?temp='"+temp+"'&date='"+mdate+"'&time='"+mtime+"'\n\r"; dbgSerial.print("AT+CIPSEND="); dbgSerial.println(cmd.length()); Serial.println(cmd); if(dbgSerial.find(">")) { Serial.print(">"); }else { dbgSerial.println("AT+CIPCLOSE"); Serial.println("connect timeout"); delay(1000); return; } Serial.print(cmd); dbgSerial.print(cmd); delay(2000); //Serial.find("+IPD"); while (Serial.available()) { char c = Serial.read(); dbgSerial.write(c); if(c=='\r') dbgSerial.print('\n'); } Serial.println("===="); // Wait one second before repeating :) delay (2000); } //디바이스 주소를 출력하는 함수 void printAddress(DeviceAddress deviceAddress) { for (uint8_t i = 0; i < 8; i++) { if (deviceAddress[i] < 16) Serial.print("0"); Serial.print(deviceAddress[i], HEX); } } void print2digits(int number) { if (number >= 0 && number < 10) Serial.write('0'); Serial.print(number); } //print date and time to Serial void printDateTime(time_t t) { printDate(t); Serial.print(' '); printTime(t); } //print time to Serial void printTime(time_t t) { printI00(hour(t), ':'); printI00(minute(t), ':'); printI00(second(t), ' '); } //print date to Serial void printDate(time_t t) { printI00(day(t), 0); Serial.print(monthShortStr(month(t))); Serial.print(year(t)); } //Print an integer in "00" format (with leading zero), //followed by a delimiter character to Serial. //Input value assumed to be between 0 and 99. void printI00(int val, char delim) { if (val < 10) Serial.print('0'); Serial.print(val); if (delim > 0) Serial.print(delim); return; } char * floatToString(char * outstr, double val, byte precision, byte widthp){ char temp[16]; //increase this if you need more digits than 15 byte i; temp[0]='\0'; outstr[0]='\0'; if(val < 0.0){ strcpy(outstr,"-\0"); //print "-" sign val *= -1; } if( precision == 0) { strcat(outstr, ltoa(round(val),temp,10)); //prints the int part } else { unsigned long frac, mult = 1; byte padding = precision-1; while (precision--) mult *= 10; val += 0.5/(float)mult; // compute rounding factor strcat(outstr, ltoa(floor(val),temp,10)); //prints the integer part without rounding strcat(outstr, ".\0"); // print the decimal point frac = (val - floor(val)) * mult; unsigned long frac1 = frac; while(frac1 /= 10) padding--; while(padding--) strcat(outstr,"0\0"); // print padding zeros strcat(outstr,ltoa(frac,temp,10)); // print fraction part } // generate width space padding if ((widthp != 0)&&(widthp >= strlen(outstr))){ byte J=0; J = widthp - strlen(outstr); for (i=0; i< J; i++) { temp[i] = ' '; } temp[i++] = '\0'; strcat(temp,outstr); strcpy(outstr,temp); } return outstr; } boolean connectWiFi() { //dbgSerial.println("AT+CWMODE=1"); String cmd="AT+CWJAP=\""; cmd+=SSID; cmd+="\",\""; cmd+=PASS; cmd+="\""; dbgSerial.println(cmd); Serial.println(cmd); delay(3000); if(dbgSerial.find("OK")) { Serial.println("OK, Connected to WiFi."); return true; } else { Serial.println("Can not connect to the WiFi."); return false; } } | cs |
수동으로 넣었을때처럼 데이터가 잘들어갑니다..
이제 마지막으로 데이터베이스에 쌓인 데이터를 이용하여 간단한 그래프를 그려볼것입니다.
그래프를 그리기 위해 추가로 설치해야 하는 패키지들입니다.
sudo apt-get install php5-gd php5-cli
/etc/php5/apache2/conf.d/20-gd.ini 파일을 열어서
앞에 세미콜론이 있으면 아래처럼 ; 제거합니다.
; configuration for php GD module
priority=20
extension=gd.so
아파치 서버를 재시작합니다.
sudo /etc/init.d/apache2 restart
http://www.pchart.net/download 에서 라이브러리를 다운로드 받아서 웹서버의 루트 디렉토리로 옮깁니다.
wget http://www.pchart.net/release/pChart2.1.4.tar.gz
tar xvzf pChart2.1.4.tar.gz
sudo mv pChart2.1.4 /var/www/html/pChart
이제 그래프를 그려줄 php를 작성합니다.
sudo nano /var/www/html/graph.php
db접속 패스워드만 바꾸시면 동작합니다.
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 | <?php include("pChart/class/pDraw.class.php"); include("pChart/class/pImage.class.php"); include("pChart/class/pData.class.php"); $db = mysql_connect("127.0.0.1", "root", "패스워드입력"); if ( $db == "" ) { echo " DB Connection error...\r\n"; exit(); } mysql_select_db("testdb",$db); $Requete = "SELECT * FROM `data` order by time desc limit 30"; $result = mysql_query($Requete,$db); while ($row = mysql_fetch_array($result)) { $Temperature[] = $row["temperature"]; $DataTime[] = $row["time"]; } $DataSet = new pData(); $DataSet->addPoints( $Temperature, "Temperature"); $DataSet->addPoints($DataTime,"Timestamp"); $DataSet->setAbscissa("Timestamp"); $DataSet->setXAxisName("Time"); $DataSet->setAxisName(0,"Temperature"); $DataSet->setAxisUnit(0,"°C"); $myPicture = new pImage(1024,600,$DataSet); $myPicture->setGraphArea(100,60,1000,500); $myPicture->setFontProperties(array("FontName"=>"pChart/fonts/verdana.ttf","FontSize"=>11)); $scaleSettings = array("LabelRotation"=>30, "XMargin"=>10,"YMargin"=>10, "Floating"=>TRUE,"GridR"=>200,"GridG"=>200,"GridB"=>200, "DrawSubTicks"=>TRUE,"CycleBackground"=>TRUE ); $myPicture->drawScale($scaleSettings); $myPicture->drawSplineChart(); $myPicture->drawLegend(2000,500,array("Style"=>LEGEND_NOBORDER,"Mode"=>LEGEND_HORIZONTAL)); $myPicture->Stroke(); ?> | cs |
실행해본 결과입니다... 날짜는 따로 출력해주면 될듯합니다.
1초마다 Refresh(새로 고침)을 해주도록 소스코드를 고치면 실시간으로 입력되는 데이터를 연속적으로 볼 수있습니다..
'Arduino Uno' 카테고리의 다른 글
Arduino UNO에서 dht-22 온도 / 습도 센서 사용하기 (0) | 2016.06.21 |
---|---|
arduino uno에 연결된 LCD에 현재 날짜/시간(RTC) 출력하기 (0) | 2016.06.10 |
Arduino Mega 2560에 TFT LCD 연결하기 (0) | 2016.05.30 |
아두이노 예제 - 가변저항 사용하기 (0) | 2016.01.13 |
아두이노에 LM35 온도 센서 연결하여 온도 측정하기 (0) | 2015.12.13 |
시간날때마다 틈틈이 이것저것 해보며 블로그에 글을 남깁니다.
블로그의 문서는 종종 최신 버전으로 업데이트됩니다.
여유 시간이 날때 진행하는 거라 언제 진행될지는 알 수 없습니다.
영화,책, 생각등을 올리는 블로그도 운영하고 있습니다.
https://freewriting2024.tistory.com
제가 쓴 책도 한번 검토해보세요 ^^
그렇게 천천히 걸으면서도 그렇게 빨리 앞으로 나갈 수 있다는 건.
포스팅이 좋았다면 "좋아요❤️" 또는 "구독👍🏻" 해주세요!