func.go 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704
  1. package envitem
  2. import (
  3. "bytes"
  4. "compress/gzip"
  5. "context"
  6. "crypto/md5"
  7. "database/sql/driver"
  8. "encoding/hex"
  9. "encoding/json"
  10. "errors"
  11. "fmt"
  12. "io"
  13. "metawant.greentech.com.cn/gaoyagang/gt-common/httplib"
  14. "net"
  15. "net/http"
  16. "net/url"
  17. "strconv"
  18. "strings"
  19. "sync"
  20. "time"
  21. )
  22. func SetOptions(options Options) {
  23. snapUrl = fmt.Sprintf("http://%s/api/v1/plc-current", options.GtServerIp)
  24. ctlUrl = fmt.Sprintf("http://%s/api/v1/plc/set-var-values", options.GtServerIp)
  25. plcItemUrl = fmt.Sprintf("http://%s/api/v1/plc-list/", options.GtServerIp)
  26. if options.Cache != nil {
  27. cache = options.Cache
  28. }
  29. fetchMultiItem = options.FetchMultiItem
  30. adjustValue = options.AdjustValue
  31. plcItemSecret = options.PlcItemSecret
  32. }
  33. func (m MultiEnvItem) GetProjectId() int64 {
  34. for _, item := range m {
  35. return item.ProjectId
  36. }
  37. return 0
  38. }
  39. func (m MultiEnvItem) getItemNames() []string {
  40. items := make([]string, 0)
  41. for _, item := range m {
  42. items = append(items, item.Item)
  43. }
  44. return items
  45. }
  46. func (m MultiEnvItem) FillCurrentValue() error {
  47. datas := make([]*ItemValueResp, 0)
  48. var err error
  49. if fetchMultiItem {
  50. datas, err = m.getCurrentData()
  51. } else {
  52. datas, err = m.getCurrentDataByOne()
  53. }
  54. if err != nil {
  55. return err
  56. }
  57. // 先做一个item -> key的映射
  58. ikm := make(map[string]string)
  59. for s, item := range m {
  60. ikm[item.Item] = s
  61. }
  62. for _, data := range datas {
  63. if data != nil {
  64. if k, ok := ikm[data.ItemName]; ok {
  65. m[k].Value = data.Val
  66. m[k].Htime = data.HTime
  67. _ = m[k].setPrevValue()
  68. }
  69. }
  70. }
  71. return nil
  72. }
  73. func (m MultiEnvItem) GetItemFloat64Value(key string) float64 {
  74. if envItem, ok := m[key]; ok {
  75. return envItem.GetItemFloat64Val()
  76. }
  77. return 0
  78. }
  79. func (m MultiEnvItem) GetItemInt64Value(key string) int64 {
  80. if envItem, ok := m[key]; ok {
  81. return envItem.GetItemInt64Val()
  82. }
  83. return 0
  84. }
  85. func (m MultiEnvItem) GetItemStringValue(key string) string {
  86. if envItem, ok := m[key]; ok {
  87. return envItem.GetItemStringVal()
  88. }
  89. return ""
  90. }
  91. func (m MultiEnvItem) GetItemHtime(key string) *time.Time {
  92. if envItem, ok := m[key]; ok {
  93. return envItem.GetItemHtime()
  94. }
  95. return nil
  96. }
  97. func (m MultiEnvItem) FindString() map[string]string {
  98. a := make(map[string]string)
  99. for _, item := range m {
  100. a[item.Item] = item.GetItemStringVal()
  101. }
  102. return a
  103. }
  104. func (m MultiEnvItem) FindPrevString() map[string]string {
  105. a := make(map[string]string)
  106. for _, item := range m {
  107. a[item.Item] = item.GetItemPrevStringVal()
  108. }
  109. return a
  110. }
  111. func (m MultiEnvItem) ClearValues() {
  112. for _, item := range m {
  113. item.clearValue()
  114. }
  115. }
  116. func (m MultiEnvItem) getCurrentDataByOne() ([]*ItemValueResp, error) {
  117. datas := make([]*ItemValueResp, len(m))
  118. i := 0
  119. var wg sync.WaitGroup
  120. wg.Add(len(m))
  121. for _, item := range m {
  122. go func(index int, one *EnvItem) {
  123. defer wg.Done()
  124. datas[index], _ = one.getCurrentData()
  125. }(i, item)
  126. i++
  127. }
  128. wg.Wait()
  129. return datas, nil
  130. }
  131. func (m MultiEnvItem) getCurrentData() ([]*ItemValueResp, error) {
  132. req := httplib.Post(snapUrl)
  133. data := make([]*ItemValueReq, 1)
  134. data[0] = &ItemValueReq{
  135. DeviceItems: strings.Join(m.getItemNames(), ","),
  136. ProjectId: m.GetProjectId(),
  137. }
  138. jsonBytes, err := json.Marshal(data)
  139. if err != nil {
  140. return nil, err
  141. }
  142. req.Body(jsonBytes)
  143. req.SetTimeout(time.Millisecond * 2000)
  144. r, err := req.Response()
  145. if err != nil {
  146. return nil, err
  147. }
  148. defer r.Body.Close()
  149. if r.StatusCode == 200 {
  150. resp, err := req.Bytes()
  151. if err != nil {
  152. return nil, err
  153. }
  154. res := &ItemValueResps{}
  155. err = json.Unmarshal(resp, res)
  156. if err != nil {
  157. return nil, err
  158. }
  159. if len(res.Data) == 0 {
  160. return nil, errors.New("not found envitem's value")
  161. }
  162. return res.Data, nil
  163. }
  164. return nil, errors.New(fmt.Sprintf("request statusCode: %d", r.StatusCode))
  165. }
  166. func (e *EnvItem) getCurrentData() (*ItemValueResp, error) {
  167. req := httplib.Post(snapUrl)
  168. data := make([]*ItemValueReq, 1)
  169. data[0] = &ItemValueReq{
  170. DeviceItems: e.Item,
  171. ProjectId: e.ProjectId,
  172. }
  173. jsonBytes, err := json.Marshal(data)
  174. if err != nil {
  175. return nil, err
  176. }
  177. req.Body(jsonBytes)
  178. req.SetTimeout(time.Millisecond * 2000)
  179. r, err := req.Response()
  180. if err != nil {
  181. return nil, err
  182. }
  183. defer r.Body.Close()
  184. if r.StatusCode == 200 {
  185. resp, err := req.Bytes()
  186. if err != nil {
  187. return nil, err
  188. }
  189. res := &ItemValueResps{}
  190. err = json.Unmarshal(resp, res)
  191. if err != nil {
  192. return nil, err
  193. }
  194. if len(res.Data) == 0 {
  195. return nil, errors.New("not found envitem's value")
  196. }
  197. if adjustValue && cache != nil && len(res.Data) > 0 {
  198. if rv, err := strconv.ParseInt(res.Data[0].Val, 10, 64); err == nil {
  199. adjust, _ := e.GetAdjustInt64Val()
  200. res.Data[0].Val = fmt.Sprintf("%d", rv+adjust)
  201. } else if rv, err := strconv.ParseFloat(res.Data[0].Val, 64); err == nil {
  202. adjust, _ := e.GetAdjustFloat64Val()
  203. res.Data[0].Val = fmt.Sprintf("%f", rv+adjust)
  204. }
  205. }
  206. return res.Data[0], nil
  207. }
  208. return nil, errors.New(fmt.Sprintf("request statusCode: %d", r.StatusCode))
  209. }
  210. func (e *EnvItem) getCurrentValue() (string, string, error) {
  211. resp, err := e.getCurrentData()
  212. if err != nil {
  213. return "", "", err
  214. }
  215. e.Value = resp.Val
  216. e.Htime = resp.HTime
  217. _ = e.setPrevValue()
  218. return resp.Val, resp.HTime, nil
  219. }
  220. func (e *EnvItem) getPrevValue() (string, error) {
  221. if v, ok := itemPrevValues.Load(e.Item); ok {
  222. return v.(string), nil
  223. }
  224. return "", errors.New("not found prev envitem's value")
  225. }
  226. func (e *EnvItem) setPrevValue() error {
  227. itemPrevValues.Store(e.Item, e.Value)
  228. return nil
  229. }
  230. func (e *EnvItem) GetItemFloat64Val() float64 {
  231. if e.Value == nil {
  232. e.getCurrentValue()
  233. }
  234. switch e.Value.(type) {
  235. case string:
  236. if v, e := strconv.ParseFloat(e.Value.(string), 64); e == nil {
  237. return v
  238. }
  239. }
  240. return 0
  241. }
  242. func (e *EnvItem) GetItemInt64Val() int64 {
  243. if e.Value == nil {
  244. e.getCurrentValue()
  245. }
  246. switch e.Value.(type) {
  247. case string:
  248. if v, e := strconv.ParseInt(e.Value.(string), 10, 64); e == nil {
  249. return v
  250. }
  251. }
  252. return 0
  253. }
  254. func (e *EnvItem) GetItemStringVal() string {
  255. if e.Value == nil {
  256. e.getCurrentValue()
  257. }
  258. if v, ok := e.Value.(string); ok {
  259. return v
  260. }
  261. return ""
  262. }
  263. func (e *EnvItem) GetItemHtime() *time.Time {
  264. if e.Value == nil {
  265. e.getCurrentValue()
  266. }
  267. if ht, err := time.ParseInLocation("2006-01-02 15:04:05", e.Htime, time.Local); err == nil {
  268. return &ht
  269. }
  270. return nil
  271. }
  272. func (e *EnvItem) GetItemPrevFloat64Val() float64 {
  273. ov, err := e.getPrevValue()
  274. if err != nil {
  275. return 0
  276. }
  277. if v, err := strconv.ParseFloat(ov, 64); err == nil {
  278. return v
  279. }
  280. return 0
  281. }
  282. func (e *EnvItem) GetItemPrevInt64Val() int64 {
  283. ov, err := e.getPrevValue()
  284. if err != nil {
  285. return 0
  286. }
  287. if v, err := strconv.ParseInt(ov, 10, 64); err == nil {
  288. return v
  289. }
  290. return 0
  291. }
  292. func (e *EnvItem) GetItemPrevStringVal() string {
  293. if ov, err := e.getPrevValue(); err == nil {
  294. return ov
  295. }
  296. return ""
  297. }
  298. func (e *EnvItem) GetAdjustInt64Val() (int64, error) {
  299. return getAdjustInt64Val(e.ProjectId, e.Item)
  300. }
  301. func (e *EnvItem) GetAdjustFloat64Val() (float64, error) {
  302. return getAdjustFloat64Val(e.ProjectId, e.Item)
  303. }
  304. func (e *EnvItem) GetAdjustStringVal() (string, error) {
  305. return getAdjustStringVal(e.ProjectId, e.Item)
  306. }
  307. func (e *EnvItem) SetAdjust(value string, expire time.Duration) error {
  308. if adjustValue == false {
  309. return nil
  310. }
  311. if cache == nil {
  312. return errors.New("not cache")
  313. }
  314. key := fmt.Sprintf(CACHE_ADJUST_VALUE_KEY, e.ProjectId, e.Item)
  315. scmd := cache.Set(context.Background(), key, value, expire)
  316. return scmd.Err()
  317. }
  318. func (e *EnvItem) IncreAdjust(expire time.Duration) (int64, error) {
  319. if adjustValue == false {
  320. return 0, nil
  321. }
  322. if cache == nil {
  323. return 0, errors.New("not cache")
  324. }
  325. key := fmt.Sprintf(CACHE_ADJUST_VALUE_KEY, e.ProjectId, e.Item)
  326. intCmd := cache.Incr(context.Background(), key)
  327. if intCmd.Err() != nil {
  328. return 0, intCmd.Err()
  329. }
  330. cache.Expire(context.Background(), key, expire)
  331. return intCmd.Val(), nil
  332. }
  333. func (e *EnvItem) SetValue(o, v string) error {
  334. ts := time.Now().Unix()
  335. data := make(SetCurrentsReq, 1)
  336. data[0] = SetCurrentReq{
  337. ProjectId: e.ProjectId,
  338. Item: e.Item,
  339. OldValue: o,
  340. NewValue: v,
  341. }
  342. sign := data.Sign(plcItemSecret, ts)
  343. jsonByte, _ := json.Marshal(data)
  344. requestBuf := bytes.NewBuffer(jsonByte)
  345. headers := http.Header{}
  346. headers.Add("Content-Type", "application/json")
  347. resp, err := _sendRequest(ctlUrl, "POST", fmt.Sprintf("sign=%s&timestamp=%d", sign, ts), headers, requestBuf)
  348. if err != nil {
  349. return fmt.Errorf("设置点位数据失败参数,%s,%s", string(jsonByte), err.Error())
  350. }
  351. respData := &SetCurrentResp{}
  352. err = json.Unmarshal(resp.Bytes(), respData)
  353. if err != nil {
  354. return fmt.Errorf("设置点位数据失败参数,%s,%s,%s", string(jsonByte), err.Error(), resp)
  355. }
  356. if respData.Code != http.StatusOK {
  357. return fmt.Errorf("设置点位数据失败参数,%s,%s", string(jsonByte), resp)
  358. }
  359. return nil
  360. }
  361. // WaitValue 等待点位的值, 变为指定值
  362. // v: 等待的值
  363. // op: -1: 小于, 0: 等于, 1: 大于
  364. // interval: 轮询间隔
  365. // retry: 最多尝试次数 0: 表示不限制
  366. func (e *EnvItem) WaitValue(v string, op int, interval time.Duration, retry int) <-chan bool {
  367. c := make(chan bool)
  368. go func() {
  369. t := 0
  370. for {
  371. val, _, err := e.getCurrentValue()
  372. if err == nil && strings.Compare(v, val) == op {
  373. c <- true
  374. }
  375. if retry > 0 {
  376. t += 1
  377. if t >= retry {
  378. c <- false
  379. }
  380. }
  381. time.Sleep(interval)
  382. }
  383. }()
  384. return c
  385. }
  386. // WaitNotValue 等待点位的值, 变为非指定值
  387. // vs: 非等待的值
  388. // interval: 轮询间隔
  389. // retry: 最多尝试次数 0: 表示不限制
  390. func (e *EnvItem) WaitNotValue(vs []string, interval time.Duration, retry int) <-chan string {
  391. c := make(chan string)
  392. m := make(map[string]struct{}, len(vs))
  393. for _, v := range vs {
  394. m[v] = struct{}{}
  395. }
  396. go func() {
  397. t := 0
  398. for {
  399. val, _, err := e.getCurrentValue()
  400. if _, ok := m[val]; err == nil && !ok {
  401. c <- val
  402. }
  403. if retry > 0 {
  404. t += 1
  405. if t >= retry {
  406. c <- ""
  407. }
  408. }
  409. time.Sleep(interval)
  410. }
  411. }()
  412. return c
  413. }
  414. func (e *EnvItem) ClearAdjust() {
  415. if adjustValue == false {
  416. return
  417. }
  418. if cache == nil {
  419. return
  420. }
  421. key := fmt.Sprintf(CACHE_ADJUST_VALUE_KEY, e.ProjectId, e.Item)
  422. cache.Del(context.Background(), key)
  423. }
  424. func (e *EnvItem) clearValue() {
  425. e.Value = nil
  426. e.Htime = ""
  427. }
  428. // Scan 实现方法
  429. func (d *MultiEnvItem) Scan(input interface{}) error {
  430. _ = json.Unmarshal(input.([]byte), &d)
  431. return nil
  432. }
  433. func (d MultiEnvItem) Value() (driver.Value, error) {
  434. return json.Marshal(d)
  435. }
  436. func getAdjustInt64Val(projectId int64, item string) (int64, error) {
  437. if adjustValue == false {
  438. return 0, nil
  439. }
  440. if cache == nil {
  441. return 0, errors.New("not cache")
  442. }
  443. key := fmt.Sprintf(CACHE_ADJUST_VALUE_KEY, projectId, item)
  444. scmd := cache.Get(context.Background(), key)
  445. if scmd != nil && scmd.Err() != nil {
  446. return 0, scmd.Err()
  447. }
  448. return scmd.Int64()
  449. }
  450. func getAdjustFloat64Val(projectId int64, item string) (float64, error) {
  451. if adjustValue == false {
  452. return 0, nil
  453. }
  454. if cache == nil {
  455. return 0, errors.New("not cache")
  456. }
  457. key := fmt.Sprintf(CACHE_ADJUST_VALUE_KEY, projectId, item)
  458. scmd := cache.Get(context.Background(), key)
  459. if scmd != nil && scmd.Err() != nil {
  460. return 0, scmd.Err()
  461. }
  462. return scmd.Float64()
  463. }
  464. func getAdjustStringVal(projectId int64, item string) (string, error) {
  465. if adjustValue == false {
  466. return "", nil
  467. }
  468. if cache == nil {
  469. return "", errors.New("not cache")
  470. }
  471. key := fmt.Sprintf(CACHE_ADJUST_VALUE_KEY, projectId, item)
  472. scmd := cache.Get(context.Background(), key)
  473. if scmd != nil && scmd.Err() != nil {
  474. return "", scmd.Err()
  475. }
  476. return scmd.String(), nil
  477. }
  478. func (s SetCurrentsReq) Sign(secret string, ts int64) string {
  479. jsonByte, _ := json.Marshal(s)
  480. requestBuf := bytes.NewBuffer(jsonByte)
  481. hasher := md5.New()
  482. hasher.Write([]byte(fmt.Sprintf("%s%s%d", requestBuf.Bytes(), secret, ts)))
  483. return strings.ToUpper(hex.EncodeToString(hasher.Sum(nil)))
  484. }
  485. func (v *VirtualPlcItem) GetVirtualPlcItems() (*VirtualPlcItemResp, error) {
  486. tmpPlcItemUrl := fmt.Sprintf("%s%d", plcItemUrl, v.ProjectId)
  487. req := httplib.Get(tmpPlcItemUrl)
  488. req.Param("project_id", fmt.Sprintf("%d", v.ProjectId))
  489. req.Param("pageSize", fmt.Sprintf("%d", v.PageSize))
  490. r, err := req.Response()
  491. if err != nil {
  492. return nil, err
  493. }
  494. defer r.Body.Close()
  495. if r.StatusCode == 200 {
  496. resp, err := req.Bytes()
  497. if err != nil {
  498. return nil, err
  499. }
  500. res := &VirtualPlcItemResp{}
  501. err = json.Unmarshal(resp, res)
  502. if err != nil {
  503. return nil, err
  504. }
  505. return res, nil
  506. }
  507. return nil, errors.New(fmt.Sprintf("request statusCode: %d", r.StatusCode))
  508. }
  509. func _sendRequest(urlStr, method, queryParam string, headers http.Header, postData io.Reader) (buffer *bytes.Buffer, err error) {
  510. client := &http.Client{
  511. Transport: &http.Transport{
  512. Proxy: http.ProxyFromEnvironment,
  513. DialContext: (&net.Dialer{
  514. Timeout: 3 * time.Second,
  515. KeepAlive: 10 * time.Second,
  516. }).DialContext,
  517. MaxIdleConns: 10,
  518. MaxIdleConnsPerHost: 10,
  519. IdleConnTimeout: 10 * time.Second,
  520. },
  521. Timeout: 10 * time.Second,
  522. }
  523. URL := urlStr
  524. Method := method
  525. if len(Method) == 0 {
  526. Method = "GET"
  527. }
  528. if len(queryParam) != 0 {
  529. //reqQueryParam = queryParam
  530. queryParam = strings.Replace(queryParam, " ", "%20", -1) //转义请求参数中所有的空格
  531. queryParam = strings.Replace(queryParam, "/", "%2F", -1) //转义请求参数中所有的斜线
  532. queryParam = strings.Replace(queryParam, "(", "%28", -1) //转义请求参数中所有的左括号
  533. queryParam = strings.Replace(queryParam, ")", "%29", -1) //转义请求参数中所有的右括号
  534. queryParam = strings.Replace(queryParam, ",", "%2C", -1) //转义请求参数中所有的逗号
  535. queryParam = strings.Replace(queryParam, ";", "%3B", -1) //转义请求参数中所有的分号
  536. l, err := url.Parse("?" + queryParam)
  537. if err != nil {
  538. //log.Errorf(err, "parse http url error")
  539. return nil, err
  540. }
  541. param := l.Query().Encode()
  542. URL = fmt.Sprintf("%s?%s", URL, param)
  543. }
  544. reqBuffer := &bytes.Buffer{}
  545. if postData != nil {
  546. reqBuffer.ReadFrom(postData)
  547. }
  548. req, err := http.NewRequest(Method, URL, reqBuffer)
  549. if len(headers) > 0 {
  550. for s, header := range headers {
  551. if len(header) > 0 {
  552. req.Header.Add(s, header[0])
  553. }
  554. }
  555. }
  556. if nil != err {
  557. return nil, err
  558. }
  559. resp, err := client.Do(req)
  560. if nil != err {
  561. //log.Errorf(err, "request failed:%v")
  562. return nil, err
  563. }
  564. var data io.ReadCloser
  565. switch resp.Header.Get("Content-Encoding") {
  566. case "gzip":
  567. data, err = gzip.NewReader(resp.Body)
  568. if nil != err {
  569. return nil, err
  570. }
  571. default:
  572. data = resp.Body
  573. }
  574. defer func() {
  575. if resp != nil && resp.Body != nil {
  576. resp.Body.Close()
  577. }
  578. }()
  579. buffer = &bytes.Buffer{}
  580. _, err = buffer.ReadFrom(data)
  581. if nil != err {
  582. return nil, err
  583. }
  584. return buffer, nil
  585. }