Skip to content

seco/stm32_wifi

Repository files navigation

纯新手,慢慢学。。 代码地址:https://github.com/klren0312/stm32_wifi

使用的器件:

  • STM32F103CBT6 单片机
  • ESP8266-01S wifi模块
  • SHT20温湿度

使用的Node.js服务端:

  • net模块
  • express
  • node-mysql

文章列表:

一、单片机相关代码(只提供网络相关代码)

1.esp8266相关配置代码

1)esp8266.h(宏定义一些AT指令)

\#define AT          "AT\r\n"	//测试esp8266是否工作正常
\#define CWMODE      "AT+CWMODE=3\r\n"		//设置wifi模块的模式,3为STA+AP模式
\#define RST         "AT+RST\r\n"        //重启
\#define CIFSR       "AT+CIFSR\r\n"      //查看ip
\#define CWJAP       "AT+CWJAP=\"nb\",\"123456789\"\r\n"	// 设置要连接的无线路由器的 ssid 和 password
\#define CIPSTART    "AT+CIPSTART=\"TCP\",\"119.29.201.31\",4001\r\n"		//设置服务器IP和端口
\#define CIPMODE0    "AT+CIPMODE=0\r\n"		//非透传模式
\#define CIPMODE1    "AT+CIPMODE=1\r\n"		//透传模式
\#define CIPSEND     "AT+CIPSEND\r\n"        //发送
\#define CIPSTATUS   "AT+CIPSTATUS\r\n"		//网络状态查询

extern void ESP8266_Rst(void);  
extern void ESP8266_Init(void);

2)esp8266.c(两个配置函数)

/**
  * @brief  ESP8266硬件复位
**/
void ESP8266_Rst(void)
{
	GPIO\_InitTypeDef GPIO\_InitStructure;
    //Pb5--对应ESP8266的reset引脚;
    RCC\_APB2PeriphClockCmd(RCC\_APB2Periph\_GPIOB, ENABLE);
    GPIO\_InitStructure.GPIO\_Pin = GPIO\_Pin\_5;
    GPIO\_InitStructure.GPIO\_Speed = GPIO\_Speed\_50MHz;
    GPIO\_InitStructure.GPIO\_Mode = GPIO\_Mode\_Out\_PP;
    GPIO\_Init(GPIOB, &GPIO\_InitStructure);
	GPIO\_ResetBits(GPIOB,GPIO\_Pin\_5);
	mDelay(100);
	GPIO\_SetBits(GPIOB,GPIO\_Pin\_5);
	mDelay(100);
}
/*
 *  @brief ESP8266模块初始化
 */
void ESP8266_Init(void)
{
	ESP8266_Rst();
\#if 1
    SendCmd(AT, "OK", 1000);		  //模块有效性检查
    SendCmd(CWMODE, "OK", 1000);	//模块工作模式
    SendCmd(RST, "OK", 2000);		  //模块重置
    SendCmd(CIFSR, "OK", 1000);		//查询网络信息
    SendCmd(CWJAP, "OK", 2000);		//配置需要连接的WIFI热点SSID和密码
    SendCmd(CIPSTART, "OK", 2000);	//TCP连接
    SendCmd(CIPMODE1, "OK", 1000);	//配置透传模式
	SendCmd(CIPSEND, ">", 1000);	//进入透传模式
	USART2_Clear();
\#endif
}

3)usart2.c中的SendCmd函数(USART2串口发送AT命令使用)

void SendCmd(char* cmd, char* result, int timeOut)
{
    while(1)
    {
        USART2_Clear();
        USART2_Write(USART2, (unsigned char *)cmd, strlen((const char *)cmd));
        mDelay(timeOut);
        printf("%s %d cmd:%s,rsp:%s\n", \_\_func\_\_, \_\_LINE\_\_, cmd, usart2\_rcv\_buf);
        if((NULL != strstr((const char *)usart2\_rcv\_buf, result)))	//判断是否有预期的结果
        {
            break;
        }
        else
        {
            mDelay(100);
        }
    }
}

2.拼接发送字符串代码

1)获取温湿度(sht20代码就不放出来了)

        /* 获取温湿度 */
        SHT2x\_MeasureHM(SHT20\_Measurement\_T\_HM, &temp);
        mDelay(500);
        SHT2x\_MeasureHM(SHT20\_Measurement\_RH\_HM, &humi);
        /* 转化为字符串形式 */
        sprintf(tempStr, "%d", temp);
        sprintf(humiStr, "%d", humi);

2)拼接代码(拼接成json格式)

/**
  * @brief   拼接
  * @param   pkt   缓存指针
  *	@param 	 dsid1  数据流1ID
  *	@param 	 val1   字符串形式的数据点1的值
  *	@param 	 dsid2  数据流2ID
  *	@param 	 val2   字符串形式的数据点2的值
  * @retval  整个包的长度
  */
uint32\_t HTTP\_NewPostPacket(char *pkt, char *dsid1, char *val1, char *dsid2, char *val2)
{
    char dataBuf[50] = {0};
    char lenBuf[10] = {0};
    *pkt = 0;
    sprintf(dataBuf, "{\"%s\":\"%s\",\"%s\":\"%s\"}", dsid1, val1, dsid2, val2);
    sprintf(lenBuf, "%d", strlen(dataBuf));
    strcat(pkt, dataBuf);
    return strlen(pkt);
}

二、Node.js相关代码

1.TCP服务端代码

var net = require('net')
net.createServer(function(socket){
	socket.on('data',function(data){//接受处理
		console.log('got:',data.toString());//打印接收到的数据
		socket.write('I have received') //接受后回复
	});
	socket.write('Ready to receive your message!')//客户端连接成功后回复
}).listen(4001);

三、结果截图

1.服务端打印的信息截图

服务端打印的信息截图

2.客户端串口输出的信息截图

客户端串口输出的信息截图

服务端慢慢完善中,一步一步记录学习 2017.3.17

四、Nodejs TCP 服务端和客户端

用客户端模拟STM32发送过来的字符串。主要是为了解决对字符串的处理。

1.客户端代码

const net = require('net')
const client = net.connect({port:4001},() => {//向服务端发送
    console.log("connected to server!");
    client.write("{\"tem\":\"20\",\"hum\":\"80\"}");
});
client.on('data',(data) => {//从服务端接受
    console.log(data.toString());
    client.end();
});
client.on('end',() => {//断开连接
    console.log('disconnected from server');
})

发送的字符串是json格式:"{"tem":"20","hum":"80"}"

3.服务端代码

var net = require('net')
net.createServer(function(socket){
    socket.on('data',function(data){//接受服务端数据
        console.log('got:',data.toString());
        var text = JSON.parse(data.toString());//将接收到的字符串转换成json对象
        console.log(text);//获取json对象
        console.log(text.tem);//获取tem的值
    });
    socket.on('end',function(data){
        console.log('end');
    });
    socket.write('Ready to receive your message!')
     
}).listen(4001);

缓慢完善中,一步一步记录学习 2017.3.19

五、Nodejs连接数据库,进行增删改查

目前用来解决,获取STM32发来的数据,将其保存到MYSQL中,方便后面实时显示和统计。

1.安装node的mysql模块

文档:https://www.npmjs.com/package/mysql

npm install mysql --save

2.数据库和表的创建

DROP TABLE IF EXISTS `env`;
CREATE TABLE `env` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `tem` int(11) DEFAULT NULL,
  `hum` int(11) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=7 DEFAULT CHARSET=utf8;

3.node_mysql模块的使用

var mysql = require('mysql');
var conn = mysql.createConnection({
    host:'localhost',
    user:'root',
    password:'root',
    database:'nodemysql',
    port:3306
});
conn.connect();
//查询
conn.query('SELECT * from env',function(err,rows,fields){
    if (err) throw err;
});
//增加
var post = {tem:25,hum:50}
conn.query('INSERT INTO env SET ?', post ,function(error,result,fields){
    if(error) throw error;
});
//更改
conn.query('UPDATE env SET tem=? , hum=? WHERE id = ?',[12,66,1], function(error,result,fields){
    if(error) throw error;
});
//删除
conn.query('DELETE FROM env WHERE tem=?',[12],function(error,result,fields){
    if (error) throw error;
    console.log('delete  '+ result.affectedRows + '  rows');
})
conn.end();

4.结果截图

缓慢完善中,今天任务很简单,将服务端接收到的数据存入mysql,就是前面的(二)和(三)融合 2017.3.20

六、TCP服务端接收数据并存入Mysql

1.服务端代码

var net = require('net')
var mysql = require('mysql');
var conn = mysql.createConnection({
    host:'localhost',
    user:'root',
    password:'root',
    database:'nodemysql',
    port:3306
});
conn.connect();
net.createServer(function(socket){
    socket.on('data',function(data){
        console.log('got:',data.toString());
        var text = JSON.parse(data.toString());
        console.log(text);
        console.log(text.tem);
        console.log(text.hum);
        //将温湿度数据存入mysql
        conn.query('INSERT INTO env SET ?', text, function(error,result,fields){
            if (error) throw error;
        });
    });
    socket.on('end',function(data){
        console.log('end');
    });
    socket.write('Ready to receive your message!')
}).listen(4001);

2.结果截图

1)客户端截图 客户端

2)服务端截图

3)数据库截图

3.内网穿透工具ngrok

1)下载地址:https://ngrok.com/download

2)在ngrok官网注册账号获得自己的authtoken

3)将下载的压缩包解压,进入文件夹,打开命令行,运行ngrok.exe authtoken 你的authtoken回车即可

4)运行ngrok http 80 将内网80端口映射出去。

通过单片机发送温湿度到服务端,服务端将数据处理后存入mysql数据库 2017.3.20

七、融合结果图

1.命令行

QQ截图20170320222648.png

2.数据库

QQ截图20170320222703.png

昨天完成了将STM32采集到的温湿度数据传到服务端,存入Mysql数据库的操作。今天做的是通过express连接数据库,提供接口,让前端页面通过fetch获取数据。其中遇到了跨域问题,在express中解决了。 代码地址:https://github.com/klren0312/stm32_wifi 2017.3.21

八、Express服务端与前端获取

1.Express服务端代码

1)连接数据库(前面已经有介绍了)

var mysql = require('mysql');
//数据库配置
var conn = mysql.createConnection({
    host:'localhost',
    user:'root',
    password:'root',
    database:'nodemysql',
    port:3306
});
//连接数据库
conn.connect();

2)Express跨域解决

解决了前端的已拦截跨源请求:同源策略禁止读取位于 http://127.0.0.1:3000/mysql 的远程资源。(原因:CORS 头缺少 'Access-Control-Allow-Origin')。错误

//设置所有路由无限制访问,不需要跨域
app.all('*',function(req,res,next){
    res.header("Access-Control-Allow-Origin","*");
    res.header("Access-Control-Allow-Headers","X-Requested-With");
    res.header("Access-Control-Allow-Methods","PUT,POST,GET,DELETE,OPTIONS");
    res.header("X-Powered-By",'3.2.1');
    res.header("Content-Type","application/json;charset=utf-8");
    next();
})

3)路由配置(提供接口)

两个接口,分别是温度和湿度数据。提供最新的五个数据

//温度
app.get('/tem',function(req,res){
    var tem = [];
    conn.query('SELECT * FROM env',function(err,rows,fields){
        var i  = rows.length;
        var i = rows.length;
        var j =i-1;
        var c= 0;
        while(j>=i-5){
            tem[c] = rows[j].tem;
            c++;
            j--;
        }
        res.send(JSON.stringify(tem));
    })
})
//湿度
app.get('/hum',function(req,res) {
    var hum = [];
    conn.query('SELECT * FROM env',function(err,rows,fields){
        var i = rows.length;
        var j =i-1;
        var c= 0;
        while(j>=i-5){
            hum[c] = rows[j].hum;
            c++;
            j--;
        }
        res.send(JSON.stringify(hum));
    })
     
});

4)express服务器配置

设置端口为3000

//端口:3000
var server = app.listen(3000,function(){
    var host = server.address().address;
    var port = server.address().port;

    console.log(host + "  " + port);
})

2.前端代码

使用fetch取代ajax(fetch与XMLHttpRequest(XHR)类似,fetch()方法允许你发出AJAX请求。区别在于Fetch API使用Promise,因此是一种简洁明了的API,比XMLHttpRequest更加简单易用。)

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>jsonptest</title>
</head>
<body>
    <script>
        function status(response){
            if(response.status>=200 && response.status<300){
                return Promise.resolve(response);
            }
            else{
                return Promise.reject(new Error(response.statusText));
            }
        }
        function json(response){
            return response.json();
        }
        //设置接口地址
        fetch("http://127.0.0.1:3000/tem")
        .then(status)
        .then(json)
        .then(function(data){
            console.log("请求成功,JSON解析后的响应数据为:",data);
        })
        .catch(function(err){
            console.log("Fetch错误:"+err);
        });
    </script>
</body>
</html>

3.结果截图

1)postman请求温度截图 ![tem.png](http://upload-images.jianshu.io/upload_images/2245742-6c28960573cffd58.png?imageMogr2/auto-orient/strip%7CimageView2

2)postman请求湿度截图 hum.png

3)前端fetch获取到的温度数据截图

QQ截图20170321205801.png

继续缓慢学习。昨天前端获取到了数据,今天我们来通过ECharts来显示数据。 代码地址:https://github.com/klren0312/stm32_wifi 2017.3.22

九、ECharts数据显示

1.ECharts基本

1)介绍 百度推出的一款纯 Javascript 的图表库。 官方网址:http://echarts.baidu.com/index.html
2)基本使用

官方网站有文档,我这里放出最基本结构(我也就会这么多。。。)

基本结构.png


2.EChart异步获取数据显示

百度遇到坑,后来发现官方快速入门就有介绍。。。

官方异步说明.png

1)建立图表代码(body里面要建个id为main的div,记得设置长宽) 两个折线图放在一起,分别显示温度和湿度。

//初始化echarts
var myChart = echarts.init(document.getElementById('main'));
//指定图表的配置项和数据
var option ={
    title:{
        text:'温湿度'
    },
    tooltip:{},//提示框
    legend:{//图例组件
        data:['温度','湿度']
    },
    xAxis:{
        data:["1","2","3","4","最新"]
    },
    yAxis:{},
    series: [{
        name: '温度',
        type: 'line',
        data: []
    },{
        name:"湿度",
        type:'line',
        data:[]
    }]
};
//使用设置的配置项和数据显示图表
myChart.setOption(option);

3)fetch获取数据并加载到图表中

//fetch相关函数
function status(response){
    if(response.status>=200 && response.status<300){
        return Promise.resolve(response);
    }
    else{
        return Promise.reject(new Error(response.statusText));
    }
}
function json(response){
    return response.json();
}
获取温度数据
fetch("http://127.0.0.1:3000/tem")
    .then(status)
    .then(json)
    .then(function(data){
        //将数据加载到对应series name的表中
        myChart.setOption({
            series: [{
                // 根据名字对应到相应的系列
                name: '温度',
                data: data
            }]
        });
    })
    .catch(function(err){
        console.log("Fetch错误:"+err);
    });
//获取湿度数据
fetch("http://127.0.0.1:3000/hum")
    .then(status)
    .then(json)
    .then(function(data){
        //将数据加载到对应series name的表中
        myChart.setOption({
             series: [{
                 // 根据名字对应到相应的系列
                 name: '湿度',
                 data: data
             }]
        });
    })
    .catch(function(err){
        console.log("Fetch错误:"+err);
    });

3.结果截图

1)温湿度折线图截图

温湿度图表.png

2)温湿度柱状图截图

温湿度柱状图表.png

今天主要完成的是表的局部刷新,然后就是整体展示。 代码地址:https://github.com/klren0312/stm32_wifi 2017.3.23

十、整合

1.EChart的局部刷新

1)setTimeout 和 setInterval setTimeout 在指定的毫秒数后,将定时任务处理的函数添加到执行队列的队尾(只是延迟执行一次) setInterval 按照指定的周期(毫秒数计时),将定时任务处理函数添加到执行队列的队尾(周期循环执行)

2)局部刷新 刚开始是准备进行全部页面进行刷新,来更新数据(原谅我的无知)

function freshWeb(){
    window.location.reload();
}
setInterval('freshWeb()',3000)

但是感到体验很差,于是我把fetch数据的相关放到一个函数中,然后用setInterval对fetch数据的函数循环执行,就可以达到ECharts数据更新的操作。

function getData(){
    fetch("http://127.0.0.1:3000/tem")
        .then(status)
        .then(json)
        .then(function(data){
            myChart.setOption({
                series: [{
                // 根据名字对应到相应的系列
                 name: '温度',
                 data: data
                }]
            });
        })
        .catch(function(err){
             console.log("Fetch错误:"+err);
        });
}
//循环执行fetch数据获取,达到ECharts数据更新的操作
setInterval('getData()',10);

2.整合实战结果图片

1)实时温湿度web显示 实时温湿度web显示页面

2)服务端数据存储

服务端

3)单片机照片

单片机

刚好手边有块Maibu手表,所以,今天我们来让Maibu也来获取显示单片机的温湿度。 代码地址:https://github.com/klren0312/stm32_wifi 2017.3.24

十一、Maibu显示

1.服务端提供接口

提供“/watch”接口

//手表推送
app.get('/watch',function(req,res){
    var tem = [];
    conn.query('SELECT * FROM env',function(err,rows,fields){
        var tem = "{ \"temhum\" :" + "\"" + rows[rows.length-1].tem + "  |  " 
        + rows[rows.length-1].hum +  "\""  + "}";
        res.send(tem);
    })
})

2.Maibu手表

1)官方网站 http://www.maibu.cc/

2)官方开发者中心 http://dev.maibu.cc/developer/index_login.do

3)开发者文档 http://maibu.cc/api/welcome.html

3.手表代码

写的比较乱,这里只放出请求数据的代码。大家可以去我的github上看完整代码

1)数据请求函数

void data_request_web()
{
    /*拼接url请求地址, 注意url的缓存大小*/
    char url[200] = ""; 
    sprintf(url, "%s", DATA_WEB);       
    /*发送一次*/
    g_comm_id_web = maibu_comm_request_web(url, "temhum", 0);//过滤
}

2)界面初始化函数

P_Window init_btc_window()
{
    static P_Window p_window;
    P_Layer layer_background = NULL;   
    p_window = app_window_create();
    if (NULL == p_window)
    {
        return NULL;
    }
    /*加入你的代码 begin*/    
    /*创建背景图层*/
    layer_background = get_layer();
    /*添加背景图层到窗口*/
    app_window_add_layer(p_window, layer_background);
    /* 添加时间栏 */
    add_time_bar(p_window);
    /*添加数据提示信息*/
    GRect frame_tem = {{0, 30}, {16, 128}};
    add_text_layer(p_window, &g_layer_id_tem, "温度", &frame_tem, GAlignLeft, U_ASCII_ARIAL_16, GColorWhite);
    /*添加数据提示信息*/
    GRect frame_hum = {{0, 30}, {16, 128}};
    add_text_layer(p_window, &g_layer_id_hum, "湿度", &frame_hum, GAlignRight, U_ASCII_ARIAL_16, GColorWhite);
    /*添加数据提示信息*/
    GRect frame_zzes = {{0,110}, {16, 128}};
    add_text_layer(p_window, &g_layer_id_zzes, "治电科技", &frame_zzes, GAlignBottom, U_ASCII_ARIAL_16, GColorWhite);
    /*添加数据*/
    GRect frame_data = {{0,60}, {16, 128}};
    add_text_layer(p_window, &g_layer_id_data, "waiting", &frame_data, GAlignCenter , U_ASCII_ARIAL_42, GColorWhite);
    return p_window;
}

3)更新数据函数

static void web_recv_callback(const uint8_t *buff, uint16_t size)
{
    char stock_gid[10];
    char i;
    maibu_get_json_str(buff, "temhum", stock_gid, 10);//过滤
    for (i=0;i<10;i++)
    {
    if (stock_gid[i]=='}')
        {
            stock_gid[i]=0;
        }
    }
    /*添加数据*/
    GRect frame_data = {{0,60}, {25, 128}};
    add_text_layer(h_window, &g_layer_id_data, stock_gid, &frame_data, GAlignCenter , U_ASCII_ARIAL_20, GColorWhite);
    
    app_window_update(h_window);
}

4.结果展示

1)模拟器 模拟器

2)手表表盘

maibu手表显示

今天完成的是手机端的开发,很简单,DCloud混合开发,用ajax来获取数据,然后EChart显示数据。 代码地址:https://github.com/klren0312/stm32_wifi 2017.3.26

十二、DCloud手机端

1.DCloud

主要用的是他们家的MUI。

1)介绍 官方介绍:最接近原生APP体验的高性能前端框架 网址:http://dev.dcloud.net.cn/mui/

2)开发工具 非常好用的前端开发工具,HBuilder。

QQ截图20170326210431.png

2.相关代码

1)AJAX请求代码 使用的是mui封装好的函数

mui.getJSON('http://10.127.5.188:3000/tem',function(data){
	lineChart.setOption({	 			    	       
	    	series: [{
	   	    // 根据名字对应到相应的系列
	    	    name: '温度',
	     	    data: data
	     	}]
	});
	temChart.setOption({
		series:[{
			data: [{value: data[data.length-1], name: '温度'}]
		}]
	})				
});
	
mui.getJSON('http://10.127.5.188:3000/hum',function(data){
	lineChart.setOption({	 			    	       
	   	series: [{
	   		// 根据名字对应到相应的系列
	   		name: '湿度',
	   		data: data
	   	}]
    });
    					
    humChart.setOption({
    	series:[{
    		data: [{value: data[data.length-1], name: '湿度'}]
    	}]
    })
});		

2)创建子界面

mui.init({
	swipeBack: false,
	statusBarBackground: '#f7f7f7',
	tureConfig: {
		doubletap: true
	},
	subpages: [{
		id: 'list',
		url: 'list.html',
		styles: {
			top: '45px',
			bottom: 0,
			bounce: 'vertical'
		}
	}]
});

3)页面跳转代码

document.getElementById('device1').addEventListener('tap', function() {
  	mui.openWindow({
  		url:'pages/device1.html',
  		id:'pages/device1.html',
  		show:{
  			aniShow:"pop-in"
  	    }
  	})
});

3.结果截图

1)两种打包方式

QQ截图20170326211112.png

2)应用截图

Screenshot_20170326-211147.png

Screenshot_20170326-211156.png

Screenshot_20170326-211738.png

4.总结

DCloud就那样,如果再让我选择一次的话,我选择APPCAN。。。

一些增加和改动。 代码地址:https://github.com/klren0312/stm32_wifi 2017.3.30

十三、第一次增补

增加时间

原本想着单片机传到tcp服务器的时候就获取时间,感觉不会 所有选择在tcp服务器将数据存入数据库的时候,通过nodejs来获取时间,然后一起存到数据库。

1)获取时间

arr.time = new Date().toLocaleString();

2)将时间一起放到json里,然后存到数据库

var text =JSON.parse(data.toString());
var arr = {};
arr.tem = text.temperature;
arr.hum = text.humidity;
arr.indoor = text.Red_Led;
arr.time = new Date().toLocaleString();
arr.x = text.Xg;
arr.y = text.Yg;
arr.z = text.Zg;
conn.query('INSERT INTO pet SET ?', arr, function(error,result,fields){
    if (error) throw error;
});

2.接口服务器处理时间,并提供接口

由于new Date().toLocaleString();提供的时间格式是"2017/3/30 上午10:30:07",而我们只使用后面的具体时间,不用日期。所以我们需要处理一下,在暴露出去。

1)处理时间 使用split函数,分隔符为空格,然后将空格前后的数据存入数组,我们只要下标为1的那个数据即可。

var timeorigin= rows[j].time;
var timeafter= timeorigin.split(" ");

处理时间

2)然后就是提供接口,提供最新的五个

app.get('/time',function(req,res){
    var time = [];
    conn.query('SELECT * FROM pet',function(err,rows,fields){
        var i = rows.length;
        var j = i - 5;
        var c = 0 ;
        while(j<i){
            //事件处理
            var timeorigin= rows[j].time;
            var timeafter= timeorigin.split(" ");
            //处理后的时间存入数组
            time[c] = timeafter[1];
            c++;
            j++;
        }
        res.send(JSON.stringify(time));
    })
})

3.前端获取时间,并显示到折线图

在ECharts折线图的x轴显示时间

1)获取显示代码

fetch("http://127.0.0.1:3000/time")
    .then(status)
    .then(json)
    .then(function(data){
        // 折线图湿度
        myChart.setOption({
            xAxis:{
                data:data
            }
        });
    })
    .catch(function(err){
        console.log("Fetch错误:"+err);
    });

4.结果显示

1)数据库

数据库

2)折线图X轴时间显示

折线图X轴时间显示

Swagger 建立 api文档 代码地址:https://github.com/klren0312/stm32_wifi 2017.4.4

十四、第二次增补

1.Swagger

(1)介绍 The World's Most Popular API Framework(世界最流行的API模板)。

(2)工具

  • Swagger Editor api文档编辑
  • Swagger Codegen 代码生成
  • Swagger UI 显示界面

(3)网址 http://swagger.io/

swagger

2.api文档制作

(1)Swagger Editor编写

  • 1
  • 2
  • 3

(2)Swagger UI界面 1.在express中定义静态文件路由app.use('/static',express.static('public')); 2.将Swagger UI 代码clone到本地 git clone https://github.com/swagger-api/swagger-ui.git 3.在express项目文件夹中,新建一个叫public文件夹,将clone后的代码中的dist文件下所有文件复制到public文件夹。 4.http://localhost:3000/static/index.html访问即可 QQ截图20170404161611.png

3.界面测试

QQ截图20170404161842.png

QQ截图20170404161852.png

数据库没学好,导致这个问题想了两天,最后还是靠google。。。 代码地址:https://github.com/klren0312/stm32_wifi 2017.4.9

十五、数据库功能增加

原来的数据是一直向下存储,导致数据库数据原来越多,我的想法就是只保留最新的五个,其他的都删除。

1.stackoverflow

(1)介绍 是一个与程序相关的IT技术问答网站。用户可以在网站免费提交问题,浏览问题,索引相关内容,在创建主页的时候使用简单的HTML。在问题页面,不会弹出任何广告,销售信息,JavaScript 窗口等。(百度百科)

2.解决方案

QQ截图20170409104041.png

3.测试代码

只保留最后五行数据,也就是最新的五个数据

conn.query('DELETE FROM pet WHERE id NOT IN ( SELECT id FROM ( SELECT id FROM pet ORDER BY id DESC LIMIT 5 ) foo )',function(err,rows,fields){
    if (err) throw err;      
});

4.结果

(1)执行前

QQ截图20170409111504.png

(2)执行后

QQ截图20170409111552.png

(3)由于id我采用的是自动递增,所以删除前面的之后,后面的id不会重新从1开始排序,这个问题不大。

想通过微博来定时发送相关数据(由于没认证,所以接口调用一下就到了限制,明天继续。。。) 代码地址:https://github.com/klren0312/stm32_wifi 2017.4.10

十六、微博通知

1.weibo开发者平台

(1)网址 http://open.weibo.com/ (2)创建应用

QQ截图20170409215555.png

(3)获取App Key 和 App Secret

QQ截图20170409215631.png

(4)了解OAuth2认证流程

oAuth2_01.gif

2.Nodejs的nodeweibo模块

(1)github网址 https://github.com/vczero/node-weibo

pak.png

(2)安装 npm install nodeweibo

3.nodeweibo模块使用

(1)编写setting.json配置文件

QQ截图20170409220201.png

(2)开启微博认证 ,获取 OAuth2需要的code。会弹出授权页面,同意后会在跳转后的页面url上看到code。

var Weibo = require('nodeweibo');
var setting = require('./setting.json');
Weibo.init(setting);
Weibo.authorize();

QQ截图20170409220330.png

(3)接着是通过code获取access_token,注意下面的程序和上面是独立的。access_token每次都是一样的,所以当授权到期,只要进行上面的开启微博认证即可。

var Weibo = require('nodeweibo');
var setting = require('./setting.json');
Weibo.init(setting);
var paras={
    code:"上面程序获得到的code",
    grant_type:"authorization_code"
};
Weibo.OAuth2.access_token(paras,function(data){
    console.log(data);//命令行中会打印出access_token
})

QQ截图20170409220811.png

(4)接下来调用statuses/update接口,来发送微博信息

var Weibo = require('nodeweibo');
var setting = require('./setting.json');
Weibo.init(setting);
// access_token应该是不变的,status里面试要发送的内容
var para = {
    "access_token":"上面获得到的access_token",
    "status":"微博发送的信息"
}
Weibo.Statuses.update(para,function(data){
    console.log(data);
})

QQ截图20170409221049.png

4.nodejs定时发送模块

(1)github网址 https://github.com/node-schedule/node-schedule

(2)安装 npm install node-schedule

(3)时间格式

QQ截图20170409221743.png

5.node-schedule模块使用

(1)每分钟的第一秒进行操作

var rule = new schedule.RecurrenceRule();
rule.second = 2;
var j = schedule.scheduleJob(rule,function(){
    var para = {
        "access_token":"",
        "status":"测试"
    }
    Weibo.Statuses.update(para,function(data){
        console.log(data);
    })  
});

将获取到的温湿度和其他信息通过微博发布 代码地址:https://github.com/klren0312/stm32_wifi 2017.4.12

十七、微博发送信息

1.配置数据库

var mysql = require('mysql');
//数据库配置
var conn = mysql.createConnection({
    host:'localhost',
    user:'root',
    password:'root',
    database:'nodemysql',
    port:3306
});
//连接数据库
conn.connect();

2.微博发送

从数据库中获取到最新的传感器信息,然后通过微博发送

conn.query('SELECT * FROM pet',function(err,rows,fields){
    var indoor;
    if (rows[rows.length-1].indoor == 1) {
        indoor = "on the home";
    }else{
        indoor = "not on the home";
    }
    //组成信息字符串
    var tem = "  temperature:"  + rows[rows.length-1].tem +"'C  ||  Pet:"+indoor 
    +"  ||  humidity:"+ rows[rows.length-1].hum + "%  ||  by ZZES_IOT";
    //微博发送json
    var para = {
        "access_token":"2.00IlvKmF05EX2B7c5e17d246vKldGE",
        "status":tem
    }
    //微博发送
    Weibo.Statuses.update(para,function(data){
        console.log(data);
    })
})

3.现象结果

(1)后台打印的log

QQ截图20170412082224.png

(2)微博效果

QQ截图20170412083821.png

通过蓝牙实现对设备的控制,这样板子的USART1用来电脑串口,USART2用来wifi传输,USART3用来蓝牙控制。之所以选择蓝牙,为了防止远程有人控制设备(主要是wifi控制协议不会写。。。) 2017.4.19

十八、蓝牙控制

1.蓝牙模块

(1)DX-BT05模块配置

AT+NAME zzesiot //设置ble名称
AT+BAUD 9600 //设置波特率为9600
AT+ROLE=0 //设置为从设备
AT+START //设备工作

2.stm32串口配置

USART3的的引脚为PB10-tx,PB11-rx

(1)配置USART3

void Usart3_Init(unsigned int baud)
{
	GPIO_InitTypeDef gpioInitStruct;
	USART_InitTypeDef usartInitStruct;
	NVIC_InitTypeDef nvicInitStruct;
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART3,ENABLE);
	//PB10 TXD
	gpioInitStruct.GPIO_Mode  = GPIO_Mode_AF_PP;
	gpioInitStruct.GPIO_Pin = GPIO_Pin_10;
	gpioInitStruct.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOB,&gpioInitStruct);
	//PB11 RXD
	gpioInitStruct.GPIO_Mode = GPIO_Mode_IN_FLOATING;
	gpioInitStruct.GPIO_Pin = GPIO_Pin_11;
	gpioInitStruct.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOB,&gpioInitStruct);
	//串口参数
	usartInitStruct.USART_BaudRate = baud;
	usartInitStruct.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
	usartInitStruct.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
	usartInitStruct.USART_Parity = USART_Parity_No;
	usartInitStruct.USART_StopBits = USART_StopBits_1;
	usartInitStruct.USART_WordLength = USART_WordLength_8b;
	USART_Init(USART3,&usartInitStruct);
	USART_Cmd(USART3,ENABLE);
	USART_ITConfig(USART3,USART_IT_RXNE,ENABLE);
	//中断
	nvicInitStruct.NVIC_IRQChannel = USART3_IRQn;
	nvicInitStruct.NVIC_IRQChannelCmd = ENABLE;
	nvicInitStruct.NVIC_IRQChannelPreemptionPriority = 0;
	nvicInitStruct.NVIC_IRQChannelSubPriority = 0;
	NVIC_Init(&nvicInitStruct);
}

(2)USART3中断函数

void USART3_IRQHandler(void)
{
	if(USART_GetITStatus(USART3, USART_IT_RXNE) != RESET)
	{
		if(usart3Len >= 64)	
			usart3Len = 0;
		usart3Buf[usart3Len++] = USART3->DR;
		USART_ClearFlag(USART3, USART_FLAG_RXNE);
	}
}

3.主函数

(1)初始化usart3,设置波特率为9600

Usart3_Init(9600);	

(2)蓝牙接受命令判断(led配置相关代码就没放上来了)

while(1){
	if(usart3Len > 0){
		if(strcmp(usart3Buf, "666") == 0){
			UsartPrintf(USART3, "发送的命令为:\r\n%s\r\n", usart3Buf);
				ledOn(LED1);//led1亮
			}
		else if(strcmp(usart3Buf, "233") == 0){
			UsartPrintf(USART3, "发送的命令为:\r\n%s\r\n", usart3Buf);
			ledOff(LED1);//led1灭
		}
		memset(usart3Buf, 0, sizeof(usart3Buf));
		usart3Len = 0;
	}
	DelayXms(100);
}

4.结果

(1)手机端发送和接受蓝牙数据截图

使用的是在app商城下的蓝牙串口助手,自己的蓝牙串口助手正在开发。。。

手机端发送和接受蓝牙数据

(2)单片机现象

  • 当发送666时,led1亮 led1亮
  • 当发送233时,led1灭 led1灭

就是用小娜打开软件,然后软件自动操作。 感谢kelannan 2017.4.21

十九、Cortana控制

1.Cortana介绍

微软发布的全球第一款个人智能助理

2.控制原理

1)通过小娜打开自己写的软件,软件通过串口发送相关命令,控制单片机。 2)当然小娜打开软件可以通过手打和语音。

3.Arduino程序编写

纯是为了方便,你也可以用stm32,stm32串口相关程序和我前面的蓝牙的串口代码是一样的。 串口接收到“F”则开灯,接收到“T”则关灯

char Val;   
int Pin1 = 13;
void setup() {
Serial.begin(9600);  
    pinMode(Pin1, OUTPUT);
 digitalWrite(Pin1, LOW);
}
void loop() {
 if(Serial.available()>0){ 
    Val=Serial.read(); //read it
    Serial.print(Val);
  }  
 if (Val=='F'){
    digitalWrite(Pin1, HIGH);//开灯
 }
 if(Val=='T'){
  digitalWrite(Pin1,LOW);  //关灯
 }
delay(10);
}

4.C#程序编写

(1)在Visual Studio中新建C#命令行程序

新建C#命令行程序

(2)引入串口相关的包 using System.IO.Ports;

(3)主程序

向指定串口发送相关字符命令

  • 开灯程序
namespace CortanaControl
{
    class Program
    {
        static void Main(string[] args)
        {
            SerialPort port = new SerialPort("COM3", 9600, Parity.None, 8, StopBits.One);
            port.Open();
            port.Write("F");
            port.Close();
        }
    }
}
  • 关灯程序
namespace CortanaControl
{
    class Program
    {
        static void Main(string[] args)
        {
            SerialPort port = new SerialPort("COM3", 9600, Parity.None, 8, StopBits.One);
            port.Open();
            port.Write("T");
            port.Close();
        }
    }
}

(4)将运行生成的程序分别改名放到一起

生成的程序

5.让Cortana能搜索到我们的程序

将程序的快捷方式放到开始菜单的文件夹内,当然不是那么简单的放。

(1)找到文件夹 这是我的文件夹位置 C:\Users\zzes\AppData\Roaming\Microsoft\Windows\Start Menu\Programs\ zzes是我的用户名,其他应该都是一样的

(2)到上面的位置后,可以新建一个文件夹,名称随便取,主要是把程序放里面不乱。 比如我建的文件夹IOT

新建一个文件夹

(3)进入IOT文件夹,在里面新建快捷方式

新建快捷方式

(4)在对象位置的框中写入

%SystemRoot%\system32\cmd.exe /C start "" "D:\open.exe"

对象位置

(5)在快捷方式名称中写你要在Cortana中查找的名称 比如开灯

4.png

(6)关灯也是一样的步骤

5.png

(7)小娜搜索测试

QQ截图20170420225338.png

6.结果

http://v.youku.com/v_show/id_XMjcyMDA5MDMwNA==.html

在本机windows下运行Nodejs时,基本命令行黑框开着就行了。但是要把要把nodejs程序部署到自己的远程centos服务器,要用到Nodejs的PM2模块让Nodejs程序在后台运行。 2017.4.24

二十、Nodejs_PM2

1.PM2模块

(1)介绍 Advanced, production process manager for Node.js (2)官网 http://pm2.keymetrics.io/

QQ截图20170424232316.png

2.PM2使用

(1)全局安装

npm install pm2 -g

(2)开启PM2

pm2 start tcpiot.js

QQ截图20170424231755.png

QQ截图20170424231815.png

(3)PM2监视器

pm2 monit

QQ截图20170424231946.png

(4)列出PM2后台运行的Nodejs程序

pm2 list

list.png

(5)关闭PM2后台运行的tcpiot.js

pm2 stop tcpiot.js

stop.png

(6)重启PM2后台关闭的tcpiot.js

pm2 restart tcpiot.js

restart.png

(7)删除PM2后台的tcpiot.js

pm2 delete tcpiot.js

delete.png

3.结果

可以让Nodejs程序在后台运行,不会随着命令行关闭而关闭。

实现对于一些指定数据超过预期的时候,会发送邮件报警 2017.4.29

二十一、邮件通知

1.nodejs邮件模块

(1)emailjs模块 send emails, html and attachments (files, streams and strings) from node.js to any smtp server (2)github网址 https://github.com/eleith/emailjs (3)安装

npm install emailjs --save

2.邮件发送

定时使用前面说过的node-schedule模块

(1)引入相关模块

var email = require("emailjs");//发送邮件模块
var mysql = require('mysql');//mysql模块
var schedule = require('node-schedule');//定时模块

(2)配置数据库

//数据库配置
var conn = mysql.createConnection({
    host:'localhost',
    user:'root',
    password:'root',
    database:'nodemysql',
    port:3306
})
//连接数据库
conn.connect();

(3)发送邮箱的配置

//配置邮箱
var server = email.server.connect({
    user: "[email protected]",//邮箱用户名
    password:"xxxxxx",//密码
    host:"smtp服务器的地址",//查看使用的邮箱的smtp服务器地址
    ssl:true
});

(4)定时发送

//每分钟第十秒的时候检测
var rule = new schedule.RecurrenceRule();
rule.second = 10;//定义为每分钟第十秒
var j = schedule.scheduleJob(rule,function(){
    //数据库查询
    conn.query('SELECT * FROM pet',function(err,rows,fields){
        //例如获取温度的值,存入tem
        var tem = rows[rows.length-1].tem;
        //定义邮件内容
        var temmsg = {
            text:"tempreture is " + tem + ",please  be careful",//邮件内容
            from: "[email protected]",//发送方
            to:"[email protected]",//接收方
            subject:"PetHose tem"
        };
        //判断如果温度大于或等于30度,就发送邮件通知
        if(tem>=30){
            //邮件发送
            server.send(temmsg,function(err,message){
                console.log(err || "ok");
            });
        }
    });
});

3.结果

(1)当温度大于或等于三十度的时候,发送邮件 接收到的报警邮件

网络控制,就是通过前端页面点击控制按钮,然后通过ajax来出发后端函数将命令存入数据库,然后后端TCP服务端将命令从数据库取出,发送给单片机。当然只是简单的一个命令实现。 2017.5.1

二十二、网络控制

1.建立一个存储命令的表

(1)比如staus表

数据库.jpg

2.编写express接口

提供给前端按钮通过AJAX触发

(1)当触发/buttonclick1,则将命令1存入数据库; 当触发/buttonclick0,则将命令0存入数据库。

//按钮api
app.get('/buttonclick1',function(req,res){
    //增加
    var post = {status:1};
    conn.query('INSERT INTO status SET ?', post ,function(error,result,fields){
        if(error) throw error;
    });
    res.send("1");
    res.end();// 如果不执行end(), 那么前端网页则会一直等待response
})
app.get('/buttonclick0',function(req,res){
    //增加
    var post = {status:0};
    conn.query('INSERT INTO status SET ?', post ,function(error,result,fields){
        if(error) throw error;
    });
    res.send("0");
    res.end();// 如果不执行end(), 那么前端网页则会一直等待response
})

3.编写前端按钮

(1)HTML

就两个button,一个开,一个关

<button onclick="buttonclk1()">开启</button> 
<button onclick="buttonclk0()">关闭</button> 

(2)javascript

ajax触发,使用fetch

function status(response){
     if(response.status>=200 && response.status<300){
        return Promise.resolve(response);
     }else{
        return Promise.reject(new Error(response.statusText));
     }
}
function json(response){
    return response.json();
}
//开启按钮触发函数
function buttonclk1(){
    fetch("http://127.0.0.1:3000/buttonclick1")//后端提供的buttonclick1接口
        .then(status)
        .then(json)
        .then(function(data){
           console.log("请求成功,JSON解析后的响应数据为:",data);
        })
        .catch(function(err){
            console.log("Fetch错误:"+err);
    });
}   
//关闭按钮触发函数
function buttonclk0(){
    fetch("http://127.0.0.1:3000/buttonclick0")///后端提供的buttonclick0接口
        .then(status)
        .then(json)
        .then(function(data){               console.log("请求成功,JSON解析后的响应数据为:",data);
    })
    .catch(function(err){
        console.log("Fetch错误:"+err);
    });
}

(3)点击触发结果

前端.jpg

4.TCP服务端

在和单片机建立通信后,就会一直从数据库获取最新的命令,然后发送给单片机。

(1)代码

将其写入socket.on内

conn.query('SELECT * FROM status',function(err,rows,fields){
    if (err) throw err;
    socket.write(rows[rows.length-1].status+"");//必须要字符串才能发,用toString()报错,有人知道为啥吗?
})          

5.STM32代码

当然还是基于onenet的代码,感谢中移物联,代码写的很易读!具体只放处理数据的代码。

(1)处理数据,控制LED的开关

void Net_Event(unsigned char *dataPtr){
    if(strstr((char *)dataPtr, "1"))
    {
        UsartPrintf(USART_DEBUG, "¿ª\r\n");
        Led4_Set(LED_ON);
    }
    else if(strstr((char *)dataPtr,"0")){
        UsartPrintf(USART_DEBUG,"¹Ø\r\n");
        Led4_Set(LED_OFF);
    }
}

6.串口截图

串口显示.jpg

About

stm32 + esp8266 与 nodejs 服务端通信

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published