python2.7入門---CGI編程&表單操做&cookie操做

    看到標題咱們首先有個疑問,什麼是CGI?CGI 目前由NCSA維護,NCSA定義CGI爲:CGI(Common Gateway Interface),通用網關接口,它是一段程序,運行在服務器上如:HTTP服務器,提供同客戶端HTML頁面的接口。爲了更好的瞭解CGI是如何工做的,咱們能夠從在網頁上點擊一個連接或URL的流程:html

 

  • 一、使用你的瀏覽器訪問URL並鏈接到HTTP web 服務器。
  • 二、Web服務器接收到請求信息後會解析URL,並查找訪問的文件在服務器上是否存在,若是存在返回文件的內容,不然返回錯誤信息。
  • 三、瀏覽器從服務器上接收信息,並顯示接收的文件或者錯誤信息。

    CGI程序能夠是Python腳本,PERL腳本,SHELL腳本,C或者C++程序等。咱們來看下架構圖:python


    在你進行CGI編程前,確保你的Web服務器支持CGI及已經配置了CGI的處理程序。Apache 支持CGI 配置,設置好CGI目錄:git

 

ScriptAlias /cgi-bin/ /var/www/cgi-bin/

    全部的HTTP服務器執行CGI程序都保存在一個預先配置的目錄。這個目錄被稱爲CGI目錄,並按照慣例,它被命名爲/var/www/cgi-bin目錄。CGI文件的擴展名爲.cgi,python也可使用.py擴展名。默認狀況下,Linux服務器配置運行的cgi-bin目錄中爲/var/www。若是你想指定其餘運行 CGI 腳本的目錄,能夠修改 httpd.conf 配置文件,以下所示:web

 

<Directory "/var/www/cgi-bin"> AllowOverride None Options +ExecCGI Order allow,deny Allow from all </Directory>

    在 AddHandler 中添加 .py 後綴,這樣咱們就能夠訪問 .py 結尾的 python 腳本文件:編程

 

AddHandler cgi-script .cgi .pl .py

    首先,咱們使用Python建立第一個CGI程序,文件名爲hello.py,文件位於/var/www/cgi-bin目錄中,內容以下:瀏覽器

 

#!/usr/bin/python # -*- coding: UTF-8 -*- print "Content-type:text/html" print # 空行,告訴服務器結束頭部 print '<html>' print '<head>' print '<meta charset="utf-8">' print '<title>Hello World - 個人第一個 CGI 程序!</title>' print '</head>' print '<body>' print '<h2>Hello World! 我是來自luyaran的第一CGI程序</h2>' print '</body>' print '</html>'

    文件保存後修改 hello.py,修改文件權限爲 755:緩存

 

chmod 755 hello.py 

    以上程序在瀏覽器訪問 http://localhost/cgi-bin/hello.py 顯示結果以下:安全

 

Hello World! 我是來自luyaran的第一CGI程序

    這個的hello.py腳本是一個簡單的Python腳本,腳本第一行的輸出內容"Content-type:text/html"發送到瀏覽器並告知瀏覽器顯示的內容類型爲"text/html"。用 print 輸出一個空行用於告訴服務器結束頭部信息。hello.py文件內容中的" Content-type:text/html"即爲HTTP頭部的一部分,它會發送給瀏覽器告訴瀏覽器文件的內容類型。HTTP頭部的格式以下:服務器

 

HTTP 字段名: 字段內容

    例如:cookie

 

Content-type: text/html

    如下表格介紹了CGI程序中HTTP頭部常用的信息:

 

描述
Content-type: 請求的與實體對應的MIME信息。例如: Content-type:text/html
Expires: Date 響應過時的日期和時間
Location: URL 用來重定向接收方到非請求URL的位置來完成請求或標識新的資源
Last-modified: Date 請求資源的最後修改時間
Content-length: N 請求的內容長度
Set-Cookie: String 設置Http Cookie

    全部的CGI程序都接收如下的環境變量,這些變量在CGI程序中發揮了重要的做用:

 

變量名 描述
CONTENT_TYPE 這個環境變量的值指示所傳遞來的信息的MIME類型。目前,環境變量CONTENT_TYPE通常都是:application/x-www-form-urlencoded,他表示數據來自於HTML表單。
CONTENT_LENGTH 若是服務器與CGI程序信息的傳遞方式是POST,這個環境變量即便從標準輸入STDIN中能夠讀到的有效數據的字節數。這個環境變量在讀取所輸入的數據時必須使用。
HTTP_COOKIE 客戶機內的 COOKIE 內容。
HTTP_USER_AGENT 提供包含了版本數或其餘專有數據的客戶瀏覽器信息。
PATH_INFO 這個環境變量的值表示緊接在CGI程序名以後的其餘路徑信息。它經常做爲CGI程序的參數出現。
QUERY_STRING 若是服務器與CGI程序信息的傳遞方式是GET,這個環境變量的值即便所傳遞的信息。這個信息經跟在CGI程序名的後面,二者中間用一個問號'?'分隔。
REMOTE_ADDR 這個環境變量的值是發送請求的客戶機的IP地址,例如上面的192.168.1.67。這個值老是存在的。並且它是Web客戶機須要提供給Web服務器的惟一標識,能夠在CGI程序中用它來區分不一樣的Web客戶機。
REMOTE_HOST 這個環境變量的值包含發送CGI請求的客戶機的主機名。若是不支持你想查詢,則無需定義此環境變量。
REQUEST_METHOD 提供腳本被調用的方法。對於使用 HTTP/1.0 協議的腳本,僅 GET 和 POST 有意義。
SCRIPT_FILENAME CGI腳本的完整路徑
SCRIPT_NAME CGI腳本的的名稱
SERVER_NAME 這是你的 WEB 服務器的主機名、別名或IP地址。
SERVER_SOFTWARE 這個環境變量的值包含了調用CGI程序的HTTP服務器的名稱和版本號。例如,上面的值爲Apache/2.2.14(Unix)

    如下是一個簡單的CGI腳本輸出CGI的環境變量:

 

#!/usr/bin/python # -*- coding: UTF-8 -*- # filename:test.py import os print "Content-type: text/html" print print "<meta charset=\"utf-8\">" print "<b>環境變量</b><br>"; print "<ul>" for key in os.environ.keys(): print "<li><span style='color:green'>%30s </span> : %s </li>" % (key,os.environ[key]) print "</ul>"

    將以上點保存爲 test.py ,並修改文件權限爲 755,執行結果以下:


    瀏覽器客戶端經過兩種方法向服務器傳遞信息,這兩種方法就是 GET 方法和 POST 方法。首先咱們來看下使用GET方法發送編碼後的用戶信息到服務端,數據信息包含在請求頁面的URL上,以"?"號分割, 以下所示:

 

http://www.test.com/cgi-bin/hello.py?key1=value1&key2=value2

    有關 GET 請求的其餘一些註釋:

 

  • GET 請求可被緩存
  • GET 請求保留在瀏覽器歷史記錄中
  • GET 請求可被收藏爲書籤
  • GET 請求不該在處理敏感數據時使用
  • GET 請求有長度限制
  • GET 請求只應當用於取回數據

    如下是一個簡單的URL,使用GET方法向hello_get.py程序發送兩個參數:

 

/cgi-bin/test.py?name=luyaran&url=http://www.luyaran.com

    如下爲hello_get.py文件的代碼:

 

#!/usr/bin/python # -*- coding: UTF-8 -*- # filename:test.py # CGI處理模塊 import cgi, cgitb # 建立 FieldStorage 的實例化 form = cgi.FieldStorage() # 獲取數據 site_name = form.getvalue('name') site_url = form.getvalue('url') print "Content-type:text/html" print print "<html>" print "<head>" print "<meta charset=\"utf-8\">" print "<title>luyaran CGI 測試實例</title>" print "</head>" print "<body>" print "<h2>%s官網:%s</h2>" % (site_name, site_url) print "</body>" print "</html>"

    測試結果你們能夠本身運行來看下,這裏就不贅述了。如下是一個經過HTML的表單使用GET方法向服務器發送兩個數據,提交的服務器腳本一樣是hello_get.py文件,hello_get.html 代碼以下:

 

<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title>luyaran(luyaran.com)</title> </head> <body> <form action="/cgi-bin/hello_get.py" method="get"> 站點名稱: <input type="text" name="name"> <br /> 站點 URL: <input type="text" name="url" /> <input type="submit" value="提交" /> </form> </body> </html>

    默認狀況下 cgi-bin 目錄只能存放腳本文件,咱們將 hello_get.html 存儲在 test 目錄下,修改文件權限爲 755:

 

chmod 755 hello_get.html

    運行效果就不贅述了啊,我們直接來看post。使用POST方法向服務器傳遞數據是更安全可靠的,像一些敏感信息如用戶密碼等須要使用POST傳輸數據。如下一樣是hello_get.py ,它也能夠處理瀏覽器提交的POST表單數據:

 

#!/usr/bin/python # -*- coding: UTF-8 -*- # CGI處理模塊 import cgi, cgitb # 建立 FieldStorage 的實例化 form = cgi.FieldStorage() # 獲取數據 site_name = form.getvalue('name') site_url = form.getvalue('url') print "Content-type:text/html" print print "<html>" print "<head>" print "<meta charset=\"utf-8\">" print "<title>luyaran CGI 測試實例</title>" print "</head>" print "<body>" print "<h2>%s官網:%s</h2>" % (site_name, site_url) print "</body>" print "</html>"

    如下爲表單經過POST方法(method="post")向服務器腳本 hello_get.py 提交數據:

 

<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title>luyaran(luyaran.com)</title> </head> <body> <form action="/cgi-bin/hello_get.py" method="post"> 站點名稱: <input type="text" name="name"> <br /> 站點 URL: <input type="text" name="url" /> <input type="submit" value="提交" /> </form> </body> </html>

    再來看CheckBox這個鬼。checkbox用於提交一個或者多個選項數據,HTML代碼以下:

 

<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title>luyaran(luyaran.com)</title> </head> <body> <form action="/cgi-bin/checkbox.py" method="POST" target="_blank"> <input type="checkbox" name="luyaran" value="on" /> luyaran <input type="checkbox" name="google" value="on" /> Google <input type="submit" value="選擇站點" /> </form> </body> </html>

    如下爲 checkbox.py 文件的代碼:

 

#!/usr/bin/python # -*- coding: UTF-8 -*- # 引入 CGI 處理模塊  import cgi, cgitb # 建立 FieldStorage的實例  form = cgi.FieldStorage() # 接收字段數據 if form.getvalue('google'): google_flag = "是" else: google_flag = "否" if form.getvalue('luyaran'): luyaran_flag = "是" else: luyaran_flag = "否" print "Content-type:text/html" print print "<html>" print "<head>" print "<meta charset=\"utf-8\">" print "<title>luyaran CGI 測試實例</title>" print "</head>" print "<bodyluyaran是否選擇了 : %s</h2>" % luyaran_flag print "<h2> Google 是否選擇了 : %s</h2>" % google_flag print "</body>" print "</html>"

    修改 checkbox.py 權限:

 

chmod 755 checkbox.py

    而後運行文件就行了。你們能夠感覺下。。。咱們再來看radio。Radio 只向服務器傳遞一個數據,HTML代碼以下:

 

<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title>luyaran(luyaran.com)</title> </head> <body> <form action="/cgi-bin/radiobutton.py" method="post" target="_blank"> <input type="radio" name="site" value="luyaran" /> luyaran <input type="radio" name="site" value="google" /> Google <input type="submit" value="提交" /> </form> </body> </html>

    radiobutton.py 腳本代碼以下:

 

#!/usr/bin/python # -*- coding: UTF-8 -*- # 引入 CGI 處理模塊  import cgi, cgitb # 建立 FieldStorage的實例  form = cgi.FieldStorage() # 接收字段數據 if form.getvalue('site'): site = form.getvalue('site') else: site = "提交數據爲空" print "Content-type:text/html" print print "<html>" print "<head>" print "<meta charset=\"utf-8\">" print "<title>luyaran CGI 測試實例</title>" print "</head>" print "<body>" print "<h2> 選中的網站是 %s</h2>" % site print "</body>" print "</html>"

    而後就是修改 radiobutton.py 權限了,完事以後就能夠運行來看下效果。再來看textare。Textarea 向服務器傳遞多行數據,HTML代碼以下:

 

<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title>luyaran(luyaran.com)</title> </head> <body> <form action="/cgi-bin/textarea.py" method="post" target="_blank"> <textarea name="textcontent" cols="40" rows="4"> 在這裏輸入內容... </textarea> <input type="submit" value="提交" /> </form> </body> </html>

    textarea.py 腳本代碼以下:

 

#!/usr/bin/python # -*- coding: UTF-8 -*- # 引入 CGI 處理模塊  import cgi, cgitb # 建立 FieldStorage的實例  form = cgi.FieldStorage() # 接收字段數據 if form.getvalue('textcontent'): text_content = form.getvalue('textcontent') else: text_content = "沒有內容" print "Content-type:text/html" print print "<html>" print "<head>"; print "<meta charset=\"utf-8\">" print "<title>luyaran CGI 測試實例</title>" print "</head>" print "<body>" print "<h2> 輸入的內容是:%s</h2>" % text_content print "</body>" print "</html>"

    完事仍是修改權限再運行。而後就是下拉框。HTML 下拉框代碼以下:

 

<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title>luyaran(luyaran.com)</title> </head> <body> <form action="/cgi-bin/dropdown.py" method="post" target="_blank"> <select name="dropdown"> <option value="luyaran" selected>luyaran</option> <option value="google">Google</option> </select> <input type="submit" value="提交"/> </form> </body> </html>

    dropdown.py 腳本代碼以下所示:

 

#!/usr/bin/python # -*- coding: UTF-8 -*- # 引入 CGI 處理模塊  import cgi, cgitb # 建立 FieldStorage的實例  form = cgi.FieldStorage() # 接收字段數據 if form.getvalue('dropdown'): dropdown_value = form.getvalue('dropdown') else: dropdown_value = "沒有內容" print "Content-type:text/html" print print "<html>" print "<head>" print "<meta charset=\"utf-8\">" print "<title>luyaran CGI 測試實例</title>" print "</head>" print "<body>" print "<h2> 選中的選項是:%s</h2>" % dropdown_value print "</body>" print "</html>"

    運行以前不要忘記修改權限哦。接下來就看下cookie怎麼運用。在 http 協議一個很大的缺點就是不對用戶身份的進行判斷,這樣給編程人員帶來很大的不便,而 cookie 功能的出現彌補了這個不足。cookie 就是在客戶訪問腳本的同時,經過客戶的瀏覽器,在客戶硬盤上寫入紀錄數據 ,當下次客戶訪問腳本時取回數據信息,從而達到身份判別的功能,cookie 經常使用在身份校驗中。http cookie的發送是經過http頭部來實現的,他早於文件的傳遞,頭部set-cookie的語法以下:

 

Set-cookie:name=name;expires=date;path=path;domain=domain;secure 
  • name=name: 須要設置cookie的值(name不能使用";"和","號),有多個name值時用 ";" 分隔,例如:name1=name1;name2=name2;name3=name3
  • expires=date: cookie的有效期限,格式: expires="Wdy,DD-Mon-YYYY HH:MM:SS"
  • path=path: 設置cookie支持的路徑,若是path是一個路徑,則cookie對這個目錄下的全部文件及子目錄生效,例如: path="/cgi-bin/",若是path是一個文件,則cookie指對這個文件生效,例如:path="/cgi-bin/cookie.cgi"。
  • domain=domain: 對cookie生效的域名,例如:domain="www.runoob.com"
  • secure: 若是給出此標誌,表示cookie只能經過SSL協議的https服務器來傳遞。
  • cookie的接收是經過設置環境變量HTTP_COOKIE來實現的,CGI程序能夠經過檢索該變量獲取cookie信息。

    Cookie的設置很是簡單,cookie會在http頭部單獨發送。如下實例在cookie中設置了name 和 expires:

 

#!/usr/bin/python # -*- coding: UTF-8 -*- #  print 'Content-Type: text/html' print 'Set-Cookie: name="luyaran";expires=Wed, 28 Aug 2016 18:30:00 GMT' print print """ <html>     <head>         <meta charset="utf-8">         <title>luyaran(luyaran.com)</title>     </head> <body> <h1>Cookie set OK!</h1> </body> </html> """

    將以上代碼保存到 cookie_set.py,並修改 cookie_set.py 權限:

 

chmod 755 cookie_set.py

    以上實例使用了 Set-Cookie 頭信息來設置Cookie信息,可選項中設置了Cookie的其餘屬性,如過時時間Expires,域名Domain,路徑Path。這些信息設置在 "Content-type:text/html"以前。而後就來看cookie信息的檢索。Cookie信息檢索頁很是簡單,Cookie信息存儲在CGI的環境變量HTTP_COOKIE中,存儲格式以下:

 

key1=value1;key2=value2;key3=value3....

    如下是一個簡單的CGI檢索cookie信息的程序:

 

#!/usr/bin/python # -*- coding: UTF-8 -*- # 導入模塊 import os import Cookie print "Content-type: text/html" print print """ <html> <head> <meta charset="utf-8"> <title>luyaran(luyaran.com)</title> </head> <body> <h1>讀取cookie信息</h1> """ if 'HTTP_COOKIE' in os.environ: cookie_string=os.environ.get('HTTP_COOKIE') c=Cookie.SimpleCookie() c.load(cookie_string) try: data=c['name'].value print "cookie data: "+data+"<br>" except KeyError: print "cookie 沒有設置或者已過去<br>" print """ </body> </html> """

    將以上代碼保存到 cookie_get.py,並修改 cookie_get.py 權限,完事以後運行就會看到效果了。

    好啦,此次的分享到這裏就差很少結束了。後續的會繼續更新的。若是感受不錯的話,請多多點贊支持哦。。。

  原文連接:https://blog.csdn.net/luyaran/article/details/79993226

相關文章
相關標籤/搜索