mongodao.go 18 KB

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