registry.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458
  1. package main
  2. import (
  3. "errors"
  4. "flag"
  5. "fmt"
  6. "sparrow/pkg/generator"
  7. "sparrow/pkg/models"
  8. "sparrow/pkg/rpcs"
  9. opentracing "github.com/opentracing/opentracing-go"
  10. "github.com/opentracing/opentracing-go/ext"
  11. )
  12. const (
  13. flagAESKey = "aeskey"
  14. )
  15. var confAESKey = flag.String(flagAESKey, "", "use your own aes encryting key.")
  16. var errDbActFailt = errors.New("数据库操作失败")
  17. type Registry struct {
  18. keygen *generator.KeyGenerator
  19. }
  20. // NewRegistry create new registry
  21. func NewRegistry() (*Registry, error) {
  22. gen, err := generator.NewKeyGenerator(*confAESKey)
  23. if err != nil {
  24. return nil, err
  25. }
  26. return &Registry{
  27. keygen: gen,
  28. }, nil
  29. }
  30. func setVendor(target *models.Vendor, src *models.Vendor) {
  31. target.ID = src.ID
  32. target.VendorName = src.VendorName
  33. target.VendorDescription = src.VendorDescription
  34. target.VendorKey = src.VendorKey
  35. target.CreatedAt = src.CreatedAt
  36. target.UpdatedAt = src.UpdatedAt
  37. }
  38. func setProduct(target *models.Product, src *models.Product) {
  39. target.ID = src.ID
  40. target.ProductName = src.ProductName
  41. target.ProductDescription = src.ProductDescription
  42. target.ProductKey = src.ProductKey
  43. target.ProductConfig = src.ProductConfig
  44. target.VendorID = src.VendorID
  45. target.CreatedAt = src.CreatedAt
  46. target.UpdatedAt = src.UpdatedAt
  47. }
  48. func setApplication(target *models.Application, src *models.Application) {
  49. target.ID = src.ID
  50. target.AppName = src.AppName
  51. target.AppDescription = src.AppDescription
  52. target.AppKey = src.AppKey
  53. target.ReportUrl = src.ReportUrl
  54. target.AppToken = src.AppToken
  55. target.AppDomain = src.AppDomain
  56. target.CreatedAt = src.CreatedAt
  57. target.UpdatedAt = src.UpdatedAt
  58. }
  59. func setDevice(target *models.Device, src *models.Device) {
  60. target.ID = src.ID
  61. target.ProductID = src.ProductID
  62. target.DeviceIdentifier = src.DeviceIdentifier
  63. target.DeviceSecret = src.DeviceIdentifier
  64. target.DeviceKey = src.DeviceKey
  65. target.DeviceName = src.DeviceName
  66. target.DeviceDescription = src.DeviceDescription
  67. target.DeviceVersion = src.DeviceVersion
  68. target.CreatedAt = src.CreatedAt
  69. target.UpdatedAt = src.UpdatedAt
  70. }
  71. // SaveVendor will create a vendor if the ID field is not initialized
  72. // if ID field is initialized, it will update the conresponding vendor.
  73. func (r *Registry) SaveVendor(vendor *models.Vendor, reply *models.Vendor) error {
  74. db, err := getDB()
  75. if err != nil {
  76. return err
  77. }
  78. if vendor.ID == 0 {
  79. // if ID field is not initialized, will generate key first
  80. err = db.Save(vendor).Error
  81. if err != nil {
  82. return err
  83. }
  84. key, err := r.keygen.GenRandomKey(int64(vendor.ID))
  85. if err != nil {
  86. return err
  87. }
  88. vendor.VendorKey = key
  89. }
  90. err = db.Save(vendor).Error
  91. if err != nil {
  92. return err
  93. }
  94. cache := getCache()
  95. cacheKey := fmt.Sprintf("Vendor:%v", vendor.ID)
  96. if _, ok := cache.Get(cacheKey); ok {
  97. cache.Delete(cacheKey)
  98. }
  99. // write cache
  100. cache.Set(cacheKey, vendor)
  101. setVendor(reply, vendor)
  102. return nil
  103. }
  104. // SaveApplication will create a application if the ID field is not initialized
  105. // if ID field is initialized, it will update the conresponding application.
  106. func (r *Registry) SaveApplication(app *models.Application, reply *models.Application) error {
  107. db, err := getDB()
  108. if err != nil {
  109. return err
  110. }
  111. if app.ID == 0 {
  112. err = db.Save(app).Error
  113. if err != nil {
  114. return err
  115. }
  116. key, err := r.keygen.GenRandomKey(int64(app.ID))
  117. if err != nil {
  118. return err
  119. }
  120. app.AppKey = key
  121. }
  122. err = db.Save(app).Error
  123. if err != nil {
  124. return err
  125. }
  126. cache := getCache()
  127. cacheKey := fmt.Sprintf("Application:%v", app.ID)
  128. if _, ok := cache.Get(cacheKey); ok {
  129. cache.Delete(cacheKey)
  130. }
  131. setApplication(reply, app)
  132. return nil
  133. }
  134. // ValidateApplication try to validate the given app key.
  135. // if success, it will reply the corresponding application
  136. func (r *Registry) ValidateApplication(key string, reply *models.Application) error {
  137. db, err := getDB()
  138. if err != nil {
  139. return err
  140. }
  141. id, err := r.keygen.DecodeIDFromRandomKey(key)
  142. if err != nil {
  143. return err
  144. }
  145. cache := getCache()
  146. cacheKey := fmt.Sprintf("Application:%v", id)
  147. if cacheValue, ok := cache.Get(cacheKey); ok {
  148. app := cacheValue.(*models.Application)
  149. setApplication(reply, app)
  150. } else {
  151. err = db.First(reply, id).Error
  152. if err != nil {
  153. return err
  154. }
  155. var storage models.Application
  156. storage = *reply
  157. cache.Set(cacheKey, &storage)
  158. }
  159. if reply.AppKey != key {
  160. return errors.New("app key not match.")
  161. }
  162. return nil
  163. }
  164. // FindVendor will find product by specified ID
  165. func (r *Registry) FindVendor(id int32, reply *models.Vendor) error {
  166. db, err := getDB()
  167. if err != nil {
  168. return err
  169. }
  170. cache := getCache()
  171. cacheKey := fmt.Sprintf("Vendor:%v", id)
  172. if cacheValue, ok := cache.Get(cacheKey); ok {
  173. vendor := cacheValue.(*models.Vendor)
  174. setVendor(reply, vendor)
  175. } else {
  176. err = db.First(reply, id).Error
  177. if err != nil {
  178. return err
  179. }
  180. var storage models.Vendor
  181. storage = *reply
  182. cache.Set(cacheKey, &storage)
  183. }
  184. return nil
  185. }
  186. // GetVendors will get all vendors in the platform.
  187. func (r *Registry) GetVendors(noarg int, reply *[]models.Vendor) error {
  188. db, err := getDB()
  189. if err != nil {
  190. return err
  191. }
  192. return db.Find(reply).Error
  193. }
  194. // GetApplications will get all applications in the platform.
  195. func (r *Registry) GetApplications(noarg int, reply *[]models.Application) error {
  196. db, err := getDB()
  197. if err != nil {
  198. return err
  199. }
  200. return db.Find(reply).Error
  201. }
  202. // FindApplication will find product by specified ID
  203. func (r *Registry) FindApplication(id int32, reply *models.Application) error {
  204. db, err := getDB()
  205. if err != nil {
  206. return err
  207. }
  208. cache := getCache()
  209. cacheKey := fmt.Sprintf("Application:%v", id)
  210. if cacheValue, ok := cache.Get(cacheKey); ok {
  211. app := cacheValue.(*models.Application)
  212. setApplication(reply, app)
  213. } else {
  214. err = db.First(reply, id).Error
  215. if err != nil {
  216. return err
  217. }
  218. var storage models.Application
  219. storage = *reply
  220. cache.Set(cacheKey, &storage)
  221. }
  222. return nil
  223. }
  224. // RegisterDevice try to register a device to our platform.
  225. // if the device has already been registered,
  226. // the registration will success return the registered device before.
  227. func (r *Registry) RegisterDevice(args *rpcs.ArgsDeviceRegister, reply *models.Device) error {
  228. spanCtx, _ := opentracing.GlobalTracer().Extract(opentracing.TextMap, opentracing.TextMapCarrier(args.SpanCarrier))
  229. span := opentracing.StartSpan("RegisterDevice", ext.RPCServerOption(spanCtx))
  230. defer span.Finish()
  231. span.SetTag("args", args)
  232. db, err := getDB()
  233. if err != nil {
  234. return err
  235. }
  236. product := &models.Product{}
  237. err = r.ValidateProduct(args.ProductKey, product)
  238. if err != nil {
  239. return err
  240. }
  241. identifier := genDeviceIdentifier(product.VendorID, product.ID, args.DeviceCode)
  242. if db.Where(&models.Device{DeviceIdentifier: identifier}).First(reply).RecordNotFound() {
  243. // device is not registered yet.
  244. reply.ProductID = int32(product.ID)
  245. reply.DeviceIdentifier = identifier
  246. reply.DeviceName = product.ProductName // product name as default device name.
  247. reply.DeviceDescription = product.ProductDescription
  248. reply.DeviceVersion = args.DeviceVersion
  249. reply.VendorID = uint(product.VendorID)
  250. reply.ModuleName = args.ModuleName
  251. err = db.Save(reply).Error
  252. if err != nil {
  253. return err
  254. }
  255. // generate a random device key with hex encoding.
  256. reply.DeviceKey, err = r.keygen.GenRandomKey(int64(reply.ID))
  257. if err != nil {
  258. return err
  259. }
  260. // generate a random password with base64 encoding.
  261. reply.DeviceSecret, err = generator.GenRandomPassword()
  262. if err != nil {
  263. return err
  264. }
  265. err = db.Save(reply).Error
  266. if err != nil {
  267. return err
  268. }
  269. } else {
  270. //delete cache
  271. cache := getCache()
  272. cacheKey := fmt.Sprintf("Device:%v", identifier)
  273. if _, ok := cache.Get(cacheKey); ok {
  274. cache.Delete(cacheKey)
  275. }
  276. // device has aleady been saved. just update version info.
  277. reply.DeviceVersion = args.DeviceVersion
  278. err = db.Save(reply).Error
  279. if err != nil {
  280. return err
  281. }
  282. }
  283. span.LogKV("device_identifier", reply.DeviceIdentifier)
  284. span.LogKV("device_key", reply.DeviceKey)
  285. span.LogKV("device_secret", reply.DeviceSecret)
  286. return nil
  287. }
  288. // FindDeviceByIdentifier will find the device by indentifier
  289. func (r *Registry) FindDeviceByIdentifier(identifier string, reply *models.Device) error {
  290. db, err := getDB()
  291. if err != nil {
  292. return err
  293. }
  294. cache := getCache()
  295. cacheKey := fmt.Sprintf("Device:%v", identifier)
  296. if cacheValue, ok := cache.Get(identifier); ok {
  297. device := cacheValue.(*models.Device)
  298. setDevice(reply, device)
  299. } else {
  300. err = db.Where(&models.Device{
  301. DeviceIdentifier: identifier,
  302. }).First(reply).Error
  303. if err != nil {
  304. return err
  305. }
  306. var storage models.Device
  307. storage = *reply
  308. cache.Set(cacheKey, &storage)
  309. }
  310. return nil
  311. }
  312. // FindDeviceById will find the device with given id
  313. func (r *Registry) FindDeviceById(args *rpcs.ArgsDeviceAuth, reply *models.Device) error {
  314. spanCtx, _ := opentracing.GlobalTracer().Extract(opentracing.TextMap, opentracing.TextMapCarrier(args.SpanCarrier))
  315. span := opentracing.StartSpan("FindDeviceById", ext.RPCServerOption(spanCtx))
  316. defer span.Finish()
  317. span.SetTag("args", args)
  318. db, err := getDB()
  319. if err != nil {
  320. return err
  321. }
  322. d := &models.Device{}
  323. d.ID = uint(args.DeviceID)
  324. err = db.Where(d).First(reply).Error
  325. if err != nil {
  326. return err
  327. }
  328. span.LogKV("device", reply)
  329. return nil
  330. }
  331. // ValidateDevice will validate a device key and return the model if success.
  332. func (r *Registry) ValidateDevice(key string, device *models.Device) error {
  333. id, err := r.keygen.DecodeIDFromRandomKey(key)
  334. if err != nil {
  335. return err
  336. }
  337. args := rpcs.ArgsDeviceAuth{
  338. DeviceID: id,
  339. }
  340. err = r.FindDeviceById(&args, device)
  341. if err != nil {
  342. return err
  343. }
  344. if device.DeviceKey != key {
  345. return errors.New("device key not match.")
  346. }
  347. return nil
  348. }
  349. // UpdateDeviceInfo will update a device info by identifier
  350. func (r *Registry) UpdateDeviceInfo(args *rpcs.ArgsDeviceUpdate, reply *models.Device) error {
  351. db, err := getDB()
  352. if err != nil {
  353. return err
  354. }
  355. err = r.FindDeviceByIdentifier(args.DeviceIdentifier, reply)
  356. if err != nil {
  357. return err
  358. }
  359. //delete cache
  360. cache := getCache()
  361. cacheKey := fmt.Sprintf("Device:%v", args.DeviceIdentifier)
  362. if _, ok := cache.Get(cacheKey); ok {
  363. cache.Delete(cacheKey)
  364. }
  365. reply.DeviceName = args.DeviceName
  366. reply.DeviceDescription = args.DeviceDescription
  367. err = db.Save(reply).Error
  368. if err != nil {
  369. return err
  370. }
  371. return nil
  372. }
  373. // CreateRule create a new rule with specified parameters.
  374. func (r *Registry) CreateRule(args *models.Rule, reply *rpcs.ReplyEmptyResult) error {
  375. db, err := getDB()
  376. if err != nil {
  377. return err
  378. }
  379. return db.Save(args).Error
  380. }
  381. // QueryRules queries rules by trigger and rule type.
  382. func (r *Registry) QueryRules(args *models.Rule, reply *[]models.Rule) error {
  383. db, err := getDB()
  384. if err != nil {
  385. return err
  386. }
  387. err = db.Where(args).Find(reply).Error
  388. if err != nil {
  389. return err
  390. }
  391. return nil
  392. }