时间:2021-07-01 10:21:17 帮助过:28人阅读
在大三上的课程中,海涛老师要求项目中运用rest进行数据采集。我两眼懵逼,啥是rest呀?然后就去网上找了学习资料。然后之后就着手开始自己写一个rest api。为什么要自己写呢?因为我没用框架
,第一次使用php做网站,我想先打好基础再考虑高层次的东西,就没有用框架。其他人用的诸如laravel之类的PHP框架自己本身会带rest机制。对于我一个没用框架的孩子,只能自己默默写了。
当然,有参考例子哦http://www.gen-x-design.com/archives/create-a-rest-api-with-php/,不过这个例子一点都不全面,我自己补充了很多。
setMethod($request_method);
// set the raw data, so we can access it if needed (there may be
// other pieces to your requests)
$return_obj->setRequestVars($data);
if(isset($data['data']))
{
// translate the JSON to an Object for use however you want
$return_obj->setData(json_decode($data['data']));
}
return $return_obj;
}
public static function sendResponse($status = 200, $body = '', $content_type = 'text/html')
{
$status_header = 'HTTP/1.1 ' . $status . ' ' . RestUtils::getStatusCodeMessage($status);
// set the status
header($status_header);
// set the content type
header('Content-type: ' . $content_type);
// pages with body are easy
if($body != '')
{
// send the body
echo $body;
exit;
}
// we need to create the body if none is passed
else
{
// create some body messages
$message = '';
// this is purely optional, but makes the pages a little nicer to read
// for your users. Since you won't likely send a lot of different status codes,
// this also shouldn't be too ponderous to maintain
switch($status)
{
case 401:
$message = 'You must be authorized to view this page.';
break;
case 404:
$message = 'The requested URL ' . $_SERVER['REQUEST_URI'] . ' was not found.';
break;
case 500:
$message = 'The server encountered an error processing your request.';
break;
case 501:
$message = 'The requested method is not implemented.';
break;
}
// servers don't always have a signature turned on (this is an apache directive "ServerSignature On")
$signature = ($_SERVER['SERVER_SIGNATURE'] == '') ? $_SERVER['SERVER_SOFTWARE'] . ' Server at ' . $_SERVER['SERVER_NAME'] . ' Port ' . $_SERVER['SERVER_PORT'] : $_SERVER['SERVER_SIGNATURE'];
// this should be templatized in a real-world solution
$body = '
' . $status . ' ' . RestUtils::getStatusCodeMessage($status) . '
' . RestUtils::getStatusCodeMessage($status) . '
' . $message . '
' . $signature . '
';
echo $body;
exit;
}
}
public static function getStatusCodeMessage($status)
{
// these could be stored in a .ini file and loaded
// via parse_ini_file()... however, this will suffice
// for an example
$codes = Array(
100 => 'Continue',
101 => 'Switching Protocols',
200 => 'OK',
201 => 'Created',
202 => 'Accepted',
203 => 'Non-Authoritative Information',
204 => 'No Content',
205 => 'Reset Content',
206 => 'Partial Content',
300 => 'Multiple Choices',
301 => 'Moved Permanently',
302 => 'Found',
303 => 'See Other',
304 => 'Not Modified',
305 => 'Use Proxy',
306 => '(Unused)',
307 => 'Temporary Redirect',
400 => 'Bad Request',
401 => 'Unauthorized',
402 => 'Payment Required',
403 => 'Forbidden',
404 => 'Not Found',
405 => 'Method Not Allowed',
406 => 'Not Acceptable',
407 => 'Proxy Authentication Required',
408 => 'Request Timeout',
409 => 'Conflict',
410 => 'Gone',
411 => 'Length Required',
412 => 'Precondition Failed',
413 => 'Request Entity Too Large',
414 => 'Request-URI Too Long',
415 => 'Unsupported Media Type',
416 => 'Requested Range Not Satisfiable',
417 => 'Expectation Failed',
500 => 'Internal Server Error',
501 => 'Not Implemented',
502 => 'Bad Gateway',
503 => 'Service Unavailable',
504 => 'Gateway Timeout',
505 => 'HTTP Version Not Supported'
);
return (isset($codes[$status])) ? $codes[$status] : '';
}
}
class RestRequest
{
private $request_vars;
private $data;
private $http_accept;
private $method;
public function __construct()
{
$this->request_vars = array();
$this->data = '';
$this->http_accept = (strpos($_SERVER['HTTP_ACCEPT'], 'json')) ? 'json' : 'xml';
$this->method = 'get';
}
public function setData($data)
{
$this->data = $data;
}
public function setMethod($method)
{
$this->method = $method;
}
public function setRequestVars($request_vars)
{
$this->request_vars = $request_vars;
}
public function getData()
{
return $this->data;
}
public function getMethod()
{
return $this->method;
}
public function getHttpAccept()
{
return $this->http_accept;
}
public function getRequestVars()
{
return $this->request_vars;
}
}
$data = RestUtils::processRequest();
switch($data->getMethod())
{
case 'get':
// print_r($data->getRequestVars());
$infoType=$data->getRequestVars()['infoType'];
if (array_key_exists("username",$data->getRequestVars())) {
$username=$data->getRequestVars()['username'];
if($data->getHttpAccept() == 'json'){
if($infoType=="health"){
$healthService=new HealthService();
$result=$healthService->getUserHealth($username);
if($result==""){
RestUtils::sendResponse(404, "Not Found");
}
else{
RestUtils::sendResponse(200, json_encode($result), 'application/json');
}
}
}
else if ($data->getHttpAccept() == 'xml'){
if($infoType=="health"){
$healthService=new HealthService();
$result=$healthService->getUserHealth($username);
if($result==""){
RestUtils::sendResponse(404, "NOT FOUND");
}
else{
$health=new HealthXML();
RestUtils::sendResponse(200, $health->create_xml($result),'application/xml');
}
}
}
}
break;
case 'post':
$infoType=$data->getRequestVars()['infoType'];
if($infoType=="health"){
$username=$data->getRequestVars()['username'];
$height=$data->getRequestVars()['height'];
$weight=$data->getRequestVars()['weight'];
$date=$data->getRequestVars()['date'];
$healthService=new HealthService();
$ID=$healthService->addHealthWithDate($username, $height, $weight, $date);
$health=new HealthXML();
$data_array = array(
array(
'ID' => $ID,
)
);
RestUtils::sendResponse(201, $health->create_IDxml($data_array));
}
break;
case 'put':
// print_r($data->getRequestVars());
$infoType=$data->getRequestVars()['infoType'];
// echo $infoType;
// echo $infoType=="health"?"true":"false";
if($infoType=="health"){
$ID=$data->getRequestVars()['ID'];
$username=$data->getRequestVars()['username'];
$height=$data->getRequestVars()['height'];
$weight=$data->getRequestVars()['weight'];
$date=$data->getRequestVars()['date'];
$healthService=new HealthService();
$result=$healthService->modifyHealthWithDate($ID, $username, $height, $weight, $date);
if($result=="false"){
RestUtils::sendResponse(400, "Bad Request");
}
else{
$health=new HealthXML();
RestUtils::sendResponse(200, $health->create_xml($result),'application/xml');
}
}
break;
case 'delete':
$infoType=$data->getRequestVars()['infoType'];
if($infoType=="health"){
$ID=$data->getRequestVars()['ID'];
$healthService=new HealthService();
$result=$healthService->deleteHealth($ID);
if($result==false){
RestUtils::sendResponse(404, "Not Found");
}
else{
RestUtils::sendResponse(204, "No Content");
}
}
break;
}
class HealthXML{
// 创建XML单项
function create_item($ID_data,$username_data, $height_data , $weight_data, $dateTime_data)
{
$item = "- \n";
$item .= "
" . $ID_data . " \n";
$item .= "" . $username_data . " \n";
$item .= "" . $height_data . " \n";
$item .= "" . $weight_data . " \n";
$item .= " " . $dateTime_data . " \n";
$item .= " \n";
return $item;
}
function create_xml($data_array){
$title_size = 1;
$xml = "\n";
$xml .= "\n";
foreach ($data_array as $data) {
$xml .= $this->create_item($data['ID'],$data['username'], $data['height'], $data['weight'],$data['dateTime']);
}
$xml .= "\n";
return $xml;
}
// 创建XML单项
function create_IDitem($ID_data)
{
$item = "- \n";
$item .= "
" . $ID_data . " \n";
$item .= " \n";
return $item;
}
function create_IDxml($data_array){
$title_size = 1;
$xml = "\n";
$xml .= "\n";
foreach ($data_array as $data) {
$xml .= $this->create_IDitem($data['ID']);
}
$xml .= "\n";
return $xml;
}
}
?>难点是对于http请求头的解析真是让我吐血啊,那个时候我还不知道chrome可以看请求内容,我的做法是简陋地输出来看。之后解析时狂补了一通正则表达式,最后用正则顺利解析了。
我这不能API直接用啊,里面有我网站里分析健康数据的例子,你们也可以看到,后半段大量地出现“health",就是我的网站中的获取健康数据的例子啊。
因为时间紧急,所以只用半天写出来的API,大概是又简陋又有很多不足的吧。不过想到这个项目最终检查时有那么多同学没有实现rest,而我没有用框架自己学习rest,实现了一个自己的api,也是挺开心的。
以上就介绍了自己创建一个Rest API,包括了方面的内容,希望对PHP教程有兴趣的朋友有所帮助。