畢設(shè)和OAuth協(xié)議相關(guān),而要理解OAuth協(xié)議就必須理解HTTP GET/POST方法。因此研究了一下如何使用Web API或MVC構(gòu)造POST報(bào)文并實(shí)現(xiàn)客戶端與服務(wù)器端的交互。
我使用的工具是Visual Studio 2013 + Web API 2 + MVC 5。
在兩個(gè)不同的VS2013實(shí)例中分別新建兩個(gè)Web項(xiàng)目,都選擇空模板,其中一個(gè)命名為Client,采用MVC架構(gòu),另一個(gè)命名為Server,采用Web API架構(gòu)。
這里需要兩個(gè)不同的VS2013實(shí)例是為了能使兩個(gè)IIS Express服務(wù)在同一臺(tái)機(jī)器上同時(shí)運(yùn)行。
我們先來(lái)看看客戶端:
由于只是個(gè)Demo,因此我們只在客戶端中新建一個(gè)空的MVC控制器,將其命名為HomeController。它會(huì)自帶一個(gè)Index()方法。我們?cè)谶@里面寫(xiě)好構(gòu)造請(qǐng)求并發(fā)送請(qǐng)求的代碼:
namespace Client.Controllers
{
public class HomeController : Controller
{
//
// GET: /Home/
public ActionResult Index()
{
string url = ""; // 這里我們還不知道服務(wù)器的url,因此留空
#region 構(gòu)造POST請(qǐng)求
HttpWebRequest request = WebRequest.Create(url) as HttpWebRequest;
request.Method = "POST";
request.ContentType = "application/x-www-form-urlencoded";
request.Host = "localhost:14340";
string body = "client_id=123&redirect_uri="+url+"&response_type=code";
byte[] bodyBytes = Encoding.UTF8.GetBytes(body);
Stream requestStream = request.GetRequestStream();
requestStream.Write(bodyBytes,0,bodyBytes.Length);
#endregion
#region 發(fā)送請(qǐng)求并取得響應(yīng)
// 必須使用HttpWebRequest.GetResponse()方法,請(qǐng)求才能被發(fā)送并取得響應(yīng)
HttpWebResponse response = request.GetResponse() as HttpWebResponse;
StreamReader sr = new StreamReader(response.GetResponseStream());
ViewBag.Response = sr.ReadToEnd();
#endregion
return View();
}
}
}
接下來(lái)我們構(gòu)造服務(wù)器端:
在Server項(xiàng)目中,新建一個(gè)Web API 2控制器。命名隨意(我這里命名為ApiTestController),在控制器中寫(xiě)入一個(gè)方法Post,返回HttpResponseMessage類型。
本來(lái)我一直以為POST的參數(shù)可以在Request屬性中獲取,但是仔細(xì)檢查了Request屬性的所有屬性和方法,包括擴(kuò)展方法,都沒(méi)有找到可以直接讀取POST報(bào)文請(qǐng)求體的屬性或方法。查閱資料得知,如果要在POST請(qǐng)求中獲取請(qǐng)求體的內(nèi)容,就必須將參數(shù)封裝在一個(gè)類中。于是我們建立一個(gè)類RequestArgs如下:
namespace Server.Models
{
///
/// 對(duì)于POST請(qǐng)求,必須對(duì)請(qǐng)求體的所有參數(shù)建立一個(gè)模型類。
///
public class RequestArgs
{
public string client_id { get; set; }
public string redirect_uri { get; set; }
public string response_type { get; set; }
}
}
這里的參數(shù)和上文客戶端Action方法中的請(qǐng)求報(bào)文內(nèi)容必須保持名稱一致(包括大小寫(xiě))
之后我們?cè)賹?xiě)入控制器的POST方法如下:
namespace Server.Controllers
{
public class ApiTestController : ApiController
{
public HttpResponseMessage Post(RequestArgs args)
{
string client_id = args.client_id;
string redirect_uri = args.redirect_uri;
string response_type = args.response_type;
return Request.CreateResponse(HttpStatusCode.OK, "I have received your message. Client_id = "+client_id+", RedirectUri = "+redirect_uri+"response_type = "+response_type);
}
}
}
這里我們直接返回了報(bào)文的內(nèi)容,而在實(shí)際的項(xiàng)目中,我們可以自己對(duì)傳入的參數(shù)進(jìn)行處理。
之后我們編譯運(yùn)行服務(wù)器端(最好是按F5進(jìn)入調(diào)試模式運(yùn)行)。可以看到出現(xiàn)的Web頁(yè)面似乎是一個(gè)錯(cuò)誤頁(yè)面,但是不用管它,我們的服務(wù)器已經(jīng)開(kāi)始運(yùn)行了。
記下服務(wù)器的地址(一般也就是端口號(hào)會(huì)變化),將這個(gè)服務(wù)器的地址填入客戶端的url變量。
編譯運(yùn)行客戶端,如果出現(xiàn)返回的信息(這里是“I have received your message”和參數(shù)信息),說(shuō)明我們的請(qǐng)求成功了。
更多信息請(qǐng)查看IT技術(shù)專欄