-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathcheck_jsonp.go
186 lines (173 loc) · 5.07 KB
/
check_jsonp.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
package scripts
import (
"github.com/robertkrimen/otto/ast"
"github.com/robertkrimen/otto/parser"
"io/ioutil"
"net/http"
"net/url"
"regexp"
)
//基于AST的JSONP劫持漏洞检测
//1.解析js路径,检查query所有key是否满足正则 (?m)(?i)(callback)|(jsonp)|(^cb$)|(function)
//2.referer配置为同域,请求js获取响应
//3.js响应生成AST,如果满足
// a) Callee.Name == callback函数名
// b) 递归遍历AST 获取所有的字段和对应的value
// c) 字段为敏感字段(满足正则(?m)(?i)(uid)|(userid)|(user_id)|(nin)|(name)|(username)|(nick)),且value不为空
//4.替换Referer后再请求一次,重新验证步骤3
//
//调用方式
//入参:js路径
//返回:是否存在漏洞,err
//result, err := CheckSenseJsonp("http://127.0.0.1/jsonp_env/getUser.php?id=1&jsoncallback=callbackFunction")
func CheckSenseJsonp(jsUrl string)(bool, error){
queryMap, domainString, err := UrlParser(jsUrl)
if err != nil{
return false, err
}
isCallback, callbackFuncName, err := CheckJSIsCallback(queryMap)
if isCallback{
// referer: host 请求
normalRespContent, err := GetJsResponse(jsUrl, domainString)
if err != nil{
return false, err
}
isJsonpNormal , err := CheckJsRespAst(normalRespContent, callbackFuncName)
if err != nil{
return false, err
}
// 如果包含敏感字段 将 referer 置空 再请求一次
if isJsonpNormal{
noRefererContent, err := GetJsResponse(jsUrl, "")
if err != nil{
return false, err
}
isJsonp , err := CheckJsRespAst(noRefererContent, callbackFuncName)
if err != nil{
return false, err
}
return isJsonp, nil
}
}
return false, nil
}
func UrlParser(jsUrl string)(url.Values, string, error){
urlParser, err := url.Parse(jsUrl)
if err != nil{
return nil, "", err
}
// 拼接原始referer
domainString := urlParser.Scheme + "://" + urlParser.Host
return urlParser.Query(), domainString, nil
}
func CheckJSIsCallback(queryMap url.Values) (bool,string, error){
var re = regexp.MustCompile(`(?m)(?i)(callback)|(jsonp)|(^cb$)|(function)`)
for k, v :=range queryMap {
regResult := re.FindAllString(k, -1)
if len(regResult) > 0 && len(v)>0 {
return true, v[0], nil
}
}
return false, "",nil
}
func CheckIsSensitiveKey(key string) (bool,error) {
var re = regexp.MustCompile(`(?m)(?i)(uid)|(userid)|(user_id)|(nin)|(name)|(username)|(nick)`)
regResult := re.FindAllString(key, -1)
if len(regResult) > 0 {
return true, nil
}
return false, nil
}
func GetJsResponse(jsUrl string, referer string) (string, error) {
req, err := http.NewRequest("GET", jsUrl, nil)
if err != nil {
return "", nil
}
req.Header.Set("Referer", referer)
resp, err := goHTTPClient.Do(req)
if err != nil {
return "", err
}
defer resp.Body.Close()
body, err := ioutil.ReadAll(resp.Body)
if err != nil {
return "", err
}
if resp.StatusCode == 200 {
return string(body), nil
}
return "", nil
}
func CheckJsRespAst(content string, funcName string) (bool,error){
// 解析js语句,生成 *ast.Program 或 ErrorList
program, err := parser.ParseFile(nil, "", content, 0)
if err != nil{
return false, err
}
if len(program.Body) > 0 {
statement := program.Body[0]
expression := statement.(*ast.ExpressionStatement).Expression
expName := expression.(*ast.CallExpression).Callee.(*ast.Identifier).Name
// 表达式中函数名与query函数名不一致 直接返回false
if funcName != expName{
return false, err
}
argList := expression.(*ast.CallExpression).ArgumentList
for _, arg := range argList{
result := DealAstExpression(arg)
if result != true{
continue
}
return result, nil
}
}
//ast树为空 直接返回
return false, nil
}
func DealAstExpression(expression ast.Expression)bool{
objectLiteral, isObjectLiteral := expression.(*ast.ObjectLiteral)
if isObjectLiteral{
values := objectLiteral.Value
for _, value := range values{
result := DealAstProperty(value)
if result != true{
continue
}
return result
}
}
return false
}
func DealAstProperty(value ast.Property)bool{
secondLevelValue := value.Value
// 表达式中是数组/对象的 递归
objectLiteral, isObjectLiteral := secondLevelValue.(*ast.ObjectLiteral)
arrayLiteral, isArrayLiteral := secondLevelValue.(*ast.ArrayLiteral)
stringLiteral, isStringLiteral := secondLevelValue.(*ast.StringLiteral)
numberLiteral, isNumberLiteral := secondLevelValue.(*ast.NumberLiteral)
if isObjectLiteral {
thirdLevelValue := objectLiteral.Value
for _, v := range thirdLevelValue {
DealAstProperty(v)
}
} else if isArrayLiteral {
thirdLevelValue := arrayLiteral.Value
for _, v := range thirdLevelValue {
DealAstExpression(v)
}
} else if isStringLiteral{
// 表达式中value为字符串/数字的 才会检测key value
thirdLevelValue := stringLiteral.Value
isSensitiveKey, _ := CheckIsSensitiveKey(value.Key)
if isSensitiveKey && thirdLevelValue != ""{
return true
}
} else if isNumberLiteral {
thirdLevelValue := numberLiteral.Value
isSensitiveKey, _ := CheckIsSensitiveKey(value.Key)
if isSensitiveKey && thirdLevelValue != 0{
return true
}
}
return false
}