pm21-dragon/lectures/lecture-11/network and web technologies.ipynb
2025-01-10 10:14:09 +01:00

22 KiB
Raw Blame History

None <html> <head> </head>

Network and web technologies

(I.e. how can we interact with services on other computers from Python?)

Teletype machines "were adapted to provide a user interface to early mainframe computers and minicomputers, sending typed data to the computer and printing the response."

WACsOperateTeletype.jpg WWII era teletype machines. Source: https://commons.wikimedia.org/wiki/File:WACsOperateTeletype.jpg

Let's start with an idealized "serial port"

serial-port.png

Data is carried as a serial stream of bits (1s and 0s, high and low voltage).

We can think about teletype machines, USB, ethernet cables, WiFi/WLAN, and so on:

(From https://www.electroschematics.com/wp-content/uploads/2010/01/usb-wiring-connection.jpg?fit=1024%2C768) usb-wiring-connection.jpg

receiver-wiring-to-FC.png Source: https://www.expresslrs.org/quick-start/receivers/wiring-up/

(Ethernet cables get more sophisticated. https://serverfault.com/questions/449416/why-do-ethernet-cables-have-8-wires . This is a theme that repeats itself in computer networking.)

But this is not a "network", only two computers

From serial to network

We can build a robust network composed of many individual serial links by defining a common language -- namely packets with framing information - such as a destination address - and a payload. The individual nodes in the network use the framing information to forward the packet or to consume it.

This standardization has allowed hardware and software producers to interoperate by having common interfaces. Here is an ethernet frame:

Ethernet_Type_II_Frame_format.svg.png Source: https://commons.wikimedia.org/wiki/File:Ethernet_Type_II_Frame_format.svg

USB also uses the idea of frames to establish a standard for a robust single link "network". Here is a USB frame:

640px-USB_signal_example.svg.png "Data packets would have address field and payload between the packet ID and the end of packet." Source: https://commons.wikimedia.org/wiki/File:USB_signal_example.svg

640px-Speedport_W_921V,_Deutsche_Telekom-7294.jpg Source: https://de.m.wikipedia.org/wiki/Datei:Speedport_W_921V,_Deutsche_Telekom-7294.jpg

640px-Speedport_W_921V,_Deutsche_Telekom-7296.jpg Source: https://commons.wikimedia.org/wiki/File:Speedport_W_921V,_Deutsche_Telekom-7296.jpg

Wireless-router-network-diagram.png Source: https://www.conceptdraw.com/How-To-Guide/what-is-a-wireless-network

DHCP - Dynamic Host Configuration Protocol

DHCP is the protocol used when your computer first joins a network and asks for an IP address.

DHCP-protocol.png Source: https://www.boardinfinity.com/blog/dynamic-host-configuration-protocol-dhcp/

These notes give an overview of the most important of these levels with the idea of understanding what your computer is doing when you use the web browser or how a program you write would interact automatically with, for example, a remote server to perform a BLAST query.

TCP and UDP

From https://microchipdeveloper.com/tcpip:tcp-vs-udp tcp-vs-udp.jpg

In addition to source and destination address, TCP and UDP frames have "port" and thus address has a 2^16 (65536) TCP ports and 2^16 UDP ports.

In [1]:
!curl http://google.com/bananas
<!DOCTYPE html>
<html lang=en>
  <meta charset=utf-8>
  <meta name=viewport content="initial-scale=1, minimum-scale=1, width=device-width">
  <title>Error 404 (Not Found)!!1</title>
  <style>
    *{margin:0;padding:0}html,code{font:15px/22px arial,sans-serif}html{background:#fff;color:#222;padding:15px}body{margin:7% auto 0;max-width:390px;min-height:180px;padding:30px 0 15px}* > body{background:url(//www.google.com/images/errors/robot.png) 100% 5px no-repeat;padding-right:205px}p{margin:11px 0 22px;overflow:hidden}ins{color:#777;text-decoration:none}a img{border:0}@media screen and (max-width:772px){body{background:none;margin-top:0;max-width:none;padding-right:0}}#logo{background:url(//www.google.com/images/branding/googlelogo/1x/googlelogo_color_150x54dp.png) no-repeat;margin-left:-5px}@media only screen and (min-resolution:192dpi){#logo{background:url(//www.google.com/images/branding/googlelogo/2x/googlelogo_color_150x54dp.png) no-repeat 0% 0%/100% 100%;-moz-border-image:url(//www.google.com/images/branding/googlelogo/2x/googlelogo_color_150x54dp.png) 0}}@media only screen and (-webkit-min-device-pixel-ratio:2){#logo{background:url(//www.google.com/images/branding/googlelogo/2x/googlelogo_color_150x54dp.png) no-repeat;-webkit-background-size:100% 100%}}#logo{display:inline-block;height:54px;width:150px}
  </style>
  <a href=//www.google.com/><span id=logo aria-label=Google></span></a>
  <p><b>404.</b> <ins>Thats an error.</ins>
  <p>The requested URL <code>/bananas</code> was not found on this server.  <ins>Thats all we know.</ins>

Let's practice with the Star Wars API.

curl https://swapi.py4e.com/api/people/1/

We will use the requests library.

In [6]:
import requests
In [7]:
response = requests.get('https://swapi.py4e.com/api/people/4/')
response.text
Out[7]:
'{"name":"Darth Vader","height":"202","mass":"136","hair_color":"none","skin_color":"white","eye_color":"yellow","birth_year":"41.9BBY","gender":"male","homeworld":"https://swapi.py4e.com/api/planets/1/","films":["https://swapi.py4e.com/api/films/1/","https://swapi.py4e.com/api/films/2/","https://swapi.py4e.com/api/films/3/","https://swapi.py4e.com/api/films/6/"],"species":["https://swapi.py4e.com/api/species/1/"],"vehicles":[],"starships":["https://swapi.py4e.com/api/starships/13/"],"created":"2014-12-10T15:18:20.704000Z","edited":"2014-12-20T21:17:50.313000Z","url":"https://swapi.py4e.com/api/people/4/"}'
In [8]:
type(response)
Out[8]:
requests.models.Response
In [9]:
type(response.text)
Out[9]:
str
In [10]:
response.headers
Out[10]:
{'Date': 'Fri, 10 Jan 2025 09:07:09 GMT', 'Content-Type': 'application/json', 'Transfer-Encoding': 'chunked', 'Connection': 'keep-alive', 'Vary': 'Accept-Encoding, Accept, Cookie', 'X-Frame-Options': 'SAMEORIGIN', 'ETag': 'W/"84f396e29689886d73163ffeea41d4b8"', 'Allow': 'GET, HEAD, OPTIONS', 'X-Clacks-Overhead': 'GNU Terry Pratchett', 'Content-Encoding': 'gzip', 'Age': '10', 'Cache-Control': 'max-age=14400', 'cf-cache-status': 'HIT', 'Report-To': '{"endpoints":[{"url":"https:\\/\\/a.nel.cloudflare.com\\/report\\/v4?s=k1tyqow%2BTHfXBggdLlKOGfY3YUvPQIZw2P1EowugfuLk%2F2nzXWw0X%2FjfUeLAhQcRikRxCoEH%2FDEcASi998U%2FcN70vLqQIRPjvoxABihBwDr9ApD2b%2BpBaqDMJ2ujpKeewA%3D%3D"}],"group":"cf-nel","max_age":604800}', 'NEL': '{"success_fraction":0,"report_to":"cf-nel","max_age":604800}', 'Server': 'cloudflare', 'CF-RAY': '8ffb83dfbba02c0a-STR', 'alt-svc': 'h3=":443"; ma=86400', 'server-timing': 'cfL4;desc="?proto=TCP&rtt=4615&min_rtt=4583&rtt_var=1314&sent=4&recv=7&lost=0&retrans=0&sent_bytes=2830&recv_bytes=781&delivery_rate=630387&cwnd=251&unsent_bytes=0&cid=59f4e5c9b008e54f&ts=22&x=0"'}
In [11]:
x = response.json()
x
Out[11]:
{'name': 'Darth Vader',
 'height': '202',
 'mass': '136',
 'hair_color': 'none',
 'skin_color': 'white',
 'eye_color': 'yellow',
 'birth_year': '41.9BBY',
 'gender': 'male',
 'homeworld': 'https://swapi.py4e.com/api/planets/1/',
 'films': ['https://swapi.py4e.com/api/films/1/',
  'https://swapi.py4e.com/api/films/2/',
  'https://swapi.py4e.com/api/films/3/',
  'https://swapi.py4e.com/api/films/6/'],
 'species': ['https://swapi.py4e.com/api/species/1/'],
 'vehicles': [],
 'starships': ['https://swapi.py4e.com/api/starships/13/'],
 'created': '2014-12-10T15:18:20.704000Z',
 'edited': '2014-12-20T21:17:50.313000Z',
 'url': 'https://swapi.py4e.com/api/people/4/'}
In [12]:
x = response.json()
print(type(x))
x['name']
<class 'dict'>
Out[12]:
'Darth Vader'
In [13]:
requests.get('https://swapi.py4e.com/api/people/2/').json()['name']
Out[13]:
'C-3PO'
In [14]:
# webserver1.py from https://ruslanspivak.com/lsbaws-part1/

import socket
HOST, PORT = '0.0.0.0', 4448

listen_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
listen_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
listen_socket.bind((HOST, PORT))
listen_socket.listen(1)
print(f'Serving HTTP on port {PORT} ...')
while True:
    print("waiting for connection")
    client_connection, client_address = listen_socket.accept()
    print("Got connection from {}".format(client_address))
    request_data = client_connection.recv(1024)
    print(request_data.decode('utf-8'))

    http_response = b"""\
HTTP/1.1 200 OK

Hello, World! Guten tag.
"""
    client_connection.sendall(http_response)
    client_connection.close()
Serving HTTP on port 4448 ...
waiting for connection
Got connection from ('127.0.0.1', 61404)
GET / HTTP/1.1

waiting for connection
---------------------------------------------------------------------------
KeyboardInterrupt                         Traceback (most recent call last)
Cell In[14], line 13
     11 while True:
     12     print("waiting for connection")
---> 13     client_connection, client_address = listen_socket.accept()
     14     print("Got connection from {}".format(client_address))
     15     request_data = client_connection.recv(1024)

File ~/anaconda3/envs/wm01-dragon/lib/python3.11/socket.py:294, in socket.accept(self)
    287 def accept(self):
    288     """accept() -> (socket object, address info)
    289 
    290     Wait for an incoming connection.  Return a new socket
    291     representing the connection, and the address of the client.
    292     For IP sockets, the address info is a pair (hostaddr, port).
    293     """
--> 294     fd, addr = self._accept()
    295     sock = socket(self.family, self.type, self.proto, fileno=fd)
    296     # Issue #7995: if no default timeout is set and the listening
    297     # socket had a (non-zero) timeout, force the new socket in blocking
    298     # mode to override platform-specific socket flags inheritance.

KeyboardInterrupt: 

With telnet:

telnet 127.0.0.1 4448
GET / HTTP/1.1
Host: blahblah.com

HTML, CSS, and Javascript and WASM - browser languages

https://developer.mozilla.org/en-US/docs/Web

E.g. https://codepen.io/picks/pens/

JSON

In [2]:
import json

data1 = {'key1':[1.2,{32: 64,"c":"sdfg'dfsg"},3,4]}

print(type(data1))
buf = json.dumps(data1)
print(type(buf))
# buf

data2 = json.loads(buf)
print(type(data2))
data2
<class 'dict'>
<class 'str'>
<class 'dict'>
Out[2]:
{'key1': [1.2, {'32': 64, 'c': "sdfg'dfsg"}, 3, 4]}
In [3]:
buf
Out[3]:
'{"key1": [1.2, {"32": 64, "c": "sdfg\'dfsg"}, 3, 4]}'
</html>