我的小工具,用C和python实现远程读卡器,远程读写消费卡片

简介: 我的小工具,用C和python实现远程读卡器,远程读写消费卡片

这个远程读卡器就是一普通usb口或串口的读卡器,只不过配合一个电脑软件作为tcp服务器。这样,程序员可以在公司电脑上运行程序连到服务器上。服务器端操作控制现场的读卡器。

晚上睡不着,想着白天工作上的事,让现场同事配合抓包,读取一下卡上数据,但现场同事连串口助手都没听说过,指望不上了。想到可以做一个远程读卡器。


对调试现场读卡之类的程序太有用了。想抓包现场卡上数据,很easy了。封装一些读卡接口。甚至可以实现远程仿真调试程序读卡消费逻辑 。或者实现一完全软件模拟的pos机。


这想法可行吧,我抽空打算实现一个。至少对我的工作很有帮助吧。目前已实现了部分电脑程序模拟车载pos机通信。可以模拟pos机与采集前置服务通信,下载票价,传消费记录。后续加进去读卡功能。跟采集前置服务调通信我一般电脑上先调通,再移植到车载机上,效率提高不少。省得找机器,找流量卡,开外网端口映射,接仿真器。现场再也不用担心车载机记录全采出来咋入库了。用这个模拟程序直接解析模拟车载机通信就传上去。


说做就做。由于我是终端组的,不像平台他们有现成的读卡器。于是想到用车载POS机的读卡模块配合串口做一个。作为读卡器用。服务器端考虑用python,有很多模块可以用,实现起来快。


一、功能简介:


本工具实现一个在电脑上运行的模拟车载机终端。只是功能上的模拟,并不是与车载机完全一样。用现场的B502机器作为模拟终端的读卡模块,在公司远程读、写、消费现场的M1卡,CPU卡,并按照车载机中的记录格式存储记录到电脑上,然后通过与采集前置通信把记录传给采集前置服务。全采的车载机记录也可以拿过来用,通过此工具传给采集前置。


实现了供本地电脑调用的读卡库,通信库和记录存储库,所有库的接口均与嵌入式终端程序中的文件库,卡库,通信库接口一致。此工具只是利用卡库,通信库和文件库的一个例子。若需要在某个具体项目中读写卡,抓包,测卡,消费,上传记录等,需根据提供的接口单独做自己的应用。


二、实现原理:


1.移植嵌入式程序中的文件库到电脑上,实现参数保存和记录存储模块。


2.用电脑上的socket通信封装嵌入式终端上的通信接口。


3.修改B502机器程序,把B502机器做成一个读卡器。


4.做一个TCP服务程序,开启服务端口,运行在远程的电脑上,服务程序通过电脑串口控制B502读卡器并提供读写卡服务接口,供客户端调用。


5.客户端根据提供的接口做应用,实现远程读写卡,消费,记录存储,上传。


 


三、操作说明


现场操作:


1.升级B502车载机程序,把B502机器当作读卡器使用,串口连接到电脑上。


2.在电脑上安装并运行读写卡服务程序,为客户端提供读写卡服务。


3.开启外网端口映射,没有固定IP也不是问题。只要现场的电脑能连外网,就可以把现场电脑的端口映射到外网,可以使用免费的端口映射工具。


4.提供现场映射过的公网IP地址和端口给开发人员。


5.把待读取的卡片放到机器上。


缺点:


现场必须有B502机器,且有串口线。B502机器还需要接12V的电源,体积大也不方便携带。所以用 B502机器做读卡器,有局限性,不通用。不过公司现有E711读卡器,该读卡器只需软件改造一下,升级下程序,就可以配合后台运行的TcpServer读写卡服务实现远程读卡器。消费速度上受网络原因有影响,不过远程读卡器只作为调试测试的手段,不是专门用来消费卡片的。实测消费完M1卡大概需要6秒钟。


客户端接口:


文件库libmyfile.a


通信库libmycom.a


读卡库libmycard.a


算法库libmycalc.a


客户端接口:


文件库libmyfile.a


通信库libmycom.a


读卡库libmycard.a


算法库libmycalc.a


接口和车载机程序的保持一致,参见头文件。


读卡库libmycard.a


CPU卡操作接口:


ICC_APDU_Exchange(……)


M1卡操作接口:


ICC_MiOne_LoadKey(…… );                        


ICC_MiOne_CheckKey(…… )


ICC_MiOne_BlkRead(……)


ICC_MiOne_BlkWrite(……)


ICC_MiOne_Value(……)


……


若是CPU卡,客户端可直接用TCP调试工具发指令调试卡片。


例如:


选择卡片3F00目录,直接在TCP助手上文本发送:00A40000023F00


若是选择现场PSAM卡的3F00目录,直接在TCP助手上文本发送:P00A40000023F00,前面多加了一个字母P


若是M1卡,指令有点儿复杂,客户端最好写个小程序,调用接口读写卡。


一天时间,初步实现了。原理就是TCP的socket通信转串口通信。源码如下;


#coding=utf-8
#author:yangyongzhen
#QQ:534117529
import sys,threading,time;
import serial;
import binascii,encodings;
import re;
import os;
from socket import *
from struct import *;
from myutil import *;
#''代表服务器为localhost
#在一个非保留端口号上进行监听
class ComThread:
    def __init__(self, Port=0):
        self.l_serial = None;
        self.alive = False;
        self.waitEnd = None;
        self.port = Port;
        #TCP部分
        self.myHost = ''
        self.myPort = 5050
        self.sockobj = socket(AF_INET, SOCK_STREAM)
        self.connection = None
        #数据
        self.snddata = ''
        self.rcvdata = ''
    def waiting(self):
        if not self.waitEnd is None:
            self.waitEnd.wait();
    def SetStopEvent(self):
        if not self.waitEnd is None:
            self.waitEnd.set();
        self.alive = False;
        self.stop();
    def start(self):
        self.l_serial = serial.Serial();
        self.l_serial.port = self.port;
        self.l_serial.baudrate = 115200;
        self.l_serial.timeout = 2;  #秒
        self.l_serial.open();
        if self.l_serial.isOpen():
            self.waitEnd = threading.Event();
            self.alive = True;
            self.thread_read = None;
            self.thread_read = threading.Thread(target=self.FirstReader);
            self.thread_read.setDaemon(1);
            self.thread_read.start();
            self.thread_write = None;
            self.thread_write = threading.Thread(target=self.FirstWriter);
            self.thread_write.setDaemon(1);
            self.thread_write.start();
            #TCP部分
            self.thread_TcpServer = None;
            self.thread_TcpServer = threading.Thread(target=self.TcpServer);
            self.thread_TcpServer.setDaemon(1);
            self.thread_TcpServer.start();
            self.thread_TcpSend = None;
            self.thread_TcpSend = threading.Thread(target=self.TcpSend);
            self.thread_TcpSend.setDaemon(1);
            self.thread_TcpSend.start();
            return True;
        else:
            return False;
    def FirstReader(self):
        while self.alive:
            # 接收间隔
            time.sleep(0.1);
            try:
                data = '';
                n = self.l_serial.inWaiting();
                if n:
                    data = self.l_serial.read(n);
                    #for l in xrange(len(data)):
                        #print '%02X' % ord(data[l]),
                    # 发送数据
                    print 'serial recv:'
                    print data;
                    self.snddata = data
                    #print_hex(data);
                # 判断结束
            except Exception, ex:
                print str(ex);
        self.waitEnd.set();
        self.alive = False;
    def FirstWriter(self):
        while self.alive:
            # 接收间隔
            time.sleep(0.1);
            try:
                #snddata = raw_input('\nenter data send:\n')
                if self.rcvdata!='':
                    self.l_serial.write(self.rcvdata); 
                    print 'serial send:'
                    print self.rcvdata;
                    self.rcvdata = '';
                #print_hex(snddata);
            except Exception, ex:
                print str(ex);
        self.waitEnd.set();
        self.alive = False;
    def TcpServer(self):
      self.sockobj.bind((self.myHost, self.myPort))
        self.sockobj.listen(1)
        print '\nTcpServer listen at 5050 oK!\n'
        while True:
            # 接收间隔
            time.sleep(0.5);
            self.connection, address = self.sockobj.accept()
            print 'Server connected by', address
            self.snddata = ''
            self.rcvdata = ''
            while True:
            #读取客户端套接字的下一行
            data = self.connection.recv(1024)
            #如果没有数量的话,那么跳出循环
            if not data: break
            #发送一个回复至客户端
            self.rcvdata = data
            #connection.send('Echo=>' + data)
            self.connection.close()
        self.waitEnd.set();
        self.alive = False;
    def TcpSend(self):
        while True:
            # 接收间隔
            time.sleep(0.5);
            while True:
              time.sleep(0.5);
              try:
                if not self.connection is None:
                  if self.snddata != '':
                    self.connection.send(self.snddata)
                    self.snddata = ''
              except Exception, ex:
            pass        
    def stop(self):
        self.alive = False;
        self.thread_read.join();
        if self.l_serial.isOpen():
            self.l_serial.close();
#测试用部分
if __name__ == '__main__':
    print 'CardTest TcpServer  - Simple Test Card Tool 1.00\n' 
    print 'Copyright (c) Newcapec 2015-2016.\n'
    com =raw_input('please enter com port(1-9):')
    rt = ComThread(int(com)-1);
    try:
        if rt.start():
            rt.waiting();
            rt.stop();
        else:
            pass;            
    except Exception,se:
        print str(se);
    if rt.alive:
        rt.stop();
    os.system("pause")
    print '';
    print 'End OK .';
    del rt;


实现的效果图:



用C语言封装一下读写卡接口,可以做个客户端测试了:



 

相关文章
|
30天前
|
测试技术 数据库 Python
Python装饰器实战:打造高效性能计时工具
在数据分析中,处理大规模数据时,分析代码性能至关重要。本文介绍如何使用Python装饰器实现性能计时工具,在不改变现有代码的基础上,方便快速地测试函数执行时间。该方法具有侵入性小、复用性强、灵活度高等优点,有助于快速发现性能瓶颈并优化代码。通过设置循环次数参数,可以更准确地评估函数的平均执行时间,提升开发效率。
103 61
Python装饰器实战:打造高效性能计时工具
|
2天前
|
JavaScript 搜索推荐 Android开发
【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
21 8
【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
|
2天前
|
数据采集 JavaScript Android开发
【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
23 7
【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
|
1月前
|
数据可视化 算法 数据挖掘
Python时间序列分析工具Aeon使用指南
**Aeon** 是一个遵循 scikit-learn API 风格的开源 Python 库,专注于时间序列处理。它提供了分类、回归、聚类、预测建模和数据预处理等功能模块,支持多种算法和自定义距离度量。Aeon 活跃开发并持续更新至2024年,与 pandas 1.4.0 版本兼容,内置可视化工具,适合数据探索和基础分析任务。尽管在高级功能和性能优化方面有提升空间,但其简洁的 API 和完整的基础功能使其成为时间序列分析的有效工具。
79 37
Python时间序列分析工具Aeon使用指南
|
25天前
|
存储 数据采集 数据处理
如何在Python中高效地读写大型文件?
大家好,我是V哥。上一篇介绍了Python文件读写操作,今天聊聊如何高效处理大型文件。主要方法包括:逐行读取、分块读取、内存映射(mmap)、pandas分块处理CSV、numpy处理二进制文件、itertools迭代处理及linecache逐行读取。这些方法能有效节省内存,提升效率。关注威哥爱编程,学习更多Python技巧。
|
26天前
|
存储 JSON 对象存储
如何使用 Python 进行文件读写操作?
大家好,我是V哥。本文介绍Python中文件读写操作的方法,包括文件读取、写入、追加、二进制模式、JSON、CSV和Pandas模块的使用,以及对象序列化与反序列化。通过这些方法,你可以根据不同的文件类型和需求,灵活选择合适的方式进行操作。希望对正在学习Python的小伙伴们有所帮助。欢迎关注威哥爱编程,全栈路上我们并肩前行。
|
1月前
|
存储 算法 Serverless
剖析文件共享工具背后的Python哈希表算法奥秘
在数字化时代,文件共享工具不可或缺。哈希表算法通过将文件名或哈希值映射到存储位置,实现快速检索与高效管理。Python中的哈希表可用于创建简易文件索引,支持快速插入和查找文件路径。哈希表不仅提升了文件定位速度,还优化了存储管理和多节点数据一致性,确保文件共享工具高效运行,满足多用户并发需求,推动文件共享领域向更高效、便捷的方向发展。
|
2月前
|
数据可视化 编译器 Python
Manim:数学可视化的强大工具 | python小知识
Manim(Manim Community Edition)是由3Blue1Brown的Grant Sanderson开发的数学动画引擎,专为数学和科学可视化设计。它结合了Python的灵活性与LaTeX的精确性,支持多领域的内容展示,能生成清晰、精确的数学动画,广泛应用于教育视频制作。安装简单,入门容易,适合教育工作者和编程爱好者使用。
558 7
|
3月前
|
JavaScript 前端开发 开发者
探索 DrissionPage: 强大的Python网页自动化工具
DrissionPage 是一个基于 Python 的网页自动化工具,结合了浏览器自动化的便利性和 requests 库的高效率。它提供三种页面对象:ChromiumPage、WebPage 和 SessionPage,分别适用于不同的使用场景,帮助开发者高效完成网页自动化任务。
348 4
|
3月前
|
开发者 Python
探索Python中的列表推导式:简洁而强大的工具
【10月更文挑战第41天】 在编程的世界中,效率与简洁是永恒的追求。本文将深入探讨Python编程语言中一个独特且强大的特性——列表推导式(List Comprehension)。我们将通过实际代码示例,展示如何利用这一工具简化代码、提升性能,并解决常见编程问题。无论你是初学者还是资深开发者,掌握列表推导式都将使你的Python之旅更加顺畅。

热门文章

最新文章