actions.go 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  1. package main
  2. import (
  3. "context"
  4. "encoding/hex"
  5. "errors"
  6. "fmt"
  7. "math/rand"
  8. "net/http"
  9. "sparrow/pkg/models"
  10. "sparrow/pkg/rpcs"
  11. "sparrow/pkg/server"
  12. "sparrow/pkg/token"
  13. "strings"
  14. "github.com/opentracing/opentracing-go/ext"
  15. "github.com/opentracing/opentracing-go"
  16. "github.com/martini-contrib/render"
  17. )
  18. const (
  19. ErrOK = 0
  20. ErrSystemFault = 10001
  21. ErrDeviceNotFound = 10002
  22. ErrWrongSecret = 10003
  23. ErrProtocolNotSuported = 10004
  24. )
  25. func renderError(code int, err error) Common {
  26. result := Common{}
  27. result.Code = code
  28. result.Message = err.Error()
  29. server.Log.Error(err.Error())
  30. return result
  31. }
  32. // DeviceRegisterArgs device register args
  33. type DeviceRegisterArgs struct {
  34. ProductKey string `json:"product_key" binding:"required"`
  35. DeviceCode string `json:"device_code" binding:"required"`
  36. Version string `json:"version" binding:"required"`
  37. ModuleName string `json:"module" binding:"required"`
  38. }
  39. // DeviceAuthArgs device authentication args
  40. type DeviceAuthArgs struct {
  41. DeviceId int64 `json:"device_id" binding:"required"`
  42. DeviceSecret string `json:"device_secret" binding:"required"`
  43. Protocol string `json:"protocol" binding:"required"`
  44. }
  45. // RegisterDevice 设备激活
  46. func RegisterDevice(args DeviceRegisterArgs, r render.Render) {
  47. server.Log.Printf("ACTION RegisterDevice, args:: %v ", args)
  48. span, ctx := opentracing.StartSpanFromContext(context.Background(), "RegisterDevice")
  49. defer span.Finish()
  50. rpcargs := &rpcs.ArgsDeviceRegister{
  51. ProductKey: args.ProductKey,
  52. DeviceCode: args.DeviceCode,
  53. DeviceVersion: args.Version,
  54. ModuleName: args.ModuleName,
  55. }
  56. device := &models.Device{}
  57. ext.SpanKindRPCClient.Set(span)
  58. err := server.RPCCallByName(ctx, rpcs.RegistryServerName, "Registry.RegisterDevice", rpcargs, device)
  59. if err != nil {
  60. r.JSON(http.StatusOK, renderError(ErrSystemFault, err))
  61. return
  62. }
  63. server.Log.Infof("register device success: %v", device)
  64. result := DeviceRegisterResponse{}
  65. result.Data = DeviceRegisterData{
  66. DeviceId: int64(device.ID),
  67. DeviceSecret: device.DeviceSecret,
  68. DeviceKey: device.DeviceKey,
  69. DeviceIdentifier: device.DeviceIdentifier,
  70. }
  71. r.JSON(http.StatusOK, result)
  72. return
  73. }
  74. // AuthDevice device auth
  75. func AuthDevice(args DeviceAuthArgs, r render.Render) {
  76. server.Log.Printf("ACTION AuthDevice, args:: %v", args)
  77. device := &models.Device{}
  78. span, ctx := opentracing.StartSpanFromContext(context.Background(), "AuthDevice")
  79. defer span.Finish()
  80. ext.SpanKindRPCClient.Set(span)
  81. arg := uint64(args.DeviceId)
  82. err := server.RPCCallByName(ctx, rpcs.RegistryServerName, "Registry.FindDeviceById", &arg, device)
  83. if err != nil {
  84. r.JSON(http.StatusOK, renderError(ErrDeviceNotFound, err))
  85. return
  86. }
  87. if device.DeviceSecret != args.DeviceSecret {
  88. fmt.Printf("%s, %s\r\n", device.DeviceSecret, args.DeviceSecret)
  89. // device secret is wrong.
  90. r.JSON(http.StatusOK, renderError(ErrWrongSecret, errors.New("wrong device secret.")))
  91. return
  92. }
  93. hepler := token.NewHelper(*confRedisHost)
  94. token, err := hepler.GenerateToken(device.RecordId)
  95. if err != nil {
  96. r.JSON(http.StatusOK, renderError(ErrSystemFault, err))
  97. return
  98. }
  99. var hosts []string
  100. switch args.Protocol {
  101. case "http":
  102. hosts, err = server.GetServerHosts(strings.ToUpper(args.Protocol)+"Access", "httphost")
  103. case "mqtt":
  104. hosts, err = server.GetServerHosts(strings.ToUpper(args.Protocol)+"Access", "tcphost")
  105. case "coap":
  106. hosts, err = server.GetServerHosts(strings.ToUpper(args.Protocol)+"Access", "udphost")
  107. default:
  108. err = errors.New("unsuported protocol: " + args.Protocol)
  109. }
  110. if err != nil {
  111. r.JSON(http.StatusOK, renderError(ErrProtocolNotSuported, err))
  112. return
  113. }
  114. // just get a random host
  115. host := hosts[rand.Intn(len(hosts))]
  116. result := DeviceAuthResponse{}
  117. result.Data = DeviceAuthData{
  118. AccessToken: hex.EncodeToString(token),
  119. AccessAddr: host,
  120. }
  121. server.Log.Infof("auth device success: %v, token: %x, access: %v", device, token, host)
  122. r.JSON(http.StatusOK, result)
  123. return
  124. }