Maxbox Starter 18: Start With Arduino Programming
Maxbox Starter 18: Start With Arduino Programming
Maxbox Starter 18: Start With Arduino Programming
Lets begin with HTTP (Hypertext Transfer Protocol) and TCP. TCP/IP stands for Transmission Control Protocol and Internet Protocol. TCP/IP can mean many things, but in most cases, it refers to the network protocol itself. Each computer on a TCP/IP network has a unique address associated with it, the so called IPAddress. Some computers may have more than one address associated with them. An IP address is a 32-bit number and is usually represented in a dot notation, e.g. 192.168.0.1. Each section represents one byte of the 32-bit address. In maXbox a connection with HTTP represents an object. In our case we will operate with the localhost. It is common for computers to refer to themselves with the name localhost and the IP number 127.0.0.1. When HTTP is used on the Internet, browsers like Firefox act as clients and the application that is hosting the website like softwareschule.ch acts as the server.
We have to name the game, means the programs name is above. This example requires two objects from the classes: TIdCustomHTTPServer and TComPort so the second one is to connect and transport with the COM Ports to Arduino (see below). TComPort by Dejan Crnila2 are Delphi/C++ Builder serial communications components. It is generally easy to use for basic serial communications purposes, alternative to the TurboPower ASYNCPro. It includes 5 components: TComPort, TComDataPacket, TComComboBox, TComRadioGroup and TComLed. With these tools you can build serial communication application easier and faster than ever. First we start with the web server and second we explain the COM port. After creating the object in line 75 we use the first methods to configure our server calling Port and IP. The object makes a bind connection with the Active method by passing a web server configuration.
75 HTTPServer:= TIdCustomHTTPServer.Create(self);
So the object HTTPServer has some methods and properties like Active you can find in the TIdCustomHTTPServer.pas unit or IdHTTPServer library. A library is a collection of code or classes, which you can include in your program. By storing your commonly used code in a library, you can reuse code; another advantage of modular programming. Once a unit is tested its stable to use.
2
http://sourceforge.net/projects/comport/ 2
Indy is designed to provide a very high level of abstraction. Much more stuff or intricacies and details of the TCP/IP stack are hidden from the Indy programmer. A typical Indy client session looks like this:
with IndyClient do begin Host:= 'zip.pbe.com'; // Host to call Port:= 6000; // Port to call the server on Connect; // get something to with it end;
Indy is different than other so called Winsock components you may be familiar with. If you've worked with other components, the best approach for you may be to forget how they work. Nearly all other components use non-blocking (asynchronous) calls and act asynchronously. They require you to respond to events, set up state machines, and often perform wait loops. In facts there are 2 programming models used in TCP/IP applications. Non Blocking means that the application will not be blocked when the application socket read/write data. This is efficient, because your application don't have to wait for a connection. Unfortunately, it is complicated to develop.
So lets get back to our HTTP Create in line 75. In line 81 and 82 you see the port and IP address configuration of a const in line 6, instead of IP you can also set a host name as a parameter.
76 77 78 79 80 81 82 83 84 85 86 with HTTPServer do begin if Active then Free; if not Active then begin Bindings.Clear; bindings.Add; bindings.items[0].Port:= APORT; bindings.items[0].IP:= IPADDR; //'127.0.0.1'; Active:= true; onCommandGet:= @HTTPServerGet; PrintF('Listening HTTP on %s:%d.', [Bindings[0].IP,Bindings[0].Port]); end;
Host names are "human-readable" names for IP addresses. An example host name is max.kleiner.com, the www is just a convention and not necessary. Every host name has an equivalent IP address, e.g. www.hower.org = 207.65.96.71. Host names are used both to make it easier on us humans, and to allow a computer to change its IP address without causing all of its potential clients (callers) to lose track of it. Often called URL or links because a host address is normally inserted at the top of the browser as one of the first items to look at for searching. The full target of the request message is given by the URL property. Usually, this is a URL that can be broken down into the protocol (HTTP), Host (server system), script name (server application), path info (location on the host), and a query. So far we have learned little about HTTP and host names. Now its time to run your program at first with F9 (if you havent done yet) and learn something about GET and HTML. The program (server) generates a standard HTML output or other formats (depending on the MIME type) after downloading with GET or HEAD. So our command to shine on a LED is ../LED and to switch off is ../DEL (127.0.0.1:8000/LED). Those are GET commands send with the browser. The first line identifies the request as a GET. A GET request message asks the Web server application to return the content associated with the URI that follows the word GET. The following shows the magic behind in the method HTTPServerGet():
40 procedure HTTPServerGet(aThr: TIdPeerThread; reqInf: TIdHTTPRequestInfo; 41 respInf: TIdHTTPResponseInfo);
One word concerning the thread: In the internal architecture there are 2 threads categories. First is a listener thread that listens and waits for a connection. So we don't have to worry about threads, the built in thread TIdPeerThread will be served by Indy through a parameter: OK. Its time for magix:
54 if uppercase(localcom) = uppercase('/LED') then begin 55 56 57 cPort.WriteStr('1') writeln(localcom+ ': LED on'); RespInfo.ContentText:= getHTMLContentString('LED is: ON');
58 end else 59 if uppercase(localcom) = uppercase('/DEL') then begin 60 61 cPort.WriteStr('A'); writeln(localcom+ ': LED off');
62
OFF')
63 end;
HTTP request messages contain many headers that describe information about the client, the target of the request, the way the request should be handled, and any content sent with the request. Each header is identified by a name, such as "Host" followed by a string value. When an HTML hypertext link is selected (or the user otherwise specifies a URL), the browser collects information about the protocol, the specified domain, the path to the information, the date and time, the operating environment, the browser itself, and other content information. It then composes a request. The acronym HTML stands for Hyper Text Markup Language - the primary programming language used to write content on the web. One of a practical way to learn much more about actually writing HTML is to get in the maXbox editor and load or open a web-file with the extension html. Or you copy the output and paste it in a new maXbox instance. Then you click on the right mouse click (context menu) and change to HTML Syntax!
When the browser starts from the script the server is ready for commands to pass as chars in line 55 or 60 to the serial communication. When a the server application finishes with our client request, it lights the LED and constructs a page of HTML code or other MIME content, and passes the result back (via the server in TIdHTTPResponseInfo ) to the client for display.
56 57 writeln(localcom+ ': LED on'); RespInfo.ContentText:= getHTMLContentString('LED is: ON');
Have you tried the program, its also possible to test the server without Arduino or a browser. The Compile button is also used to check that your code is correct, by verifying the syntax before the program starts. Another way to check the syntax before run is F2 or the Syntax Check in the menu Program. When you run this code from the script 102_pas_http_download.txt you will see the content (first 10 lines) of the site in HTML format with the help of the method memo2.lines.add:
begin idHTTP:= TIdHTTP.Create(NIL) try memo2.lines.text:= idHTTP.Get2('http://127.0.0.1') for i:= 1 to 10 do memo2.lines.add(IntToStr(i)+' :'+memo2.lines[i]) finally idHTTP.Free end
The Object TIdHTTP is a dynamically allocated block of memory whose structure is determined by its class type. Each object has a unique copy of every field defined in the class, but all instances of a class share the same methods. With the method Get1 you can download files.
11 begin 12 13 14 15 16 myURL:= 'http://www.softwareschule.ch/download/maxbox_examples.zip'; zipStream:= TFileStream.Create('myexamples2.zip', fmCreate) idHTTP:= TIdHTTP.Create(NIL) try idHTTP.Get1(myURL, zipStream)
Of course a lot of lines to get a file from the web try it shorter with the magic function wGet():
wGet('http://www.softwareschule.ch/download/maxbox_starter17.pdf','mytestpdf.pdf');
It downloads the entire file into memory if the data is compressed (Indy does not support streaming decompression for HTTP yet). Next we come closer to the main event of our web server in line 40, its the event onCommandGet with the corresponding event handler method @HTTPServerGet() and one object of TIdPeerThread. You can use them at server as the way to serve files of many kinds!
The Arduino can be used to develop stand-alone interactive objects or it can be connected to a computer to retrieve or send data to the Arduino and then act on that data (e.g. Send sensor data out to the web or write data on a control LED). Now we change the code site to the Arduino Editor to explain how he handles our commands (chars).
Serial.begin tells the Arduino to start serial communications and the number within the parenthesis, in this case 9600, sets the baud rate (characters per second) that the serial line will communicate at. int val = 0; int ledPin1 = 2; void setup() { pinMode(ledPin1,OUTPUT); Serial.begin(9600); ..} In the main loop we have an if statement. The condition it is checking the value in (Serial.read). The Serial.available command checks to see if any characters have been sent down the serial line. If any characters have been received then the condition is met and the code within the if statements code block is now executed, you see if 1 then ON and if A then OFF. The condition it is checking is simply a Char its up to you to code a protocol of your own .
void loop () { val = Serial.read(); if (val !=-1){ if (val=='1'){ digitalWrite(ledPin1,HIGH); } else if (val=='A'){ digitalWrite(ledPin1,LOW); } // read the serial port
// variable to store the data from the serial port // LED connected to digital pin 2 or the inbuilt 13!
Serial.print("Data entered: "); and this is our way of sending data back from the Arduino to the PC. In this case the print command sends whatever is within the parenthesis to the PC, via the USB cable, where we can read it in the Serial Monitor window or in maXbox. In Line 394 we find a last function of the RTL (Runtime Library) of Indy:
83 Writeln(DateTimeToInternetStr(Now, true))
We get the real time zone based time back! This information of RTL functions is contained in various unit files that are a standard part of Delphi or Indy. This collection of units is referred to as the RTL (run time library). The RTL contains a very large number of functions and procedures for you to use. By the way: In my research and in my debugging, I found that the function GetTimeZoneInformation was returning a value oriented towards converting a Time from GMT to Local Time. We wanted to do the reverse for getting the difference. The issue with TIdMessage.UseNowForTime = False bug was that the TIdMessage was calling Borland's date function instead of using the Date property, see Appendix.
1.4 FrameWorkFlow
At last but not least some words about sockets and streams. A listening server socket component automatically accepts client connection requests when they are received. You receive notification every time this occurs in an OnCommandGet event. Server connections are formed by server sockets when a listening socket accepts a client request. A description of the server socket that completes the connection to the client is sent to the client when the server accepts the connection. The connection is then established when the client socket receives this description and completes the connection. 7
Socket connections can be divided into three basic types, which reflect how the connection was initiated and what the local socket is connected to. These are Client connections. Listening connections. Server connections.
Once the connection to a client socket is completed, the server connection is indistinguishable from a client connection. Both end points have the same capabilities and receive the same types of events. Only the listening connection is fundamentally different, as it has only a single endpoint. Sockets provide the interface between your network server or client application and the networking software. You must provide the interface between your application and the clients that use it. You can copy the API of a standard third party server (such as Apache), or you can design and publish your own API.
Sockets let your network application communicate with other systems over the network. Each socket can be viewed as an endpoint in a network connection. It has an address that specifies: The system on which it is running. The types of interfaces it understands. The port it is using for the connection.
A full description of a socket connection includes the addresses of the sockets on both ends of the connection. You can describe the address of each socket endpoint by supplying both the IP address or host and the port number. Many of the protocols that control activity on the Internet are defined in Request for Comment (RFC) documents that are created, updated, and maintained by the Internet Engineering Task Force (IETF), the protocol engineering and development arm of the Internet. There are several important RFCs that you will find useful when writing Internet applications: RFC822, "Standard for the format of ARPA Internet text messages," describes the structure and content of message headers.
RFC1521, "MIME (Multipurpose Internet Mail Extensions) Part One: Mechanisms for Specifying and Describing the Format of Internet Message Bodies," describes the method used to encapsulate and transport multipart and multiformat messages. RFC1945, "Hypertext Transfer ProtocolHTTP/1.0," describes a transfer mechanism used to distribute collaborative hypermedia documents.
In the next line we just start a browser to test our server in a so called frame work flow
31 procedure letOpenBrowser; 32 // TS_ShellExecuteCmd = (seCmdOpen,seCmdPrint,seCmdExplore); 33 begin 34 //ShellAPI.ShellExecute(Handle,PChar('open'),'http://127.0.0.1:80/',Nil,Nil,0); 35 S_ShellExecute('http:'+IPADDR+':'+IntToStr(APORT)+'/','',seCmdOpen)
36 end;
Try to change the IP address in line 68 of the IP:= IPADDR with a DHCP or dynDNS address, so you can reach Arduino from an Android App, but change also the const in line 10. Try to get data back from Arduino as a test case like this: Serial.print() in Arduino and cPort.ReadStr() in maXbox
Function ReadStr( var Str: string; Count: Integer): Integer'); //CPort Lib //S_ShellExecute('http:'+IPADDR+':'+IntToStr(APORT)+'/soa_delphi.pdf','',seCmdOpen)
Some notes at last about firewalls or proxy-servers. It depends on your network infrastructure to get a file or not, maybe you cant download content cause of security reasons and it stops with Socket-Error # 10060 and a time out error. Furthermore, it also depends on the firewall in use at both ends. If it's automatic and recognises data that needs a response automatically it will work. It needs an administrator to open ports etc. youre stuffed or configured. A firewall that only allows connections to the listening port will also prevent the remote debugger from working. But after all HTTP lets you create clients that can communicate with an application server that is protected by a firewall. Hope you did learn in this tutorial the topic of Arduino with a web server. The Arduino is an amazing device and will enable you to make anything from interactive works of art to robots. With a little enthusiasm to learn how to program the Arduino and make it interact with other components a well as a bit of imagination, you can build anything you want. The Arduino can also be extended with the use of Shields which circuit boards are containing other devices (e.g. GPS receivers, LED Cubes, LCD Displays, MIDI Synthesizers, Ethernet connections, etc.) that you can simply slot into the top of your Arduino to get extra functionality. Next Tutorial Nr. 19 shows the topic More Physical Computing with the Arduino Board and Android. Arduino is an open-source single-board microcontroller, descendant of the open-source Wiring platform designed to make the process of using electronics in multitude projects more accessible. The Arduino can be connected to LEDs, Dot Matrix displays, LED displays, buttons, switches, motors, temperature sensors, pressure sensors, distance sensors, webcams, printers, GPS receivers, Ethernet modules and so on. 9
The Arduino board is made of an Atmel AVR Microprocessor, a crystal or oscillator (basically a crude clock that sends time pulses to the microcontroller to enable it to operate at the correct what type of Arduino you have, you may also have a USB connector to enable it to be connected to a PC or Linux to upload or retrieve data. The board exposes the microcontrollers I/O (Input/Output) pins to enable you to connect those pins to other circuits, buses or to sensors, etc.
Feedback @
[email protected]
Literature: Kleiner et al., Patterns konkret, 2003, Software & Support Links of maXbox, Arduino and Indy:
http://www.softwareschule.ch/maxbox.htm http://www.indyproject.org/Sockets/index.EN.aspx http://en.wikipedia.org/wiki/Arduino http://sourceforge.net/projects/maxbox http://sourceforge.net/apps/mediawiki/maxbox/ http://sourceforge.net/projects/delphiwebstart
10
1.5 Appendix
Function DateTimeToInternetStr(const Value: TDateTime): String; var strOldFormat, strOldTFormat, strDate: String; wDay, wMonth, wYear:WORD; begin {needed to prevent wild results} Result := ''; strOldFormat := ShortDateFormat ; ShortDateFormat := 'DD.MM.YYYY'; // Date case DayOfWeek(Value) of 1: strDate := 'Sun, '; 2: strDate := 'Mon, '; 3: strDate := 'Tue, '; 4: strDate := 'Wed, '; 5: strDate := 'Thu, '; 6: strDate := 'Fri, '; 7: strDate := 'Sat, '; end; DecodeDate(Value, wYear, wMonth, wDay); strDate := strDate + IntToStr(wDay) + #32; case wMonth of 1: strDate := strDate + 'Jan '; 2: strDate := strDate + 'Feb '; 3: strDate := strDate + 'Mar '; 4: strDate := strDate + 'Apr '; 5: strDate := strDate + 'May '; 6: strDate := strDate + 'Jun '; 7: strDate := strDate + 'Jul '; 8: strDate := strDate + 'Aug '; 9: strDate := strDate + 'Sep '; 10: strDate := strDate + 'Oct '; 11: strDate := strDate + 'Nov '; 12: strDate := strDate + 'Dec '; end; //Correction strOldTFormat := LongTimeFormat; LongTimeFormat := 'HH:NN:SS'; strDate := strDate + IntToStr(wYear) + #32 + TimeToStr(Value); Result := strDate + #32 + DateTimeToGmtOffSetStr(OffsetFromUTC,False); LongTimeFormat := strOldTFormat; { strOldTFormat := LongDateFormat; LongDateFormat := 'HH:NN:SS'; strDate := strDate + IntToStr(wYear) + #32 + TimeToStr(Value); LongDateFormat := strOldTFormat; Result := strDate + #32 + DateTimeToGmtOffSetStr(OffsetFromUTC,False); ShortDateFormat := strOldFormat ; } end;
11
13