Tang Cheng | 26b9115 | 2015-07-22 12:26:52 +0800 | [diff] [blame] | 1 | package swservice |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 2 | |
| 3 | import ( |
| 4 | "bytes" |
| 5 | "crypto/hmac" |
| 6 | "crypto/sha1" |
Tang Cheng | 26b9115 | 2015-07-22 12:26:52 +0800 | [diff] [blame] | 7 | "encoding/hex" |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 8 | "encoding/json" |
qiaowei | ca037fa | 2015-07-10 18:31:53 +0800 | [diff] [blame] | 9 | "errors" |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 10 | "fmt" |
| 11 | log "github.com/Sirupsen/logrus" |
qiaowei | ca037fa | 2015-07-10 18:31:53 +0800 | [diff] [blame] | 12 | "io/ioutil" |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 13 | "net" |
| 14 | "net/http" |
| 15 | "net/url" |
Tang Cheng | f16121a | 2015-07-27 13:46:14 +0800 | [diff] [blame^] | 16 | "strconv" |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 17 | "time" |
| 18 | ) |
Tang Cheng | 32e3256 | 2015-07-08 17:08:03 +0800 | [diff] [blame] | 19 | |
| 20 | func dailTimeout(network, addr string) (net.Conn, error) { |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 21 | // default_timeout := time.Duration(theSession.DefaultTimeout) * time.Second |
| 22 | default_timeout := time.Duration(3) * time.Second |
Tang Cheng | 32e3256 | 2015-07-08 17:08:03 +0800 | [diff] [blame] | 23 | return net.DialTimeout(network, addr, default_timeout) |
| 24 | } |
| 25 | |
| 26 | type WebSession struct { |
| 27 | AppId string |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 28 | TermId string |
| 29 | Appsecret string |
Tang Cheng | 32e3256 | 2015-07-08 17:08:03 +0800 | [diff] [blame] | 30 | BaseUrl string |
| 31 | DefaultTimeout int |
| 32 | session_key string |
| 33 | } |
| 34 | |
Tang Cheng | f16121a | 2015-07-27 13:46:14 +0800 | [diff] [blame^] | 35 | func safe_get_json_int(value interface{}) int { |
| 36 | if value == nil { |
| 37 | return 0 |
| 38 | } |
| 39 | s := fmt.Sprintf("%v", value) |
| 40 | if i, err := strconv.Atoi(s); err != nil { |
| 41 | return 0 |
| 42 | } else { |
| 43 | return i |
| 44 | } |
| 45 | } |
| 46 | |
| 47 | type ServiceResponse struct { |
| 48 | RetCode int |
| 49 | RetMsg string |
| 50 | Result map[string]interface{} |
| 51 | } |
| 52 | |
| 53 | func NewServiceResponseFromJson(json_data interface{}) *ServiceResponse { |
| 54 | if json_data == nil { |
| 55 | return nil |
| 56 | } |
| 57 | res := &ServiceResponse{} |
| 58 | res.Result = json_data.(map[string]interface{}) |
| 59 | res.RetCode = safe_get_json_int(res.Result["retcode"]) |
| 60 | res.RetMsg = res.GetStrValue("retmsg") |
| 61 | return res |
| 62 | } |
| 63 | func (r *ServiceResponse) GetIntValue(name string) int { |
| 64 | return safe_get_json_int(r.Result[name]) |
| 65 | } |
| 66 | |
| 67 | func (r *ServiceResponse) GetStrValue(name string) string { |
| 68 | if s, ok := r.Result[name]; ok { |
| 69 | return fmt.Sprintf("%v", s) |
| 70 | } else { |
| 71 | return "" |
| 72 | } |
| 73 | } |
| 74 | |
| 75 | func (r *ServiceResponse) GetFloatValue(name string) float64 { |
| 76 | if s, ok := r.Result[name]; ok { |
| 77 | t := fmt.Sprintf("%v", s) |
| 78 | if f, err := strconv.ParseFloat(t, 64); err != nil { |
| 79 | return 0.0 |
| 80 | } else { |
| 81 | return f |
| 82 | } |
| 83 | } else { |
| 84 | return 0.0 |
| 85 | } |
| 86 | } |
| 87 | |
Tang Cheng | 32e3256 | 2015-07-08 17:08:03 +0800 | [diff] [blame] | 88 | func (w *WebSession) DoGet(uri string, params map[string]string) (*http.Response, error) { |
| 89 | transport := http.Transport{Dial: dailTimeout} |
| 90 | |
| 91 | client := http.Client{Transport: &transport} |
| 92 | |
| 93 | full_url := w.BaseUrl + uri |
| 94 | |
| 95 | vl := url.Values{} |
Tang Cheng | 32e3256 | 2015-07-08 17:08:03 +0800 | [diff] [blame] | 96 | |
| 97 | if params != nil { |
| 98 | for k, v := range params { |
| 99 | vl.Add(k, v) |
| 100 | } |
| 101 | } |
Tang Cheng | 32e3256 | 2015-07-08 17:08:03 +0800 | [diff] [blame] | 102 | full_url = full_url + "?" + vl.Encode() |
Tang Cheng | f16121a | 2015-07-27 13:46:14 +0800 | [diff] [blame^] | 103 | // fmt.Printf("%v\n", full_url) |
Tang Cheng | 32e3256 | 2015-07-08 17:08:03 +0800 | [diff] [blame] | 104 | return client.Get(full_url) |
| 105 | } |
| 106 | |
| 107 | func (w *WebSession) GetTimestamp() string { |
| 108 | t := time.Now() |
| 109 | return fmt.Sprintf("%04d%02d%02d%02d%02d%02d", t.Year(), t.Month(), t.Day(), |
| 110 | t.Hour(), t.Minute(), t.Second()) |
| 111 | } |
| 112 | |
Tang Cheng | f16121a | 2015-07-27 13:46:14 +0800 | [diff] [blame^] | 113 | func (w *WebSession) SignWithKey(key, message string) string { |
| 114 | mac := hmac.New(sha1.New, []byte(key)) |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 115 | mac.Write([]byte(message)) |
| 116 | res := mac.Sum(nil) |
Tang Cheng | 26b9115 | 2015-07-22 12:26:52 +0800 | [diff] [blame] | 117 | return hex.EncodeToString(res) |
Tang Cheng | 32e3256 | 2015-07-08 17:08:03 +0800 | [diff] [blame] | 118 | } |
Tang Cheng | f16121a | 2015-07-27 13:46:14 +0800 | [diff] [blame^] | 119 | func (w *WebSession) Sign(message string) string { |
| 120 | return w.SignWithKey(w.Appsecret, message) |
| 121 | } |
Tang Cheng | 32e3256 | 2015-07-08 17:08:03 +0800 | [diff] [blame] | 122 | |
Tang Cheng | f16121a | 2015-07-27 13:46:14 +0800 | [diff] [blame^] | 123 | func (w *WebSession) NewClient(timeout int) *http.Client { |
| 124 | transport := http.Transport{Dial: func(network, addr string) (net.Conn, error) { |
| 125 | default_timeout := time.Duration(timeout) * time.Second |
| 126 | return net.DialTimeout(network, addr, default_timeout) |
| 127 | }} |
| 128 | |
| 129 | return &http.Client{Transport: &transport} |
| 130 | } |
Tang Cheng | 26b9115 | 2015-07-22 12:26:52 +0800 | [diff] [blame] | 131 | func (w *WebSession) DoPost(uri string, param map[string]string) (*http.Response, error) { |
Tang Cheng | 32e3256 | 2015-07-08 17:08:03 +0800 | [diff] [blame] | 132 | transport := http.Transport{Dial: dailTimeout} |
| 133 | client := &http.Client{Transport: &transport} |
Tang Cheng | 26b9115 | 2015-07-22 12:26:52 +0800 | [diff] [blame] | 134 | param["app_id"] = w.AppId |
Tang Cheng | f16121a | 2015-07-27 13:46:14 +0800 | [diff] [blame^] | 135 | // param["term_id"] = w.TermId |
Tang Cheng | 26b9115 | 2015-07-22 12:26:52 +0800 | [diff] [blame] | 136 | param["sign_method"] = "HMAC" |
| 137 | param["session_key"] = w.session_key |
Tang Cheng | 32e3256 | 2015-07-08 17:08:03 +0800 | [diff] [blame] | 138 | ts := w.GetTimestamp() |
Tang Cheng | 26b9115 | 2015-07-22 12:26:52 +0800 | [diff] [blame] | 139 | param["timestamp"] = ts |
| 140 | param["sign"] = w.Sign(w.AppId + w.TermId + w.session_key + ts) |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 141 | |
Tang Cheng | 32e3256 | 2015-07-08 17:08:03 +0800 | [diff] [blame] | 142 | full_url := w.BaseUrl + uri |
Tang Cheng | f16121a | 2015-07-27 13:46:14 +0800 | [diff] [blame^] | 143 | // log.Debugf("Url=%v", full_url) |
Tang Cheng | 26b9115 | 2015-07-22 12:26:52 +0800 | [diff] [blame] | 144 | data, err := json.Marshal(param) |
| 145 | if err != nil { |
| 146 | return nil, err |
| 147 | } |
| 148 | var r *http.Response |
| 149 | r, err = client.Post(full_url, "application/json", bytes.NewReader(data)) |
Tang Cheng | 32e3256 | 2015-07-08 17:08:03 +0800 | [diff] [blame] | 150 | if err != nil || r.StatusCode != 200 { |
| 151 | log.Errorf("Status=%v, err=%v", r, err) |
| 152 | } |
| 153 | return r, err |
| 154 | } |
| 155 | |
| 156 | func (w *WebSession) Auth() error { |
qiaowei | ca037fa | 2015-07-10 18:31:53 +0800 | [diff] [blame] | 157 | token, err := w.getAuthToken() |
| 158 | if err != nil { |
| 159 | return err |
| 160 | } |
| 161 | err = w.getAppAccessKey(token) |
| 162 | if err != nil { |
| 163 | return err |
| 164 | } |
Tang Cheng | 32e3256 | 2015-07-08 17:08:03 +0800 | [diff] [blame] | 165 | return nil |
| 166 | } |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 167 | |
| 168 | func NewSession(appid, appsecret, termid, baseurl string, timeout int) *WebSession { |
| 169 | return &WebSession{ |
| 170 | AppId: appid, |
| 171 | Appsecret: appsecret, |
| 172 | TermId: termid, |
| 173 | BaseUrl: baseurl, |
| 174 | DefaultTimeout: timeout, |
| 175 | } |
| 176 | } |
| 177 | |
qiaowei | ca037fa | 2015-07-10 18:31:53 +0800 | [diff] [blame] | 178 | func (w *WebSession) getAuthToken() (string, error) { |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 179 | type FormJson struct { |
| 180 | AppId string `json:"app_id"` |
| 181 | TermId string `json:"term_id"` |
| 182 | AccessToken string `json:"access_token"` |
| 183 | } |
| 184 | |
Tang Cheng | 26b9115 | 2015-07-22 12:26:52 +0800 | [diff] [blame] | 185 | uri := fmt.Sprintf("/authservice/getauth/%v/getaccesstoken", w.AppId) |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 186 | |
| 187 | params := make(map[string]string) |
| 188 | params["term_id"] = w.TermId |
| 189 | r, err := w.DoGet(uri, params) |
| 190 | |
| 191 | if err != nil || r.StatusCode != 200 { |
| 192 | fmt.Println("here") |
| 193 | // log.Errorf("Status = %v, err = %v\n", r.StatusCode, err) |
qiaowei | ca037fa | 2015-07-10 18:31:53 +0800 | [diff] [blame] | 194 | return "", errors.New("请求失败") |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 195 | } |
| 196 | |
qiaowei | ca037fa | 2015-07-10 18:31:53 +0800 | [diff] [blame] | 197 | body, err := ioutil.ReadAll(r.Body) |
| 198 | r.Body.Close() |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 199 | |
qiaowei | ca037fa | 2015-07-10 18:31:53 +0800 | [diff] [blame] | 200 | s := &FormJson{} |
| 201 | err = json.Unmarshal(body, &s) |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 202 | if err != nil { |
| 203 | log.Errorf("json unmarshal err %v", err) |
qiaowei | ca037fa | 2015-07-10 18:31:53 +0800 | [diff] [blame] | 204 | return "", errors.New("解析失败") |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 205 | } |
qiaowei | ca037fa | 2015-07-10 18:31:53 +0800 | [diff] [blame] | 206 | return s.AccessToken, nil |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 207 | } |
qiaowei | ca037fa | 2015-07-10 18:31:53 +0800 | [diff] [blame] | 208 | func (w *WebSession) getAppAccessKey(token string) error { |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 209 | type FormJson struct { |
qiaowei | ca037fa | 2015-07-10 18:31:53 +0800 | [diff] [blame] | 210 | AppId string `json:"app_id"` |
| 211 | TermId string `json:"term_id"` |
| 212 | SessionKey string `json:"session_key"` |
| 213 | CardKey string `json:"card_key"` |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 214 | } |
| 215 | |
Tang Cheng | 26b9115 | 2015-07-22 12:26:52 +0800 | [diff] [blame] | 216 | uri := fmt.Sprintf("/authservice/getauth/%v", w.AppId) |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 217 | |
| 218 | params := make(map[string]string) |
| 219 | params["term_id"] = w.TermId |
qiaowei | ca037fa | 2015-07-10 18:31:53 +0800 | [diff] [blame] | 220 | params["access_token"] = token |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 221 | params["timestamp"] = w.GetTimestamp() |
| 222 | params["v"] = "1" |
qiaowei | ca037fa | 2015-07-10 18:31:53 +0800 | [diff] [blame] | 223 | params["sign"] = w.Sign(token + params["timestamp"]) |
| 224 | params["sign_method"] = "HMAC" |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 225 | |
qiaowei | ca037fa | 2015-07-10 18:31:53 +0800 | [diff] [blame] | 226 | r, err := w.DoGet(uri, params) |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 227 | if err != nil || r.StatusCode != 200 { |
qiaowei | ca037fa | 2015-07-10 18:31:53 +0800 | [diff] [blame] | 228 | log.Errorf(" err = %v\n", err) |
| 229 | return err |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 230 | } |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 231 | |
qiaowei | ca037fa | 2015-07-10 18:31:53 +0800 | [diff] [blame] | 232 | body, err := ioutil.ReadAll(r.Body) |
| 233 | r.Body.Close() |
qiaowei | ca037fa | 2015-07-10 18:31:53 +0800 | [diff] [blame] | 234 | |
| 235 | s := &FormJson{} |
| 236 | err = json.Unmarshal(body, &s) |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 237 | if err != nil { |
| 238 | log.Errorf("json unmarshal err %v", err) |
| 239 | } |
qiaowei | ca037fa | 2015-07-10 18:31:53 +0800 | [diff] [blame] | 240 | w.session_key = s.SessionKey |
| 241 | return nil |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 242 | } |
Tang Cheng | 26b9115 | 2015-07-22 12:26:52 +0800 | [diff] [blame] | 243 | |
| 244 | func (w *WebSession) CallYKTApi(request *MessageWriter) (*MessageReader, error) { |
| 245 | call_data := request.Serialize() |
| 246 | params := make(map[string]string) |
| 247 | params["funcdata"] = call_data |
| 248 | r, err := w.DoPost("/ecardservice/ecardapi", params) |
| 249 | |
Tang Cheng | f16121a | 2015-07-27 13:46:14 +0800 | [diff] [blame^] | 250 | if err != nil { |
Tang Cheng | 26b9115 | 2015-07-22 12:26:52 +0800 | [diff] [blame] | 251 | log.Errorf(" err = %v\n", err) |
| 252 | return nil, err |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 253 | } |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 254 | |
Tang Cheng | f16121a | 2015-07-27 13:46:14 +0800 | [diff] [blame^] | 255 | if r.StatusCode != 200 { |
| 256 | return nil, errors.New(fmt.Sprintf("Request StatusCode:%v", r.StatusCode)) |
| 257 | } |
| 258 | |
Tang Cheng | 26b9115 | 2015-07-22 12:26:52 +0800 | [diff] [blame] | 259 | body, err := ioutil.ReadAll(r.Body) |
| 260 | r.Body.Close() |
| 261 | return NewMessageReader(body), nil |
zongqiang.zhang | b8ab9b4 | 2015-07-10 17:45:20 +0800 | [diff] [blame] | 262 | } |
Tang Cheng | f16121a | 2015-07-27 13:46:14 +0800 | [diff] [blame^] | 263 | |
| 264 | func (w *WebSession) CallService(path string, params map[string]interface{}, |
| 265 | sign_field []string, timeout int) (response *ServiceResponse, err error) { |
| 266 | |
| 267 | return w.CallService2(path, params, timeout, sign_field...) |
| 268 | } |
| 269 | |
| 270 | func (w *WebSession) CallService2(path string, params map[string]interface{}, timeout int, |
| 271 | sign_field ...string) (response *ServiceResponse, err error) { |
| 272 | client := w.NewClient(timeout) |
| 273 | err = nil |
| 274 | params["app_id"] = w.AppId |
| 275 | params["term_id"] = w.TermId |
| 276 | ts := w.GetTimestamp() |
| 277 | params["timestamp"] = ts |
| 278 | |
| 279 | vl := &url.Values{} |
| 280 | for k, v := range params { |
| 281 | vl.Set(k, fmt.Sprintf("%v", v)) |
| 282 | } |
| 283 | |
| 284 | sign_data := "" |
| 285 | for _, k := range sign_field { |
| 286 | if v, ok := params[k]; ok { |
| 287 | sign_data += fmt.Sprintf("%v", v) |
| 288 | } |
| 289 | } |
| 290 | sign_data += ts + w.session_key |
| 291 | vl.Set("sign_method", "HMAC") |
| 292 | log.Debugf("Sign: key[%v] data[%v]\n", w.session_key, sign_data) |
| 293 | vl.Set("sign", w.SignWithKey(w.Appsecret, sign_data)) |
| 294 | |
| 295 | full_url := w.BaseUrl + path |
| 296 | log.Debugf("CallService: %v\n", full_url) |
| 297 | var r *http.Response |
| 298 | r, err = client.Post(full_url, "application/x-www-form-urlencoded", |
| 299 | bytes.NewReader([]byte(vl.Encode()))) |
| 300 | if err != nil { |
| 301 | log.Errorf("Status=%v, err=%v", r, err) |
| 302 | return |
| 303 | } |
| 304 | if r.StatusCode != 200 { |
| 305 | log.Errorf("Request Error %v\n", r.StatusCode) |
| 306 | err = errors.New(fmt.Sprintf("Request Error, StatusCode : %v", r.StatusCode)) |
| 307 | return |
| 308 | } |
| 309 | body, err := ioutil.ReadAll(r.Body) |
| 310 | r.Body.Close() |
| 311 | var s interface{} |
| 312 | err = json.Unmarshal(body, &s) |
| 313 | if err != nil { |
| 314 | log.Errorf("json unmarshal err %v", err) |
| 315 | return |
| 316 | } |
| 317 | response = NewServiceResponseFromJson(s) |
| 318 | return |
| 319 | } |