分享
 
 
 

用VB编写标准CGI程序

王朝vb·作者佚名  2006-01-08
窄屏简体版  字體: |||超大  

<!--StartFragment-->

虽然目前已经有很多可以取代CGI且性能较CGI要高的技术(例如ASP、ISAPI、NSAPI等),但使用它们需要用到专门的知识和工具,并且利用这些技术所编制的程序也只适用于特定的Web服务器或系统平台。考虑到CGI程序的易用易学性、跨服务器平台特性等优点,CGI程序还将在WWW上占有一一席之地。本文将介绍如何用VB编写标准的CGI程序,最后本文还用VB实现了一个有关主页客户留言簿的CGI程序。

利用通用网关接口(CGI),Web服务器可以执行一些外部程序,并将这些外部程序所产生的输出结果和Web服务器所管理的静态文本、图像和声音融合在一起传给相应的Web浏览器。当客户机的浏览器向Web服务器请求一个HTML文件时,服务器在收到请求后就去寻找这个文件并将找到的文件传送给客户机。而当客户机的请求是一个CGI程序时,Web服务器将激活客户机所请求的CGI程序并把程序的执行结果传给客户机。

标准的CGI程序是通过环境变量和标准输入输出来与Web服务器交换信息的。任何一个被系统激活的进程都拥有标准输入和输出这两个文件句柄,CGI程序的进程也不例外。不过,当CGI程序被Web服务器激活以后,它的标准输入STDIN被连接到Web服务器的标准输出STDOUT上,而CGI程序的标准输出STDOUT则被连到服务器的标准输入STDIN上。因此,CGI程序从标准输入读取信息(也就是从Web服务器的标准输出读信息),而它向标准输出写信息(也就是向Web服务器的标准输入写信息)。

Web服务器一般将客户机传送来的信息放在它的标准输出和相关环境变量中,而CGI程序则从环境变量和它的标准输入(也就是Web服务器的标准输出)获取所需的信息,程序的最终输出结果则被写向它的标准输出STDOUT(也就是Web服务器的标准输入)。Web服务器将从它的标准输入STDIN(也就是CGI程序的标准输出)获取CGI程序的输出结果并将它传送给客户机。客户机、Web服务器和CGI程序之间的信息交流。Web服务器就像是客户机和CGI程序间的中介。

Web服务器、CGI程序间的这种标准框架在Unix系统下和微软Windows环境中的字符方式下可以工作得非常好,因为此时系统产生的所有进程都可以存取标准输入和标准输出。但对于微软Windows图形方式下的程序就不行了,因为它们无法存取标准输入和标准输出。为了解决这一问题,微软在Win32系统中创建了另一类型的标准输入和标准输出,程序可以通过调用Win32 API函数来存取标准输入和标准输出,不过,这就意味着使用这类标准输入和标准输出的CGI程序都必须是32位的。

微软Windows环境下的其它一些Web服务器(例如Website)则使用另外一种特殊的技术(即利用INI文件)来实现Web服务器和CGI程序间的数据交流。采用这种被称为“Win-CGI”规范编写的CGI程序通常只能在部分Web服务器上运行。一般地,支持Win-CGI的Web服务器将客户端的输入以及有关的状态信息写入到一个INI文件中,而CGI程序则从该INI文件中获取相关信息,这类程序的执行效率没有标准CGI程序高。

在进行CGI编程时,只要使CGI程序从标准输入和环境变量中获取客户机提供的信息,并将要传送给客户机的输出结果写入标准输出,剩下的信息传递工作将由Web服务器自动完成。CGI只是规定了一个标准的接口规范,只要遵守这个标准规范,程序开发人员就可以利用各种编程工具(如Perl、C、FORTRAN、Visual Basic等)进行CGI编程了。考虑到VisualBasic的强大的数据库处理能力、客户机/服务器模式的编程能力以及字符串处理能力,所以本文主要向大家介绍如何使用VB编写标准的CGI程序。

一、输入输出的处理

一个CGI程序被激活以后,它首先要做的事情就是确定系统平台、Web服务器和客户端浏览器的状态信息以及客户端用户的输入数据。此外,它还必须能够将相关信息传送给客户端,否则它将一事无成。这些操作都是通过存取环境变量和标准输入输出来完成的。用VB编写的CGI程序通过调用函数Environ( 来获取相关环境变量的值。存取标准输入输出就要在程序中使用Win32API函数GetStdHandle( )、ReadFile( )和WriteFile( ),在使用这些函数时首先必须在程序中声明它们,写声明语句时可以借助于VB提供的API文本查看器。

以下的CGI程序说明了在VB-CGI程序中如何处理环境变量和标准输入输出。该CGI程序非常简单,可将标准输入中的信息不经任何处理就返回给客户端,它可被任何表单用POST方法激活:

Declare Function GetStdHandle Lib "kernel32" (ByVal nStdHandle As Long) As Long

Declare Function ReadFile Lib "kernel32" (ByVal hFile As Long, lpBuffer As Any, ByVal nNumberOfBytesToRead As Long, lpNumberOfBytesRead As Long,lpOverlapped As Any) As Long

Declare Function WriteFile Lib "kernel32" (ByVal hFile As Long, ByVal lpBuffer As String,ByVal nNumberOfBytesToWrite As Long, lpNumberOfBytesWritten As Long,lpOverlapped As Any) As Long

Public Const STD_INPUT_HANDLE = -10&

Public Const STD_OUTPUT_HANDLE = -11&

Public Const FILE_BEGIN = 0&

Public hStdIn As Long ' 标准输入文件句柄

Public hStdOut As Long ' 标准输出文件句柄

Sub Main()

Dim CGI_ContentLength As String

Dim CGI_QueryString As String

Dim lContentLength As Long ' 标准输入中的字符串的长度

Dim sBuff As String ' 用于存储标准输入中的字符串

Dim lBytesRead As Long ' 实际读入的字符个数

Dim rc As Long

Dim sFormData As String

'调用系统函数生成标准输入输出文件句柄

hStdIn = GetStdHandle(STD_INPUT_HANDLE)

hStdOut = GetStdHandle(STD_OUTPUT_HANDLE)

'获取环境变量CONTENT_LENGTH的值,并将它转换为整型

CGI_ContentLength = Environ("CONTENT_LENGTH")

lContentLength = Val(CGI_ContentLength)

sBuff = String(lContentLength, Chr$(0))

'从标准输入中读数据

rc = ReadFile(hStdIn, ByVal sBuff, lContentLength, lBytesRead, ByVal 0&)

sFormData = Left$(sBuff, lBytesRead)

OutPut "Content-type: text/html" & vbCrLf

OutPut "<HTML><HEAD>"

OutPut "<TITLE>表单传送数据的方法POST </TITLE></HEAD>"

OutPut "<BODY><H3>表单传送数据的方法POST</H3> "

OutPut "<P>本CGI 程序使用Visual Basic编制! "

OutPut "<P>POST方法传送的数据: "

OutPut "<P>" & sBuff

OutPut "</BODY></HTML>"

End Sub

Sub OutPut(s As String) ' 定义一个向标准输出写信息的函数

Dim lBytesWritten As Long

s = s & vbCrLf

WriteFile hStdOut, s, Len(s), lBytesWritten, ByVal 0&

End Sub

一般地,用VB编译生成的CGI程序不能正确处理中文信息。这主要表现在CGI程序向STDOUT输出的中文在Web页面上无法正确显示,可通过在该中文字符串后跟着输出一些空格来解决这个问题。当使用HTML标识符<P>、</P>对Web页面进行排版时,浏览器在显示该Web页面时会吃掉多余的空格而只保留一个。在这种情况下,这些空格对Web页面的外观基本上没有什么影响。如果使用HTML标识符<PRE>、</PRE>对Web页面进行排版,则由于空格不能被浏览器吃掉,所以Web页面的外观将会受到较大的影响。不过,这时可用HTML的表格<table>、<P>来代替<PRE>对Web页面进行排版。

注意:整个CGI程序的主体必须放在MAIN()函数中。

二、URL译码与解码

由于Web服务器和浏览器不能正确处理一些特殊的字符,Web服务器和浏览器之间可能会因此而产生某种程度的误会,所以在数据被传送之前,浏览器都要对表单内客户输入的数据中的特殊字符进行URL译码。

例如,Web系统用“=”分解表单各元素的NAME和VALUE属性,用“&”分解不同表单元素的输入数据。如果在表单的输入数据中包含这些特殊的字符,并且表单的数据在传送给Web服务器前不作任何处理,则Web服务器将无法知道哪一个“=”、“&”是用户输入的,哪一个是浏览器加上的。在由表单属性ACTION定义的URL中,也可能会出现一些特殊的字符,当在CGI程序的名称和路径信息(Path Information)中出现“=”、“&”和“?”时,都会影响数据的正确传送。

URL译码(URL Encoding)就是将Web服务器所不能正确处理的特殊字符转换成它的十六进制数的形式,比如将“%”转换成“%25”、“=”转换成“%3D”等等。这些特殊的字符通常被称作Web系统的保留字符。在Web系统上无论是用GET方法还是用POST方法传送的数据都要进行URL译码。CGI程序要想处理表单传送来的数据,还必须对浏览器URL译码过的数据进行解码。因此,理解URL译码对于我们进行CGI编程是非常重要的。URL译码一般包括以下步骤:

1、浏览器将所传送的数据根据表单所包含的元素分解成“NAME=VALUE”形式,NAME和VALUE分别是表单元素的属性。其中,VALUE属性中存储客户机在表单中输入的数据:如果客户机没有输入数据,则VALUE存储的是表单定义的缺省值;如果缺省值也没有定义,则VALUE值为空。

2、代表表单中各元素的各个“NAME=VALUE”对被浏览器用“&”连接起来。

3、VALUE属性中存放的数据若含有空格,则被转换成“+”。

4、URL和输入数据中所包含的Web系统的保留字符必须被译码成其十六进制数形式。

5、被译码后的字符被表示成一个“%”和它们的十六进制数形式(即%HH)。

CGI程序从环境变量“QUERY_STRING”或标准输入中读入的数据是经过浏览器URL译码过的,故在使用这些数据以前还必须对它们进行URL解码。解码的目的是将数据还原成客户端用户在Web页面上输入时的形式。本文已经介绍了URL译码过程,URL解码过程与它正好相反,它一般包括以下步骤:

1、从浏览器用GET或POST方法所传送来的数据中找出代表各个表单元素所储存数据的“NAME=VALUE”对。

2、VALUE属性中所存放的数据若含有“+”,则被转换成空格。

3、将VALUE属性中所存放的数据的十六进制数“%HH”转换成相应的字符。

Web系统将汉字当成特殊的字符,对它也要进行URL译码。对于一个特殊的单字节字符(比如“/”),浏览器通常将它译码成十六进制数的形式(比如%2F),“%”表示它后面跟的是两位十六进制数。当VB程序对其进行处理时调用Chr$函数就可以将其恢复为原貌。而一个汉字则被浏览器译码成四位十六进制数(比如%D5%C5)。如果CGI程序还像以前那样分别调用Chr(D5)和Chr(C5),则由于D5、C5都不是正常的单字节十六进制数码,故Chr函数返回空,汉字将无法正确还原。正确的做法应该是将有关汉字的四位十六进制数一起传给函数Chr(如Chr(D5C5)),此时汉字才能被正确还原。

因此,可以让CGI程序对四位连续的十六进制数一起进行译码,以便使汉字能够被正确还原。但在这种情况下,当客户端用户输入了两个连续的Web系统保留字符时,CGI程序又可能把它们当成汉字来处理。这时可以让CGI程序在需要对四位连续的十六进制数进行译码时首先检查前面两位是否为Web系统的保留字符,如果是则仍然按照单字节的字符处理。不过如果客户端用户在表单内填写了很多汉字,则CGI程序的负担将会大大加重。事实上,在大多数情况下,客户端用户很少会使用两个连续的Web系统的保留字符,所以可以只让CGI程序对最容易出现的情形如“://”(当客户端用户在表单中输入某一URL时会出现这种情况)进行检查,本文下节提供的函数UrlDecode( )可以实现对汉字和Web系统保留字符的URL解码。

三、CGI编程实例

本节将用VB编写一个处理主页客户留言簿的CGI程序。除了要调用本文前面所介绍的Win32API函数外,程序中还调用了Win32API函数GetTempFileName()来获得一个唯一的临时文件名。程序中的函数UrlDecode()用来对客户端的输入进行URL译码。函数GetCgiValue()则用来分解字符串,根据表单元素的NAME属性获取其VALUE值,并调用UrlDecode()函数对其进行URL译码。

本程序要求在留言簿文件guests.html中使用一个定位串“<! ENDHEAD >”,将文件的开始部分和具体的客户留言部分分开。CGI程序将在“<! ENDHEAD >”所在的位置插入客户新的留言。guests.html应具有如下所示的样式:

<html>

<head><title>DHTML Zone </title></head>

<body bgcolor="#FFFFFF" text="#00000" vlink="#990000" link="#333399">

<! ENDHEAD >

<!---客户的留言部分从这开始-->

<P>……………………….

<!---客户的留言部分结束于此-->

</body></html>

这种样式将保证最后的留言出现在留言簿的最前面。如果要想使最后的留言出现在留言簿的最后面,则只需将留言簿文件中的定位字符串“<! ENDHEAD >”移到留言簿文件中客户留言部分和HTML文件结尾部分之间的位置就行了。

整个程序的完整代码如下所示:

'guestbook.bas

Declare Function GetStdHandle Lib "kernel32" (ByVal nStdHandle As Long) As Long

Declare Function ReadFile Lib "kernel32" (ByVal hFile As Long, lpBuffer As Any,ByVal nNumberOfBytesToRead As Long, lpNumberOfBytesRead As Long, lpOverlapped As Any) As Long

Declare Function WriteFile Lib "kernel32" (ByVal hFile As Long,ByVal lpBuffer As String, ByVal nNumberOfBytesToWrite As Long,lpNumberOfBytesWritten As Long, lpOverlapped As Any) As Long

Declare Function GetTempFileName Lib "kernel32" Alias "GetTempFileNameA"(ByVal lpszPath As String, ByVal lpPrefixString As String, ByVal wUnique As Long, ByVal lpTempFileName As String) As Long

Public Const STD_INPUT_HANDLE = -10&

Public Const STD_OUTPUT_HANDLE = -11&

Public Const FILE_BEGIN = 0&

Public hStdIn As Long ' 标准输入文件句柄

Public hStdOut As Long ' 标准输出文件句柄

Public sFormData As String ' 用于存储没有经过URL译码的用户输入数据

Public lContentLength As Long

Public CGI_RequestMethod As String

Sub Main()

Dim CGI_ContentLength As String, CGI_QueryString As String, sBuff As String, chinesetail As String

Dim lBytesRead As Long, rc As Long,I As Long

Dim sEmail As String, sName As String, sURL As String, sfrom As String, tempstring As String

Dim sComment As String, tempFileName As String, guestbook As String

'CGI程序的初始化工作

hStdIn = GetStdHandle(STD_INPUT_HANDLE)

hStdOut = GetStdHandle(STD_OUTPUT_HANDLE)

CGI_RequestMethod = Environ("REQUEST_METHOD")

CGI_QueryString = Environ("QUERY_STRING")

CGI_ContentLength = Environ("CONTENT_LENGTH")

lContentLength = Val(CGI_ContentLength)

sBuff = String(lContentLength, Chr$(0))

OutPut "Content-type: text/html" & vbCrLf ' 输出MIME类型

OutPut "<FONT SIZE=""+2"">"

If CGI_RequestMethod = "POST" Then

sBuff = String(lContentLength, Chr$(0))

rc = ReadFile(hStdIn, ByVal sBuff, lContentLength, lBytesRead, ByVal 0&)

sFormData = Left$(sBuff, lBytesRead)

ElseIf CGI_RequestMethod = "GET" Then

sFormData = CGI_QueryString

Else

OutPut "Unknow Form Method !"

End If

'为了在页面上正确显示中文,生成一个空格串以获取客户端用户的输入

chinesetail = String(400, " ")

sName = GetCgiValue("name")

sEmail = GetCgiValue("email")

sURL = GetCgiValue("URL")

sfrom = GetCgiValue("from")

sComment = GetCgiValue("URL_Comment")

'对客户端用户的输入进行检查

If Len(sName) = 0 Then

OutPut "<P>非常抱歉!您还没有填写姓名!" & chinesetail

Exit Sub

End If

If Len(sComment) = 0 Then

OutPut "<P>非常抱歉!您还没有提出建议!" & chinesetail

Exit Sub

End If

'获取唯一的临时文件名和留言簿文件并打开它们

tempFileName = TempFile("c:\windows\temp", "gbk")

guestbook = "e:\netscape\server\docs\guests.html"

Open tempFileName For Output As #1

Open guestbook For Input As #2

'本循环体用于将留言簿中字符串"<! ENDHEAD >"前面的内容写入临时文件

Do

Line Input #2, tempstring

Print #1, tempstring

Loop While tempstring <> "<! ENDHEAD >" And Not EOF(2)

'向临时文件中插入客户端用户的留言

Print #1, "<hr>" & vbCrLf

Print #1, "<ul>" & vbCrLf

Print #1, "<li><b>留言时间</b>:" & Date$ & " " & Time$ & vbCrLf

Print #1, "<li><b>姓名: </b>" & sName & vbCrLf

If Len(sEmail) <> 0 Then

Print #1, "<li><b>E-mail: </b><a href=""mailto:" & sEmail & """ >" & sEmail & "</a>" & vbCrLf

End If

If Len(sURL) <> 0 Then

Print #1, "<li><b>我的主页: </b> <a href=""" & sURL & """ >" & sURL & "</a>" & vbCrLf

End If

If Len(sfrom) <> 0 Then

Print #1, "<li><b>我来自: </b>" & sfrom & vbCrLf

End If

Print #1, "<li><b>我的建议: </b>" & vbCrLf

Print #1, sComment & vbCrLf

Print #1, "</ul>" & vbCrLf

'本循环体用于将留言簿剩余的东西写入留言簿

Do

Line Input #2, tempstring

Print #1, tempstring

Loop While Not EOF(2)

Close #1

Close #2

Kill guestbook '删除旧的留言簿

Name tempFileName As guestbook '将临时文件改成新的留言簿

OutPut "<P>非常感谢您的留言!" & chinesetail

OutPut "<P>欢迎您经常光顾本主页!" & chinesetail

OutPut "</FONT>"

End Sub

Sub OutPut(s As String) ' 本子程序用于向标准输出写信息

Dim lBytesWritten As Long

s = s & vbCrLf

WriteFile hStdOut, s, Len(s), lBytesWritten, ByVal 0&

End Sub

' 本子程序可以获取表单上某一元素的数据

Public Function GetCgiValue(cgiName As String) As String

Dim delim2 As Long ' position of "="

Dim delim1 As Long ' position of "&"

Dim n As Integer

Dim pointer1 As Long,pointer2 As Long,length As Long,length1 As Long

Dim tmpstring1 As String,tmpstring2 As String

pointer1 = 1

pointer2 = 1

delim2 = InStr(pointer2, sFormData, "=")

pointer2 = delim2 + 1

Do

length = delim2 - pointer1

tmpstring1 = Mid(sFormData, pointer1, length)

delim1 = InStr(pointer1, sFormData, "&")

pointer1 = delim1 + 1

length1 = delim1 - pointer2

If delim1 = 0 Then length1 = lContentLength + 1 - pointer2

If tmpstring1 = cgiName Then

tmpstring2 = Mid$(sFormData, pointer2, length1)

GetCgiValue = UrlDecode(tmpstring2)

Exit Do

End If

If delim1 = 0 Then

Exit Do

End If

delim2 = InStr(pointer2, sFormData, "=")

pointer2 = delim2 + 1

Loop

End Function

' 本函数可以对用户输入的数据进行URL解码

Public Function UrlDecode(ByVal sEncoded As String) As String

Dim pointer As Long'sEncoded position pointer

Dim pos As Long'position of InStr target

Dim temp As String

If sEncoded = "" Then Exit Function

pointer = 1

'本循环体用于将"+"转换成空格

Do

pos = InStr(pointer, sEncoded, "+")

If pos = 0 Then Exit Do

Mid$(sEncoded, pos, 1) = " "

pointer = pos + 1

Loop

pointer = 1

'本循环体用于将%XX转换成字符。对于两个连续的%XX,如果第一个%XX 不是某些特指的Web系统保留字符,将把它们转换成汉字

Do

pos = InStr(pointer, sEncoded, "%")

If pos = 0 Then Exit Do

temp = Chr$("&H" & (Mid$(sEncoded, pos + 1, 2)))

If Mid(sEncoded, pos + 3, 1) = "%" And (temp <> ":") And (temp <> "/") _

And (temp <> "(") And (temp <> ")") And (temp <> ".") And (temp <> ",") _

And (temp <> ";") And (temp <> "%") Then

Mid$(sEncoded, pos, 2) = Chr$("&H" & (Mid$(sEncoded, pos + 1, 2)) _

& (Mid$(sEncoded, pos + 4, 2)))

sEncoded = Left$(sEncoded, pos) & Mid$(sEncoded, pos + 6)

pointer = pos + 1

Else

Mid$(sEncoded, pos, 1) = temp

sEncoded = Left$(sEncoded, pos) & Mid$(sEncoded, pos + 3)

pointer = pos + 1

End If

Loop

UrlDecode = sEncoded

Exit Function

End Function

'本函数可以获得一个唯一的临时文件名

Public Function TempFile(sPath As String, sPrefix As String) As String

Dim x As Long,rc As Long

TempFile = String(127, Chr$(0))

rc = GetTempFileName(sPath, sPrefix, ByVal 0&, TempFile)

x = InStr(TempFile, Chr$(0))

If x > 0 Then TempFile = Left$(TempFile, x - 1)

End Function

CGI程序guestbook.bas所要处理的表单如下所示:

<html><head><title>贵宾留言簿</title></head>

<body>

<h3>贵宾留言簿测试</h3>

<form action="/cgi-bin/guest.exe" method="post">

您的姓名: <input type="text" name="name"><br>

您的Email信箱: <input type="text" name="email"><br>

您的主页的URL: <input type="text" name="URL"><br>

您的建议:<br> <textarea name="URL_Comment" rows=4 cols=30></textarea><br>

您来自: <input type="text" name="from"><br>

<input type="submit" value=" 留言 ">

</form>

</body></html>

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有