zoukankan      html  css  js  c++  java
  • 使用树莓派制作的远程开门器

    背景:

    话说自从我厂的商务部门搬到旁边的商务楼之后,工程师发现漂亮的前台MM也搬走了,某区只留下冷冷清清的鱼缸。然后,工程师们发现开门成了个问题。鉴于进门需要刷卡,所以没有带卡的工程师就不得不摁下门铃,等待其他人从座位上起身为自己开门。经历过多次不得不让别人来开门以及不得不为别人开门之后,终于有工程师不能忍受了(我就是其中一个),于是,决定自己动手解决问题。要知道,地球上没有能难住工程师的问题!

    研究与选择方案:

    我厂的电子门锁的开门设备是通过一个带弹簧的开关(门内,类似墙面上开灯的开关)控制的,按下开关就能打开门,按下开关的时候能够听到明显的继电器闭合的声音,因此,趁着没人的时候我把开关拆开看了一下,证实的确是通过触电控制的一个继电器。按下开关时,两个金属触电接触,继电器动作,门打开。继电器在动作后延时4秒左右恢复。这样看来,硬件层面的开门实现就非常简单了:跨接一个数字继电器,要开门时,通过电平信号控制继电器闭合并保持2秒即可。

    但要实现远程控制,必须设置一台可以通过TCP或是HTTP协议接受用户命令,并能控制开门硬件(继电器)的设备。最初考虑过Arduino,作为一个开源的硬件平台,Arduino的电平输出非常易于控制和操作。而且Arduino也有RJ45或是无线的接口模块。不过在查看了Arduino的无线模块的操作方式之后,我发现要让Arduino接入一个WPA加密的无线网络并实现一个web server并不容易。所以最终还是选择了树莓派(Raspberry PI),虽然RPI的硬件成本更高一点。

    RPI与Arduino的设计目标不同,它主要是一个超小的PC主机。幸好RPI本身提供了若干个GPIO的输出口(3.3V),用来操作数字继电器倒是没有问题。

    1. RPI的GPIO引脚的定义:http://elinux.org/RPi_Low-level_peripherals#General_Purpose_Input.2FOutput_.28GPIO.29
    2. 引脚的输出规格:http://elinux.org/RPi_BCM2835_GPIOs
    3. RPI的GPIO教程:http://log.liminastudio.com/writing/tutorials/tutorial-how-to-use-your-raspberry-pi-like-an-arduino

    请特别注意:由于GPIO的引脚是没有保护电路的,不当的硬件连接很可能会导致整块RPI板子烧掉,连接和操作时务必慎重。在确定方案时,我个人使用了面包板,电阻和二极管等电子元件搭建实验电路,并反复用万用表验证过输出电压和电流(主要是对RPI的每个脚的输出不确定)。

    操作步骤

    在RPI上安装操作系统

    拿到RPI后,首先要在RPI上安装操作系统。这里有份文档:http://log.liminastudio.com/writing/tutorials/getting-started-with-the-raspberry-pi,安装完操作系统后,确保重新启动成功(RPI的视频输出口是HDMI接口,可以直接连接在支持HDMI的显示器或是电视上,或者可以通过HDMI转VGA/DVI的连接线连接到支持VGA/DVI的显示器上。注意:无源的HDMI转VGA的线可能会损坏RPI板)。

    配置网络

    RPI本身的网络支持已经比较完善,通过有线方式接入的话,只要直接插入网线就可。考虑到无线方式部署比较方便,我采用的是无线连接的方式。

    RPI仅对部分网卡支持较好,利用RPI的官网上的无线网卡兼容信息,最终选择了一款相对便宜且能得到较好支持的网卡:TP-LINK TL-WN823N,该网卡可以从amazon.cn购买,购买地址见本文最后的列表。

    插入网卡后,修改网卡的配置信息:

        sudo vi /etc/network/interfaces
    

    修改其中与wlan0相关的部分:

        auto lo
    
        iface lo inet loopback
        iface eth0 inet dhcp
    
        allow-hotplug wlan0
        iface wlan0 inet static
            wpa-ssid 你要连接的wifi ssid
            wpa-psk 你的wpa连接密码
        address 192.168.0.20    # 设定的静态IP地址
        netmask 255.255.255.0   # 网络掩码
        gateway 192.168.0.1     # 网关
        network 192.168.0.0      # 网络地址
        #wpa-roam /etc/wpa_supplicant/wpa_supplicant.conf
        iface default inet dhcp
    

    这里我选择了使用静态地址,原因是我们不希望开门的服务器地址老是变来边去。设置好网络并确认能够正常工作后,就可以着手进行下一个步骤了。

    连接硬件

    下图是具体的连接方式,注意连在RPI上的三条线,白色(电源)、黑色(GND)、灰色(信号)分别连接在RPI GPIO的2号、6号与7号针脚上。

    编写代码

    RPI的操作系统缺省已经有一个Python的安装了。因此,我们可以使用Python来设置Web Server并控制开门的硬件设备。我选择了web.py作为Web server的开发基础,加上GPIO的python库来完成工作。

    1. Web.py的下载、安装与教程:http://webpy.org/
    2. RPI GPIO的python库安装和使用:http://log.liminastudio.com/writing/tutorials/tutorial-how-to-use-your-raspberry-pi-like-an-arduino

    接下来就是在RPI上设置Web server了。遵循我厂的传统,使用Python作为Web server的主要开发语言。

    开门硬件控制相关代码:

        import RPi.GPIO as GPIO
        import time
    
        PORT = 7
    
        def reset():
            GPIO.setmode(GPIO.BOARD)
            GPIO.setup(PORT, GPIO.IN)
    
        def initcontroller():
            GPIO.setmode(GPIO.BOARD)
            GPIO.setup(PORT, GPIO.OUT)
            GPIO.output(PORT, True)
    
        def opendoor():
            GPIO.setup(PORT, GPIO.OUT)
            GPIO.output(PORT, False)
            time.sleep(2)
            GPIO.output(PORT, True)
    

    Web server代码(基于web.py)

        import web
        from web import form
        import doorop
        import requests
    
        render = web.template.render('templates/')
    
        urls = (
                '/', 'index'
        )
        app = web.application(urls, globals())
    
        myform = form.Form(
            form.Textbox("username"),
            form.Textbox("password", form.notnull)
        )
    
        #initial door controller
        doorop.initcontroller()
    
        def verifyuser(uname, passwd):
            #这里是验证用户的逻辑,我是用公司的LDAP服务器进行验证的
            #验证成功返回True,否则返回False
            pass
    
        class index:
            def GET(self):
                form = myform()
                return render.formtest(form)
    
            def POST(self):
                form = myform()
                if not form.validates():
                    return render.formtest(form)
                else:
                    if(verifyuser(form['username'].value, form['password'].value)):
                        doorop.opendoor()
                        return "door opened!"
                    else:
                        return "Username/Password invalid!"
    
        if __name__ == "__main__":
            web.internalerror = web.debugerror
            app.run()
    

    这样就在RPI上建立了一个web server,用HTTP POST方式输入正确的用户名和口令就能实现开门。

    为了更方便地应用远程开门(例如,用手机开门显然更cool),可以编写iOS或是Android上运行的开门应用。当然,由于开门只需要发送一个HTTP POST请求,直接写成一个可以在Mac/Linux/Win下运行的命令行也可以。

    部署后的全景

    豆瓣开门

    豆瓣开门widget

    后记与感谢

    感谢所有参与这个项目的我厂同学们,豆瓣的确是个有着独特的技术文化的环境,这里不仅有许多可以互相交流想法的技术geek,还有这庞大的对所有新鲜的东西感兴趣的工程师群体,欢迎各位喜欢这个环境的其他非我厂同学们参观和加入:)

    成本与硬件列表(为方便想要动手的同学,带购买链接,点击请慎重)

    1. 树莓派(MODB-512M) 带运费,税费,约300元 http://cn.element14.com/raspberry-pi/raspbrry-modb-512m/%E5%8D%95%E6%9D%BF%E6%9C%BA-%E6%A0%91%E8%8E%93%E6%B4%BE-raspberry-pi-b%E5%9E%8B-512mb/dp/2191863?ref=lookahead

    2. TP-LINK TL-WN823N 300M迷你型无线USB网卡 55元 http://www.amazon.cn/gp/product/B009A2WQTW

    3. 连接线:2元 http://item.taobao.com/item.htm?id=13451559655>

    4. 两路数字继电器 5元 http://item.taobao.com/item.htm?id=13451559655

    5. 其他: 约60元 导线 4G SD卡一张

  • 相关阅读:
    hive笔记:复杂数据类型-array结构
    【数据库】数据库入门(四): SQL查询
    【数据库】数据库入门(三): SQL
    【数据库】数据库入门(二): 关系型数据库
    【数据库】数据库入门(一):基本概念
    mysql 多条数据中,分组获取值最大的数据记录
    log4j.properties配置详解与实例-全部测试通过
    spec开发思路以及理解
    运用 finereport 和 oracle 结合开发报表思路大总结
    Oracle的trim( )、ltrim( )、rtrim( )三个函数的用法及注意事项
  • 原文地址:https://www.cnblogs.com/guanhe/p/2832982.html
Copyright © 2011-2022 走看看