时间轴
演示案例
TP 框架-开发-配置架构&路由&MVC 模型
1、配置架构-导入使用
data:image/s3,"s3://crabby-images/230f3/230f3cf0e112679b90383b026fc705220f370dfa" alt=""
data:image/s3,"s3://crabby-images/9f11b/9f11b8a8933c4c0f14c73cd1b4819e0da9cd1480" alt=""
data:image/s3,"s3://crabby-images/93cff/93cff39b425f56afa79d14ebc78b032478201d26" alt=""
data:image/s3,"s3://crabby-images/74b5e/74b5eb1db16defdad159fd07774800d981d7fd52" alt=""
data:image/s3,"s3://crabby-images/11642/116429af127e4f2a0f4ac6488e063fc20cfceec7" alt=""
data:image/s3,"s3://crabby-images/2aba2/2aba29860a19ec1369b582df9fd88c614ef4b4a2" alt=""
data:image/s3,"s3://crabby-images/822fe/822fe96901283fc7edb5500482caa6fb827bcefd" alt=""
data:image/s3,"s3://crabby-images/0e376/0e3768c6704023849b5a29c1c501b27bc669f3e2" alt=""
data:image/s3,"s3://crabby-images/51825/51825ef51d74294726517b63bbc269802a016de9" alt=""
data:image/s3,"s3://crabby-images/944bc/944bc95ee05ab47be1b630d4f188a29b1fab629d" alt=""
2、路由访问-URL 访问
data:image/s3,"s3://crabby-images/e35f1/e35f10fb145e4f5fdda04e9ed6c76fa83bb48cc6" alt=""
data:image/s3,"s3://crabby-images/4b224/4b2247f7e9e0669fb78eafc739db629cb4c0004c" alt=""
data:image/s3,"s3://crabby-images/e50b2/e50b2edeb4e3592ad59b86293967f17509c7c557" alt=""
data:image/s3,"s3://crabby-images/04448/04448dc8f8a709ab8c1e49e7a0a6d91e898dfc6b" alt=""
非官方写法访问方式
当使用非官方的写法访问时,可以看到只能用?x=1来访问,而当使用/x/1时访问报错。
官方写法访问方式
当使用官方写法访问时,?name=12可以访问,/name/12也可以访问
<?php
namespace app\index\controller;
use think\Controller;
use think\Request;class Index extends Controller
{public function index(){return '123';}public function xiaodi(){return $this->request->param('name');}}
?>
MVC模型
对应model(模板) view(视图) controller(控制器)
其中核心代码文件在controller中
3、数据库操作-应用对象
连接数据库
首先在application下找到database.php文件
修改database(数据库名)为demo01,username为root,密码123456,端口号3306,其余根据自己需求修改。
查询数据库
在navicat中找到之前创建的news表格
在ThinkPHP—>application下新建一个test文件夹,并在这个文件夹下创建controller文件夹,其中包含Test.php
非官方写法
以之前的news.php为例
<?php
include 'config.php';
//读取news.html中的内容
$template=file_get_contents('news.html');$id=$_GET['id'] ?? '1';
$sql="select * from news where id=$id";
echo $sql;
$data=mysqli_query($con,$sql);
while($row=mysqli_fetch_row($data)){$page_title=$row[1];$heading=$row[2];$subheading=$row[3];$content=$row[4];$item=$row[5];
}$template=str_replace('{page_title}',$page_title,$template);
$template=str_replace('{heading}',$heading,$template);
$template=str_replace('{subheading}',$subheading,$template);
$template=str_replace('{content}',$content,$template);
$template=str_replace('{$item}',$item,$template);eval('?>'.$template);
//eval函数会将传递给它的字符串作为PHP代码执行,即将?>连接到$template的开头,再执行该字符串
?>
当执行?id=1 and 1=1时,可以看到这个值会被接收
当输入?id=2时会出现报错
官方写法
<?php
namespace app\Test\controller;
use think\Db;
use think\Controller;class Test extends Controller
{public function testsql(){//使用tp框架操作mysql数据库//SELECT * FROM `think_user` WHERE `id` = 1 LIMIT 1//规矩写法$id = request()->param('x');$data = Db::table('news')->where('id', $id)->find();return json($data);}
}
?>
当输入ip/index.php/test/test/testsql/x/1去查询这个数据库的时候,页面可以正常回显
并且可以发现,在/x/1后面输入任何东西都不会显示在页面上
结论
1.使用TP框架操作数据库时,默认是受到框架内置过滤保护的,而且方便开发。
2.原生态的数据库操作如果没有过滤就会受到SQL注入攻击。
4、文件上传操作-应用对象
首先在ThinkPHP—>public文件夹下新建一个upload.html,其中代码为
<form action="/index.php/test/test/upload" enctype="multipart/form-data" method="post"><input type="file" name="image" /> <br><input type="submit" value="上传" />
</form>
Test.php中代码改为
<?php
namespace app\Test\controller;
use think\Db;
use think\Controller;class Test extends Controller
{ public function testsql(){//使用tp框架操作mysql数据库//SELECT * FROM `think_user` WHERE `id` = 1 LIMIT 1//规矩写法
// $id=request()->param('x');
// $data=Db::table('news')->where('id',$id)->find();//原生写法 有安全隐患//$id=request()->param('x');//$data=Db::query("select * from news where id=$id");$username = request()->get('username/a');db('admin')->insert(['username' => $username]);return 'Update success';//return json($data);
}public function upload(){// 获取表单上传文件 例如上传了001.jpg$file = request()->file('image');//获取表单上传文件// 移动到框架应用根目录/uploads/ 目录下$info = $file->validate(['size'=>1567800,'ext'=>'jpg,png,gif'])->move( '../uploads');if($info){// 成功上传后 获取上传信息// 输出 jpgecho $info->getExtension();// 输出 20160820/42a79759f284b767dfcb2a0197904287.jpgecho $info->getSaveName();// 输出 42a79759f284b767dfcb2a0197904287.jpgecho $info->getFilename();}else{// 上传失败获取错误信息echo $file->getError();}}
}
在ThinkPHP下创建uploads文件夹,用来存储接收的文件
由于phpstudy中的根目录就是public文件夹,因此可以直接用ip/upload.php来访问
当尝试上传图片时,就会跳转到以下页面
而当上传txt、docx等文件时,就会提醒文件后缀不允许
5、前端页面渲染-MVC 模型
在ThinkPHP—>application—>index下创建view文件夹,再在view下创建index文件夹,其中包含index.html和edit.html。
edit.html代码如下:
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>sabiqudi</title>
</head>
<body>
kcnnqi
</body>
</html>
index.html代码如下:
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>{$name}</title>
</head>
<body>
{$email}
</body>
</html>
index.php代码如下:
<?php
namespace app\index\controller;
use think\Request;
use think\Controller;class Index extends Controller
{ public function index(){//return 123;$this->assign('name','ThinkPHP');$this->assign('email','thinkphp@qq.com');// 或者批量赋值$this->assign(['name' => 'ThinkPHP','email' => 'thinkphp@qq.com']);// 模板输出return $this->fetch('');//更改要渲染的页面
}
}
当index.php中为return $this->fetch('');时,默认渲染index.html,访问后页面如下:
当index.php中为return $this->fetch('edit');时,渲染edit.html,访问后页面如下:
TP 框架-安全-不安全写法&版本过滤绕过
1、内置代码写法
例子:不合规的代码写法-TP5-自写
data:image/s3,"s3://crabby-images/1f871/1f8717be326c20e41bb5e8bc99ba7f7274d65fa9" alt=""
data:image/s3,"s3://crabby-images/1242f/1242fe6a159a01e49e28cfcdb04269dceffc0d78" alt=""
data:image/s3,"s3://crabby-images/ea70d/ea70d2791e8c3068d238f923244b1f1f69244e27" alt=""
data:image/s3,"s3://crabby-images/51074/510748696a21a9786a321f0916ece20b055240a6" alt=""
data:image/s3,"s3://crabby-images/a670b/a670b9fa7dd9a7abd7d6150da61a0ff2d8fd3ac9" alt=""
data:image/s3,"s3://crabby-images/1ceeb/1ceeb54992465987f91810b9b6b752250e814eee" alt=""
data:image/s3,"s3://crabby-images/ebe5f/ebe5f8ed9f9642322917c979249445a8797ddd35" alt=""
2、框架版本安全
例子 1:写法内置安全绕过-TP5-SQL 注入
参考文章:https://www.cnblogs.com/Yhck/p/15808056.html
可以看到,由于这个漏洞出现的版本是5.0.13-5.0.15,5.1.0-5.1.5,所以后续会用到5.0.14版本进行演示。下面先用5.0.22版本做一个对比。
由于5.0.22版本不在漏洞适用范围之内,可以看到页面返回为null
之后用5.0.14版本演示,先在phpstudy中更改路径为5.0.14中的public文件夹
为了看是否指向正确,可以让代码输出123来检验
<?php
namespace app\index\controller;class Index
{public function index(){echo 123;}
}
之后也是配置数据库
将index.php中的代码改为
<?php
namespace app\index\controller;class Index
{public function index(){$username = request()->get('username/a');db('users')->where("id")->update(['username'=> $username]);}
}
按照下面的命令访问
发现虽然报错,但是直接爆出了数据库名demo01
例子 2:内置版本安全漏洞-TP5-代码执行
data:image/s3,"s3://crabby-images/0d55e/0d55ed2ccd3f9ed8e31df16577a03bc78e1de044" alt=""
data:image/s3,"s3://crabby-images/d1c49/d1c498e396a4c1eac92eec42db5df336156942b4" alt=""
逻辑越权(类似于linux中的用户等级)
如下图,假设管理员的uid=1,普通会员的uid=100。当使用select * from users where username=‘admin’来取出uid结果时,如果uid=1,则展示管理员页面。
data:image/s3,"s3://crabby-images/7057d/7057df2e8837439e2337a1aa6e651d3dd3107644" alt=""
data:image/s3,"s3://crabby-images/dba57/dba57dca20ff2aff14309f256f91935922aaa06b" alt=""