+ baidunavi
This commit is contained in:
168
platformapi/baidunavi/baidunavi.go
Normal file
168
platformapi/baidunavi/baidunavi.go
Normal file
@@ -0,0 +1,168 @@
|
|||||||
|
package baidunavi
|
||||||
|
|
||||||
|
import (
|
||||||
|
"crypto/md5"
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"net/url"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"git.rosy.net.cn/baseapi/platformapi"
|
||||||
|
"git.rosy.net.cn/baseapi/utils"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
signKey = "sn"
|
||||||
|
resultKey = "result"
|
||||||
|
prodURL = "http://api.map.baidu.com"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
StatusCodeSuccess = 0
|
||||||
|
StatusCodeInternalErr = 1 // 服务器内部错误
|
||||||
|
StatusCodeExceedDailyQuota = 301 // 永久配额超限,限制访问
|
||||||
|
StatusCodeExceedQuota = 302 // 天配额超限,限制访问
|
||||||
|
StatusCodeExceedDailyConcurrentQuota = 401 // 当前并发量已经超过约定并发配额,限制访问
|
||||||
|
StatusCodeExceedConcurrentQuota = 402 // 当前并发量已经超过约定并发配额,并且服务总并发量也已经超过设定的总并发配额,限制访问
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
CoordSysWGS84 = 1 // GPS设备获取的角度坐标,WGS84坐标
|
||||||
|
CoordSysGCJ02 = 3 // google地图、soso地图、aliyun地图、mapabc地图和amap地图所用坐标,国测局(GCJ02)坐标
|
||||||
|
CoordSysBaiDu = 5 // 百度地图采用的经纬度坐标
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
MaxCoordsConvBatchSize = 100
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
exceedLimitCodes = map[int]int{
|
||||||
|
StatusCodeExceedDailyQuota: 1,
|
||||||
|
StatusCodeExceedQuota: 1,
|
||||||
|
StatusCodeExceedDailyConcurrentQuota: 1,
|
||||||
|
StatusCodeExceedConcurrentQuota: 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
canRetryCodes = map[int]int{
|
||||||
|
StatusCodeInternalErr: 1,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
type Coordinate struct {
|
||||||
|
Lng float64 `json:"x"`
|
||||||
|
Lat float64 `json:"y"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type API struct {
|
||||||
|
client *http.Client
|
||||||
|
config *platformapi.APIConfig
|
||||||
|
ak string
|
||||||
|
sk string
|
||||||
|
}
|
||||||
|
|
||||||
|
func New(ak, sk string, config ...*platformapi.APIConfig) *API {
|
||||||
|
curConfig := platformapi.DefAPIConfig
|
||||||
|
if len(config) > 0 {
|
||||||
|
curConfig = *config[0]
|
||||||
|
}
|
||||||
|
return &API{
|
||||||
|
ak: ak,
|
||||||
|
sk: sk,
|
||||||
|
client: &http.Client{Timeout: curConfig.ClientTimeout},
|
||||||
|
config: &curConfig,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *API) signParams(apiStr string, mapData map[string]interface{}) string {
|
||||||
|
keys := make([]string, 0)
|
||||||
|
for k := range mapData {
|
||||||
|
if k != signKey {
|
||||||
|
keys = append(keys, k)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
sort.Strings(keys)
|
||||||
|
|
||||||
|
strList := []string{}
|
||||||
|
for _, k := range keys {
|
||||||
|
strList = append(strList, k+"="+url.QueryEscape(fmt.Sprint(mapData[k])))
|
||||||
|
}
|
||||||
|
finalStr := "/" + apiStr + "?" + strings.Join(strList, "&") + a.sk
|
||||||
|
// baseapi.SugarLogger.Debugf("sign str:%v", finalStr)
|
||||||
|
finalStr = url.QueryEscape(finalStr)
|
||||||
|
return fmt.Sprintf("%x", md5.Sum([]byte(finalStr)))
|
||||||
|
}
|
||||||
|
|
||||||
|
func genGetURL(baseURL, apiStr string, params map[string]interface{}) string {
|
||||||
|
keys := make([]string, 0)
|
||||||
|
for k := range params {
|
||||||
|
if k != signKey {
|
||||||
|
keys = append(keys, k)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
sort.Strings(keys)
|
||||||
|
|
||||||
|
strList := []string{}
|
||||||
|
for _, k := range keys {
|
||||||
|
strList = append(strList, k+"="+url.QueryEscape(fmt.Sprint(params[k])))
|
||||||
|
}
|
||||||
|
strList = append(strList, signKey+"="+url.QueryEscape(fmt.Sprint(params[signKey])))
|
||||||
|
queryString := "?" + strings.Join(strList, "&")
|
||||||
|
if apiStr != "" {
|
||||||
|
return baseURL + "/" + apiStr + queryString
|
||||||
|
}
|
||||||
|
return baseURL + queryString
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *API) AccessAPI(apiStr string, params map[string]interface{}) (retVal interface{}, err error) {
|
||||||
|
apiStr += "/"
|
||||||
|
params2 := utils.MergeMaps(utils.Params2Map("ak", a.ak, "output", "json"), params)
|
||||||
|
params2[signKey] = a.signParams(apiStr, params2)
|
||||||
|
|
||||||
|
err = platformapi.AccessPlatformAPIWithRetry(a.client,
|
||||||
|
func() *http.Request {
|
||||||
|
request, _ := http.NewRequest(http.MethodGet, genGetURL(prodURL, apiStr, params2), nil)
|
||||||
|
return request
|
||||||
|
},
|
||||||
|
a.config,
|
||||||
|
func(response *http.Response, bodyStr string, jsonResult1 map[string]interface{}) (errLevel string, err error) {
|
||||||
|
if jsonResult1 == nil {
|
||||||
|
return platformapi.ErrLevelRecoverableErr, fmt.Errorf("mapData is nil")
|
||||||
|
}
|
||||||
|
status := int(utils.MustInterface2Int64(jsonResult1["status"]))
|
||||||
|
if status == StatusCodeSuccess {
|
||||||
|
retVal = jsonResult1[resultKey]
|
||||||
|
return platformapi.ErrLevelSuccess, nil
|
||||||
|
}
|
||||||
|
newErr := utils.NewErrorIntCode(utils.Interface2String(jsonResult1["message"]), status)
|
||||||
|
if _, ok := exceedLimitCodes[status]; ok {
|
||||||
|
return platformapi.ErrLevelExceedLimit, newErr
|
||||||
|
} else if _, ok := canRetryCodes[status]; ok {
|
||||||
|
return platformapi.ErrLevelRecoverableErr, newErr
|
||||||
|
} else {
|
||||||
|
return platformapi.ErrLevelCodeIsNotOK, newErr
|
||||||
|
}
|
||||||
|
})
|
||||||
|
return retVal, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *API) BatchCoordinateConvert(coords []*Coordinate, fromCoordSys, toCoordSys int) (outCoords []*Coordinate, err error) {
|
||||||
|
if fromCoordSys == toCoordSys {
|
||||||
|
return coords, nil
|
||||||
|
}
|
||||||
|
var coordsStrList []string
|
||||||
|
for _, v := range coords {
|
||||||
|
coordsStrList = append(coordsStrList, fmt.Sprintf("%.6f,%.6f", v.Lng, v.Lat))
|
||||||
|
}
|
||||||
|
params := map[string]interface{}{
|
||||||
|
"coords": strings.Join(coordsStrList, ";"),
|
||||||
|
"from": fromCoordSys,
|
||||||
|
"to": toCoordSys,
|
||||||
|
}
|
||||||
|
result, err := a.AccessAPI("geoconv/v1", params)
|
||||||
|
if err == nil {
|
||||||
|
err = utils.Map2StructByJson(result, &outCoords, false)
|
||||||
|
}
|
||||||
|
return outCoords, err
|
||||||
|
}
|
||||||
41
platformapi/baidunavi/baidunavi_test.go
Normal file
41
platformapi/baidunavi/baidunavi_test.go
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
package baidunavi
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"git.rosy.net.cn/baseapi"
|
||||||
|
"git.rosy.net.cn/baseapi/utils"
|
||||||
|
|
||||||
|
"go.uber.org/zap"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
api *API
|
||||||
|
sugarLogger *zap.SugaredLogger
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
logger, _ := zap.NewDevelopment()
|
||||||
|
sugarLogger = logger.Sugar()
|
||||||
|
baseapi.Init(sugarLogger)
|
||||||
|
|
||||||
|
api = New("eL94zToVOdGDTkNQxV8dnEQ1ZRcB2UKb", "ZG0OOpOsOVURUwAkkmoHQFKRCbzn0zGb")
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestBatchCoordinateConvert(t *testing.T) {
|
||||||
|
result, err := api.BatchCoordinateConvert([]*Coordinate{
|
||||||
|
&Coordinate{
|
||||||
|
Lng: 104.057367,
|
||||||
|
Lat: 30.694686,
|
||||||
|
},
|
||||||
|
&Coordinate{
|
||||||
|
Lng: 104.057367,
|
||||||
|
Lat: 30.694686,
|
||||||
|
},
|
||||||
|
}, CoordSysGCJ02, CoordSysBaiDu)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("TestCoordinateConvert failed with error:%v", err)
|
||||||
|
} else {
|
||||||
|
t.Log(utils.Format4Output(result, false))
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user