mongodao.go 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807
  1. package dao
  2. import (
  3. "context"
  4. "encoding/json"
  5. "errors"
  6. "fmt"
  7. "io"
  8. "log"
  9. "sort"
  10. "strings"
  11. "time"
  12. "github.com/willie68/schematic-service-go/config"
  13. slicesutils "github.com/willie68/schematic-service-go/internal"
  14. "github.com/willie68/schematic-service-go/model"
  15. "go.mongodb.org/mongo-driver/bson"
  16. "go.mongodb.org/mongo-driver/bson/primitive"
  17. "go.mongodb.org/mongo-driver/mongo"
  18. "go.mongodb.org/mongo-driver/mongo/gridfs"
  19. "go.mongodb.org/mongo-driver/mongo/options"
  20. )
  21. // time to reload all users
  22. const userReloadPeriod = 1 * time.Hour
  23. const timeout = 1 * time.Minute
  24. const attachmentsCollectionName = "attachments"
  25. const schematicsCollectionName = "schematics"
  26. const tagsCollectionName = "tags"
  27. const manufacturersCollectionName = "manufacturers"
  28. const usersCollectionName = "users"
  29. // MongoDAO a mongodb based dao
  30. type MongoDAO struct {
  31. initialised bool
  32. client *mongo.Client
  33. mongoConfig config.MongoDB
  34. bucket gridfs.Bucket
  35. database mongo.Database
  36. tags model.Tags
  37. manufacturers model.Manufacturers
  38. users map[string]string
  39. ticker time.Ticker
  40. done chan bool
  41. }
  42. // InitDAO initialise the mongodb connection, build up all collections and indexes
  43. func (m *MongoDAO) InitDAO(MongoConfig config.MongoDB) {
  44. m.initialised = false
  45. m.mongoConfig = MongoConfig
  46. // uri := fmt.Sprintf("mongodb://%s:%s@%s:%d", mongoConfig.Username, mongoConfig.Password, mongoConfig.Host, mongoConfig.Port)
  47. uri := fmt.Sprintf("mongodb://%s:%d", m.mongoConfig.Host, m.mongoConfig.Port)
  48. clientOptions := options.Client()
  49. clientOptions.ApplyURI(uri)
  50. clientOptions.Auth = &options.Credential{Username: m.mongoConfig.Username, Password: m.mongoConfig.Password, AuthSource: m.mongoConfig.AuthDB}
  51. var err error
  52. m.client, err = mongo.NewClient(clientOptions)
  53. if err != nil {
  54. fmt.Printf("error: %s\n", err.Error())
  55. }
  56. ctx, cancel := context.WithTimeout(context.Background(), timeout)
  57. defer cancel()
  58. err = m.client.Connect(ctx)
  59. if err != nil {
  60. fmt.Printf("error: %s\n", err.Error())
  61. }
  62. m.database = *m.client.Database(m.mongoConfig.Database)
  63. myBucket, err := gridfs.NewBucket(&m.database, options.GridFSBucket().SetName(attachmentsCollectionName))
  64. if err != nil {
  65. fmt.Printf("error: %s\n", err.Error())
  66. }
  67. m.bucket = *myBucket
  68. m.initIndexSchematics()
  69. m.initIndexTags()
  70. m.initIndexManufacturers()
  71. m.tags = model.NewTags()
  72. m.manufacturers = model.NewManufacturers()
  73. m.users = make(map[string]string)
  74. m.initTags()
  75. m.initManufacturers()
  76. m.initUsers()
  77. m.initialised = true
  78. }
  79. func (m *MongoDAO) initIndexSchematics() {
  80. collection := m.database.Collection(schematicsCollectionName)
  81. indexView := collection.Indexes()
  82. ctx, _ := context.WithTimeout(context.Background(), timeout)
  83. cursor, err := indexView.List(ctx)
  84. if err != nil {
  85. log.Fatal(err)
  86. }
  87. defer cursor.Close(ctx)
  88. myIndexes := make([]string, 0)
  89. for cursor.Next(ctx) {
  90. var index bson.M
  91. if err = cursor.Decode(&index); err != nil {
  92. log.Fatal(err)
  93. }
  94. myIndexes = append(myIndexes, index["name"].(string))
  95. }
  96. if !slicesutils.Contains(myIndexes, "manufacturer") {
  97. ctx, _ = context.WithTimeout(context.Background(), timeout)
  98. models := []mongo.IndexModel{
  99. {
  100. Keys: bson.D{{"manufacturer", 1}},
  101. Options: options.Index().SetName("manufacturer").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  102. },
  103. {
  104. Keys: bson.D{{"model", 1}},
  105. Options: options.Index().SetName("model").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  106. },
  107. {
  108. Keys: bson.D{{"tags", 1}},
  109. Options: options.Index().SetName("tags").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  110. },
  111. {
  112. Keys: bson.D{{"subtitle", 1}},
  113. Options: options.Index().SetName("subtitle").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  114. },
  115. {
  116. Keys: bson.D{{"manufacturer", "text"}, {"model", "text"}, {"tags", "text"}, {"subtitle", "text"}, {"description", "text"}, {"owner", "text"}},
  117. Options: options.Index().SetName("$text"),
  118. },
  119. }
  120. // Specify the MaxTime option to limit the amount of time the operation can run on the server
  121. opts := options.CreateIndexes().SetMaxTime(2 * time.Second)
  122. names, err := indexView.CreateMany(context.TODO(), models, opts)
  123. if err != nil {
  124. log.Fatal(err)
  125. }
  126. log.Print("create indexes:")
  127. for _, name := range names {
  128. log.Println(name)
  129. }
  130. }
  131. }
  132. func (m *MongoDAO) initIndexTags() {
  133. collection := m.database.Collection(tagsCollectionName)
  134. indexView := collection.Indexes()
  135. ctx, _ := context.WithTimeout(context.Background(), timeout)
  136. cursor, err := indexView.List(ctx)
  137. if err != nil {
  138. log.Fatal(err)
  139. }
  140. defer cursor.Close(ctx)
  141. myIndexes := make([]string, 0)
  142. for cursor.Next(ctx) {
  143. var index bson.M
  144. if err = cursor.Decode(&index); err != nil {
  145. log.Fatal(err)
  146. }
  147. myIndexes = append(myIndexes, index["name"].(string))
  148. }
  149. if !slicesutils.Contains(myIndexes, "name") {
  150. ctx, _ = context.WithTimeout(context.Background(), timeout)
  151. models := []mongo.IndexModel{
  152. {
  153. Keys: bson.D{{"name", 1}},
  154. Options: options.Index().SetUnique(true).SetName("name").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  155. },
  156. }
  157. // Specify the MaxTime option to limit the amount of time the operation can run on the server
  158. opts := options.CreateIndexes().SetMaxTime(2 * time.Second)
  159. names, err := indexView.CreateMany(context.TODO(), models, opts)
  160. if err != nil {
  161. log.Fatal(err)
  162. }
  163. log.Print("create indexes:")
  164. for _, name := range names {
  165. log.Println(name)
  166. }
  167. }
  168. }
  169. func (m *MongoDAO) initIndexManufacturers() {
  170. collection := m.database.Collection(manufacturersCollectionName)
  171. indexView := collection.Indexes()
  172. ctx, _ := context.WithTimeout(context.Background(), timeout)
  173. cursor, err := indexView.List(ctx)
  174. if err != nil {
  175. log.Fatal(err)
  176. }
  177. defer cursor.Close(ctx)
  178. myIndexes := make([]string, 0)
  179. for cursor.Next(ctx) {
  180. var index bson.M
  181. if err = cursor.Decode(&index); err != nil {
  182. log.Fatal(err)
  183. }
  184. myIndexes = append(myIndexes, index["name"].(string))
  185. }
  186. if !slicesutils.Contains(myIndexes, "name") {
  187. ctx, _ = context.WithTimeout(context.Background(), timeout)
  188. models := []mongo.IndexModel{
  189. {
  190. Keys: bson.D{{"name", 1}},
  191. Options: options.Index().SetUnique(true).SetName("name").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  192. },
  193. }
  194. // Specify the MaxTime option to limit the amount of time the operation can run on the server
  195. opts := options.CreateIndexes().SetMaxTime(2 * time.Second)
  196. names, err := indexView.CreateMany(context.TODO(), models, opts)
  197. if err != nil {
  198. log.Fatal(err)
  199. }
  200. log.Print("create indexes:")
  201. for _, name := range names {
  202. log.Println(name)
  203. }
  204. }
  205. }
  206. func (m *MongoDAO) initTags() {
  207. ctx, _ := context.WithTimeout(context.Background(), timeout)
  208. tagsCollection := m.database.Collection(tagsCollectionName)
  209. cursor, err := tagsCollection.Find(ctx, bson.M{})
  210. if err != nil {
  211. log.Fatal(err)
  212. }
  213. defer cursor.Close(ctx)
  214. tagList := make([]string, 0)
  215. for cursor.Next(ctx) {
  216. var tag bson.M
  217. if err = cursor.Decode(&tag); err != nil {
  218. log.Fatal(err)
  219. } else {
  220. tagList = append(tagList, tag["name"].(string))
  221. }
  222. }
  223. ctx, _ = context.WithTimeout(context.Background(), timeout)
  224. schematicCollection := m.database.Collection(schematicsCollectionName)
  225. for _, tagName := range tagList {
  226. queryDoc := bson.M{
  227. "tags": tagName,
  228. }
  229. n, err := schematicCollection.CountDocuments(ctx, queryDoc, &options.CountOptions{Collation: &options.Collation{Locale: "en", Strength: 2}})
  230. if err != nil {
  231. log.Println(err)
  232. }
  233. //log.Printf("tag: %s has %d schematics.", tagName, n)
  234. m.tags.Add(tagName, int(n))
  235. }
  236. }
  237. func (m *MongoDAO) initManufacturers() {
  238. ctx, _ := context.WithTimeout(context.Background(), timeout)
  239. manufacturersCollection := m.database.Collection(manufacturersCollectionName)
  240. cursor, err := manufacturersCollection.Find(ctx, bson.M{})
  241. if err != nil {
  242. log.Fatal(err)
  243. }
  244. defer cursor.Close(ctx)
  245. manuList := make([]string, 0)
  246. for cursor.Next(ctx) {
  247. var manufacturer bson.M
  248. if err = cursor.Decode(&manufacturer); err != nil {
  249. log.Fatal(err)
  250. } else {
  251. manuList = append(manuList, manufacturer["name"].(string))
  252. }
  253. }
  254. ctx, _ = context.WithTimeout(context.Background(), timeout)
  255. schematicCollection := m.database.Collection(schematicsCollectionName)
  256. for _, manuName := range manuList {
  257. queryDoc := bson.M{
  258. "manufacturer": manuName,
  259. }
  260. n, err := schematicCollection.CountDocuments(ctx, queryDoc, &options.CountOptions{Collation: &options.Collation{Locale: "en", Strength: 2}})
  261. if err != nil {
  262. log.Println(err)
  263. }
  264. //log.Printf("manufacturer: %s has %d schematics.", manuName, n)
  265. m.manufacturers.Add(manuName, int(n))
  266. }
  267. }
  268. func (m *MongoDAO) initUsers() {
  269. m.reloadUsers()
  270. go func() {
  271. background := time.NewTicker(userReloadPeriod)
  272. for _ = range background.C {
  273. m.reloadUsers()
  274. }
  275. }()
  276. }
  277. func (m *MongoDAO) reloadUsers() {
  278. ctx, _ := context.WithTimeout(context.Background(), timeout)
  279. usersCollection := m.database.Collection(usersCollectionName)
  280. cursor, err := usersCollection.Find(ctx, bson.M{})
  281. if err != nil {
  282. log.Fatal(err)
  283. }
  284. defer cursor.Close(ctx)
  285. localUsers := make(map[string]string)
  286. for cursor.Next(ctx) {
  287. var user bson.M
  288. if err = cursor.Decode(&user); err != nil {
  289. log.Fatal(err)
  290. } else {
  291. username := user["name"].(string)
  292. password := user["password"].(string)
  293. localUsers[username] = BuildPasswordHash(password)
  294. }
  295. }
  296. m.users = localUsers
  297. if len(m.users) == 0 {
  298. admin := model.User{
  299. Name: "w.klaas@gmx.de",
  300. Password: "akteon0000",
  301. Admin: true,
  302. }
  303. m.AddUser(admin)
  304. guest := model.User{
  305. Name: "gast",
  306. Password: "gast1234",
  307. Admin: false,
  308. Guest: true,
  309. }
  310. m.AddUser(guest)
  311. }
  312. }
  313. // AddFile adding a file to the storage, stream like
  314. func (m *MongoDAO) AddFile(filename string, reader io.Reader) (string, error) {
  315. uploadOpts := options.GridFSUpload().SetMetadata(bson.D{{"tag", "tag"}})
  316. fileID, err := m.bucket.UploadFromStream(filename, reader, uploadOpts)
  317. if err != nil {
  318. fmt.Printf("error: %s\n", err.Error())
  319. return "", err
  320. }
  321. log.Printf("Write file to DB was successful. File id: %s \n", fileID)
  322. id := fileID.Hex()
  323. return id, nil
  324. }
  325. // CreateSchematic creating a new schematic in the database
  326. func (m *MongoDAO) CreateSchematic(schematic model.Schematic) (string, error) {
  327. for _, tag := range schematic.Tags {
  328. if !m.tags.Contains(tag) {
  329. m.CreateTag(tag)
  330. }
  331. }
  332. if !m.manufacturers.Contains(schematic.Manufacturer) {
  333. m.CreateManufacturer(schematic.Manufacturer)
  334. }
  335. ctx, _ := context.WithTimeout(context.Background(), timeout)
  336. collection := m.database.Collection(schematicsCollectionName)
  337. result, err := collection.InsertOne(ctx, schematic)
  338. if err != nil {
  339. fmt.Printf("error: %s\n", err.Error())
  340. return "", err
  341. }
  342. filter := bson.M{"_id": result.InsertedID}
  343. err = collection.FindOne(ctx, filter).Decode(&schematic)
  344. if err != nil {
  345. fmt.Printf("error: %s\n", err.Error())
  346. return "", err
  347. }
  348. switch v := result.InsertedID.(type) {
  349. case primitive.ObjectID:
  350. return v.Hex(), nil
  351. }
  352. return "", nil
  353. }
  354. // UpdateSchematic creating a new schematic in the database
  355. func (m *MongoDAO) UpdateSchematic(schematic model.Schematic) (string, error) {
  356. for _, tag := range schematic.Tags {
  357. if !m.tags.Contains(tag) {
  358. m.CreateTag(tag)
  359. }
  360. }
  361. if !m.manufacturers.Contains(schematic.Manufacturer) {
  362. m.CreateManufacturer(schematic.Manufacturer)
  363. }
  364. ctx, _ := context.WithTimeout(context.Background(), timeout)
  365. collection := m.database.Collection(schematicsCollectionName)
  366. filter := bson.M{"_id": schematic.ID}
  367. updateDoc := bson.D{{"$set", schematic}}
  368. result, err := collection.UpdateOne(ctx, filter, updateDoc)
  369. if err != nil {
  370. fmt.Printf("error: %s\n", err.Error())
  371. return "", err
  372. }
  373. if result.ModifiedCount != 1 {
  374. return "", errors.New("can't update document.")
  375. }
  376. err = collection.FindOne(ctx, filter).Decode(&schematic)
  377. if err != nil {
  378. fmt.Printf("error: %s\n", err.Error())
  379. return "", err
  380. }
  381. return schematic.ID.Hex(), nil
  382. }
  383. // GetSchematic getting a sdingle schematic
  384. func (m *MongoDAO) GetSchematic(schematicID string) (model.Schematic, error) {
  385. ctx, _ := context.WithTimeout(context.Background(), timeout)
  386. schematicCollection := m.database.Collection(schematicsCollectionName)
  387. objectID, _ := primitive.ObjectIDFromHex(schematicID)
  388. result := schematicCollection.FindOne(ctx, bson.M{"_id": objectID})
  389. err := result.Err()
  390. if err == mongo.ErrNoDocuments {
  391. log.Print(err)
  392. return model.Schematic{}, ErrNoDocument
  393. }
  394. if err != nil {
  395. log.Print(err)
  396. return model.Schematic{}, err
  397. }
  398. var schematic model.Schematic
  399. if err := result.Decode(&schematic); err != nil {
  400. log.Print(err)
  401. return model.Schematic{}, err
  402. } else {
  403. return schematic, nil
  404. }
  405. }
  406. // DeleteSchematic getting a sdingle schematic
  407. func (m *MongoDAO) DeleteSchematic(schematicID string) error {
  408. ctx, _ := context.WithTimeout(context.Background(), timeout)
  409. schematicCollection := m.database.Collection(schematicsCollectionName)
  410. objectID, _ := primitive.ObjectIDFromHex(schematicID)
  411. result, err := schematicCollection.DeleteOne(ctx, bson.M{"_id": objectID})
  412. if err != nil {
  413. log.Print(err)
  414. return err
  415. } else {
  416. if result.DeletedCount > 0 {
  417. return nil
  418. }
  419. return ErrNoDocument
  420. }
  421. }
  422. //GetFile getting a single from the database with the id
  423. func (m *MongoDAO) GetFilename(fileid string) (string, error) {
  424. objectID, err := primitive.ObjectIDFromHex(fileid)
  425. if err != nil {
  426. log.Print(err)
  427. return "", err
  428. }
  429. ctx, _ := context.WithTimeout(context.Background(), timeout)
  430. cursor, err := m.bucket.Find(bson.M{"_id": objectID})
  431. if err != nil {
  432. log.Print(err)
  433. return "", err
  434. }
  435. defer cursor.Close(ctx)
  436. cursor.Next(ctx)
  437. var file bson.M
  438. var filename string
  439. if err = cursor.Decode(&file); err != nil {
  440. log.Print(err)
  441. return "", err
  442. } else {
  443. filename = file["filename"].(string)
  444. }
  445. return filename, nil
  446. }
  447. //GetFile getting a single from the database with the id
  448. func (m *MongoDAO) GetFile(fileid string, stream io.Writer) error {
  449. objectID, err := primitive.ObjectIDFromHex(fileid)
  450. if err != nil {
  451. log.Print(err)
  452. return err
  453. }
  454. dStream, err := m.bucket.DownloadToStream(objectID, stream)
  455. if err != nil {
  456. log.Print(err)
  457. return err
  458. }
  459. fmt.Printf("File size to download: %v \n", dStream)
  460. return nil
  461. }
  462. // GetSchematics getting a sdingle schematic
  463. func (m *MongoDAO) GetSchematics(query string, offset int, limit int, owner string) (int64, []model.Schematic, error) {
  464. ctx, _ := context.WithTimeout(context.Background(), timeout)
  465. schematicCollection := m.database.Collection(schematicsCollectionName)
  466. var queryM map[string]interface{}
  467. err := json.Unmarshal([]byte(query), &queryM)
  468. if err != nil {
  469. log.Print(err)
  470. return 0, nil, err
  471. }
  472. queryDoc := bson.M{}
  473. for k, v := range queryM {
  474. if k == "$fulltext" {
  475. queryDoc["$text"] = bson.M{"$search": v}
  476. } else {
  477. switch v := v.(type) {
  478. // case float64:
  479. // case int:
  480. // case bool:
  481. case string:
  482. queryDoc[k] = bson.M{"$regex": v}
  483. }
  484. //queryDoc[k] = v
  485. }
  486. }
  487. data, _ := json.Marshal(queryDoc)
  488. log.Printf("mongoquery: %s\n", string(data))
  489. n, err := schematicCollection.CountDocuments(ctx, queryDoc, &options.CountOptions{Collation: &options.Collation{Locale: "en", Strength: 2}})
  490. if err != nil {
  491. log.Print(err)
  492. return 0, nil, err
  493. }
  494. cursor, err := schematicCollection.Find(ctx, queryDoc, &options.FindOptions{Collation: &options.Collation{Locale: "en", Strength: 2}})
  495. if err != nil {
  496. log.Print(err)
  497. return 0, nil, err
  498. }
  499. defer cursor.Close(ctx)
  500. schematics := make([]model.Schematic, 0)
  501. count := 0
  502. docs := 0
  503. for cursor.Next(ctx) {
  504. if count >= offset {
  505. if docs < limit {
  506. var schematic model.Schematic
  507. if err = cursor.Decode(&schematic); err != nil {
  508. log.Print(err)
  509. return 0, nil, err
  510. } else {
  511. if !schematic.PrivateFile || schematic.Owner == owner {
  512. schematics = append(schematics, schematic)
  513. docs++
  514. }
  515. }
  516. } else {
  517. break
  518. }
  519. }
  520. count++
  521. }
  522. return n, schematics, nil
  523. }
  524. // GetSchematicsCount getting a sdingle schematic
  525. func (m *MongoDAO) GetSchematicsCount(query string, owner string) (int64, error) {
  526. ctx, _ := context.WithTimeout(context.Background(), timeout)
  527. schematicCollection := m.database.Collection(schematicsCollectionName)
  528. queryDoc := bson.M{}
  529. if query != "" {
  530. var queryM map[string]interface{}
  531. err := json.Unmarshal([]byte(query), &queryM)
  532. if err != nil {
  533. log.Print(err)
  534. return 0, err
  535. }
  536. for k, v := range queryM {
  537. if k == "$fulltext" {
  538. queryDoc["$text"] = bson.M{"$search": v}
  539. } else {
  540. switch v := v.(type) {
  541. // case float64:
  542. // case int:
  543. // case bool:
  544. case string:
  545. queryDoc[k] = bson.M{"$regex": v}
  546. }
  547. //queryDoc[k] = v
  548. }
  549. }
  550. data, _ := json.Marshal(queryDoc)
  551. log.Printf("mongoquery: %s\n", string(data))
  552. }
  553. n, err := schematicCollection.CountDocuments(ctx, queryDoc, &options.CountOptions{Collation: &options.Collation{Locale: "en", Strength: 2}})
  554. if err != nil {
  555. log.Print(err)
  556. return 0, err
  557. }
  558. return n, nil
  559. }
  560. // CreateTag create a new tag in the storage
  561. func (m *MongoDAO) CreateTag(tag string) error {
  562. tag = strings.ToLower(tag)
  563. ctx, _ := context.WithTimeout(context.Background(), timeout)
  564. collection := m.database.Collection(tagsCollectionName)
  565. tagModel := bson.M{"name": tag}
  566. _, err := collection.InsertOne(ctx, tagModel)
  567. if err != nil {
  568. fmt.Printf("error: %s\n", err.Error())
  569. return err
  570. }
  571. m.tags.Add(tag, 1)
  572. return nil
  573. }
  574. //GetTags getting a list of all tags
  575. func (m *MongoDAO) GetTags() []model.Tag {
  576. sort.Slice(m.tags.List, func(i, j int) bool { return m.tags.List[i].Name < m.tags.List[j].Name })
  577. return m.tags.List
  578. }
  579. // GetTagsCount getting the count of all tags
  580. func (m *MongoDAO) GetTagsCount() int {
  581. return len(m.tags.List)
  582. }
  583. // CreateManufacturer create a new manufacturer in the storage
  584. func (m *MongoDAO) CreateManufacturer(manufacturer string) error {
  585. ctx, _ := context.WithTimeout(context.Background(), timeout)
  586. collection := m.database.Collection(manufacturersCollectionName)
  587. manufacturerModel := bson.M{"name": manufacturer}
  588. _, err := collection.InsertOne(ctx, manufacturerModel)
  589. if err != nil {
  590. fmt.Printf("error: %s\n", err.Error())
  591. return err
  592. }
  593. m.manufacturers.Add(manufacturer, 1)
  594. return nil
  595. }
  596. // GetManufacturers getting a list of all manufacturers
  597. func (m *MongoDAO) GetManufacturers() []model.Manufacturer {
  598. sort.Slice(m.manufacturers.List, func(i, j int) bool { return m.manufacturers.List[i].Name < m.manufacturers.List[j].Name })
  599. return m.manufacturers.List
  600. }
  601. // GetManufacturersCount getting the count of all manufacturers
  602. func (m *MongoDAO) GetManufacturersCount() int {
  603. return len(m.manufacturers.List)
  604. }
  605. // CheckUser checking username and password... returns true if the user is active and the password for this user is correct
  606. func (m *MongoDAO) CheckUser(username string, password string) bool {
  607. pwd, ok := m.users[username]
  608. if ok {
  609. if pwd == password {
  610. return true
  611. } else {
  612. user, ok := m.GetUser(username)
  613. if ok {
  614. if user.Password == password {
  615. return true
  616. }
  617. }
  618. }
  619. }
  620. if !ok {
  621. user, ok := m.GetUser(username)
  622. if ok {
  623. if user.Password == password {
  624. return true
  625. }
  626. }
  627. }
  628. return false
  629. }
  630. // GetUser getting the usermolde
  631. func (m *MongoDAO) GetUser(username string) (model.User, bool) {
  632. ctx, _ := context.WithTimeout(context.Background(), timeout)
  633. usersCollection := m.database.Collection(usersCollectionName)
  634. var user model.User
  635. filter := bson.M{"name": username}
  636. err := usersCollection.FindOne(ctx, filter).Decode(&user)
  637. if err != nil {
  638. fmt.Printf("error: %s\n", err.Error())
  639. return model.User{}, false
  640. }
  641. password := user.Password
  642. hash := BuildPasswordHash(password)
  643. m.users[username] = hash
  644. return user, true
  645. }
  646. // DropAll dropping all data from the database
  647. func (m *MongoDAO) DropAll() {
  648. ctx, _ := context.WithTimeout(context.Background(), timeout)
  649. collectionNames, err := m.database.ListCollectionNames(ctx, bson.D{}, &options.ListCollectionsOptions{})
  650. if err != nil {
  651. log.Fatal(err)
  652. }
  653. for _, name := range collectionNames {
  654. if name != usersCollectionName {
  655. collection := m.database.Collection(name)
  656. err = collection.Drop(ctx)
  657. if err != nil {
  658. log.Fatal(err)
  659. }
  660. }
  661. }
  662. }
  663. // Stop stopping the mongodao
  664. func (m *MongoDAO) Stop() {
  665. m.ticker.Stop()
  666. m.done <- true
  667. }
  668. // AddUser adding a new user to the system
  669. func (m *MongoDAO) AddUser(user model.User) error {
  670. if user.Name == "" {
  671. return errors.New("username should not be empty")
  672. }
  673. _, ok := m.users[user.Name]
  674. if ok {
  675. return errors.New("username already exists")
  676. }
  677. user.Password = BuildPasswordHash(user.Password)
  678. ctx, _ := context.WithTimeout(context.Background(), timeout)
  679. collection := m.database.Collection(usersCollectionName)
  680. _, err := collection.InsertOne(ctx, user)
  681. if err != nil {
  682. fmt.Printf("error: %s\n", err.Error())
  683. return err
  684. }
  685. m.users[user.Name] = user.Password
  686. return nil
  687. }
  688. // DeleteUser deletes one user from the system
  689. func (m *MongoDAO) DeleteUser(username string) error {
  690. if username == "" {
  691. return errors.New("username should not be empty")
  692. }
  693. _, ok := m.users[username]
  694. if !ok {
  695. return errors.New("username not exists")
  696. }
  697. ctx, _ := context.WithTimeout(context.Background(), timeout)
  698. collection := m.database.Collection(usersCollectionName)
  699. filter := bson.M{"name": username}
  700. _, err := collection.DeleteOne(ctx, filter)
  701. if err != nil {
  702. fmt.Printf("error: %s\n", err.Error())
  703. return err
  704. }
  705. delete(m.users, username)
  706. return nil
  707. }
  708. // ChangePWD changes the apssword of a single user
  709. func (m *MongoDAO) ChangePWD(username string, newpassword string, oldpassword string) error {
  710. if username == "" {
  711. return errors.New("username should not be empty")
  712. }
  713. pwd, ok := m.users[username]
  714. if !ok {
  715. return errors.New("username not registered")
  716. }
  717. newpassword = BuildPasswordHash(newpassword)
  718. oldpassword = BuildPasswordHash(oldpassword)
  719. if pwd != oldpassword {
  720. return errors.New("actual password incorrect")
  721. }
  722. ctx, _ := context.WithTimeout(context.Background(), timeout)
  723. collection := m.database.Collection(usersCollectionName)
  724. filter := bson.M{"name": username}
  725. update := bson.D{{"$set", bson.D{{"password", newpassword}}}}
  726. result := collection.FindOneAndUpdate(ctx, filter, update)
  727. if result.Err() != nil {
  728. fmt.Printf("error: %s\n", result.Err().Error())
  729. return result.Err()
  730. }
  731. m.users[username] = newpassword
  732. return nil
  733. }
  734. // Ping pinging the mongoDao
  735. func (m *MongoDAO) Ping() error {
  736. if !m.initialised {
  737. return errors.New("mongo client not initialised")
  738. }
  739. ctx, _ := context.WithTimeout(context.Background(), timeout)
  740. return m.database.Client().Ping(ctx, nil)
  741. }