mongodao.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427
  1. package dao
  2. import (
  3. "context"
  4. "fmt"
  5. "io"
  6. "log"
  7. "strings"
  8. "time"
  9. "github.com/willie68/schematic-service-go/config"
  10. slicesutils "github.com/willie68/schematic-service-go/internal"
  11. "github.com/willie68/schematic-service-go/model"
  12. "go.mongodb.org/mongo-driver/bson"
  13. "go.mongodb.org/mongo-driver/bson/primitive"
  14. "go.mongodb.org/mongo-driver/mongo"
  15. "go.mongodb.org/mongo-driver/mongo/gridfs"
  16. "go.mongodb.org/mongo-driver/mongo/options"
  17. )
  18. const timeout = 1 * time.Minute
  19. const attachmentsCollectionName = "attachments"
  20. const schematicsCollectionName = "schematics"
  21. const tagsCollectionName = "tags"
  22. const manufacturersCollectionName = "manufacturers"
  23. var client *mongo.Client
  24. var mongoConfig config.MongoDB
  25. var bucket gridfs.Bucket
  26. var database mongo.Database
  27. var tags []string
  28. var manufacturers []string
  29. // InitDB initialise the mongodb connection, build up all collections and indexes
  30. func InitDB(MongoConfig config.MongoDB) {
  31. mongoConfig = MongoConfig
  32. // uri := fmt.Sprintf("mongodb://%s:%s@%s:%d", mongoConfig.Username, mongoConfig.Password, mongoConfig.Host, mongoConfig.Port)
  33. uri := fmt.Sprintf("mongodb://%s:%d", mongoConfig.Host, mongoConfig.Port)
  34. clientOptions := options.Client()
  35. clientOptions.ApplyURI(uri)
  36. clientOptions.Auth = &options.Credential{Username: mongoConfig.Username, Password: mongoConfig.Password, AuthSource: mongoConfig.AuthDB}
  37. var err error
  38. client, err = mongo.NewClient(clientOptions)
  39. if err != nil {
  40. fmt.Printf("error: %s\n", err.Error())
  41. }
  42. ctx, cancel := context.WithTimeout(context.Background(), timeout)
  43. defer cancel()
  44. err = client.Connect(ctx)
  45. if err != nil {
  46. fmt.Printf("error: %s\n", err.Error())
  47. }
  48. database = *client.Database(mongoConfig.Database)
  49. myBucket, err := gridfs.NewBucket(&database, options.GridFSBucket().SetName(attachmentsCollectionName))
  50. if err != nil {
  51. fmt.Printf("error: %s\n", err.Error())
  52. }
  53. bucket = *myBucket
  54. initIndexSchematics()
  55. initIndexTags()
  56. initIndexManufacturers()
  57. tags = make([]string, 0)
  58. manufacturers = make([]string, 0)
  59. initTags()
  60. initManufacturers()
  61. }
  62. func initIndexSchematics() {
  63. collection := database.Collection(schematicsCollectionName)
  64. indexView := collection.Indexes()
  65. ctx, _ := context.WithTimeout(context.Background(), timeout)
  66. cursor, err := indexView.List(ctx)
  67. if err != nil {
  68. log.Fatal(err)
  69. }
  70. defer cursor.Close(ctx)
  71. myIndexes := make([]string, 0)
  72. for cursor.Next(ctx) {
  73. var index bson.M
  74. if err = cursor.Decode(&index); err != nil {
  75. log.Fatal(err)
  76. }
  77. myIndexes = append(myIndexes, index["name"].(string))
  78. }
  79. for _, name := range myIndexes {
  80. log.Println(name)
  81. }
  82. if !slicesutils.Contains(myIndexes, "manufaturer") {
  83. ctx, _ = context.WithTimeout(context.Background(), timeout)
  84. models := []mongo.IndexModel{
  85. {
  86. Keys: bson.D{{"manufacturer", 1}},
  87. Options: options.Index().SetName("manufacturer").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  88. },
  89. {
  90. Keys: bson.D{{"model", 1}},
  91. Options: options.Index().SetName("model").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  92. },
  93. {
  94. Keys: bson.D{{"tags", 1}},
  95. Options: options.Index().SetName("tags").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  96. },
  97. {
  98. Keys: bson.D{{"subtitle", 1}},
  99. Options: options.Index().SetName("subtitle").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  100. },
  101. {
  102. Keys: bson.D{{"manufacturer", "text"}, {"model", "text"}, {"tags", "text"}, {"subtitle", "text"}, {"description", "text"}, {"owner", "text"}},
  103. Options: options.Index().SetName("$text"),
  104. },
  105. }
  106. // Specify the MaxTime option to limit the amount of time the operation can run on the server
  107. opts := options.CreateIndexes().SetMaxTime(2 * time.Second)
  108. names, err := indexView.CreateMany(context.TODO(), models, opts)
  109. if err != nil {
  110. log.Fatal(err)
  111. }
  112. log.Print("create indexes:")
  113. for _, name := range names {
  114. log.Println(name)
  115. }
  116. }
  117. }
  118. func initIndexTags() {
  119. collection := database.Collection(tagsCollectionName)
  120. indexView := collection.Indexes()
  121. ctx, _ := context.WithTimeout(context.Background(), timeout)
  122. cursor, err := indexView.List(ctx)
  123. if err != nil {
  124. log.Fatal(err)
  125. }
  126. defer cursor.Close(ctx)
  127. myIndexes := make([]string, 0)
  128. for cursor.Next(ctx) {
  129. var index bson.M
  130. if err = cursor.Decode(&index); err != nil {
  131. log.Fatal(err)
  132. }
  133. myIndexes = append(myIndexes, index["name"].(string))
  134. }
  135. for _, name := range myIndexes {
  136. log.Println(name)
  137. }
  138. if !slicesutils.Contains(myIndexes, "name") {
  139. ctx, _ = context.WithTimeout(context.Background(), timeout)
  140. models := []mongo.IndexModel{
  141. {
  142. Keys: bson.D{{"name", 1}},
  143. Options: options.Index().SetUnique(true).SetName("name").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  144. },
  145. }
  146. // Specify the MaxTime option to limit the amount of time the operation can run on the server
  147. opts := options.CreateIndexes().SetMaxTime(2 * time.Second)
  148. names, err := indexView.CreateMany(context.TODO(), models, opts)
  149. if err != nil {
  150. log.Fatal(err)
  151. }
  152. log.Print("create indexes:")
  153. for _, name := range names {
  154. log.Println(name)
  155. }
  156. }
  157. }
  158. func initIndexManufacturers() {
  159. collection := database.Collection(manufacturersCollectionName)
  160. indexView := collection.Indexes()
  161. ctx, _ := context.WithTimeout(context.Background(), timeout)
  162. cursor, err := indexView.List(ctx)
  163. if err != nil {
  164. log.Fatal(err)
  165. }
  166. defer cursor.Close(ctx)
  167. myIndexes := make([]string, 0)
  168. for cursor.Next(ctx) {
  169. var index bson.M
  170. if err = cursor.Decode(&index); err != nil {
  171. log.Fatal(err)
  172. }
  173. myIndexes = append(myIndexes, index["name"].(string))
  174. }
  175. for _, name := range myIndexes {
  176. log.Println(name)
  177. }
  178. if !slicesutils.Contains(myIndexes, "name") {
  179. ctx, _ = context.WithTimeout(context.Background(), timeout)
  180. models := []mongo.IndexModel{
  181. {
  182. Keys: bson.D{{"name", 1}},
  183. Options: options.Index().SetUnique(true).SetName("name").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  184. },
  185. }
  186. // Specify the MaxTime option to limit the amount of time the operation can run on the server
  187. opts := options.CreateIndexes().SetMaxTime(2 * time.Second)
  188. names, err := indexView.CreateMany(context.TODO(), models, opts)
  189. if err != nil {
  190. log.Fatal(err)
  191. }
  192. log.Print("create indexes:")
  193. for _, name := range names {
  194. log.Println(name)
  195. }
  196. }
  197. }
  198. func initTags() {
  199. ctx, _ := context.WithTimeout(context.Background(), timeout)
  200. tagsCollection := database.Collection(tagsCollectionName)
  201. cursor, err := tagsCollection.Find(ctx, bson.M{})
  202. if err != nil {
  203. log.Fatal(err)
  204. }
  205. defer cursor.Close(ctx)
  206. for cursor.Next(ctx) {
  207. var tag bson.M
  208. if err = cursor.Decode(&tag); err != nil {
  209. log.Fatal(err)
  210. } else {
  211. tags = append(tags, tag["name"].(string))
  212. }
  213. }
  214. }
  215. func initManufacturers() {
  216. ctx, _ := context.WithTimeout(context.Background(), timeout)
  217. manufacturersCollection := database.Collection(manufacturersCollectionName)
  218. cursor, err := manufacturersCollection.Find(ctx, bson.M{})
  219. if err != nil {
  220. log.Fatal(err)
  221. }
  222. defer cursor.Close(ctx)
  223. for cursor.Next(ctx) {
  224. var manufacturer bson.M
  225. if err = cursor.Decode(&manufacturer); err != nil {
  226. log.Fatal(err)
  227. } else {
  228. manufacturers = append(manufacturers, manufacturer["name"].(string))
  229. }
  230. }
  231. }
  232. // AddFile adding a file to the storage, stream like
  233. func AddFile(filename string, reader io.Reader) (string, error) {
  234. uploadOpts := options.GridFSUpload().SetMetadata(bson.D{{"tag", "tag"}})
  235. fileID, err := bucket.UploadFromStream(filename, reader, uploadOpts)
  236. if err != nil {
  237. fmt.Printf("error: %s\n", err.Error())
  238. return "", err
  239. }
  240. log.Printf("Write file to DB was successful. File id: %s \n", fileID)
  241. id := fileID.Hex()
  242. return id, nil
  243. }
  244. // CreateSchematic creating a new schematic in the database
  245. func CreateSchematic(schematic model.Schematic) (string, error) {
  246. for _, tag := range schematic.Tags {
  247. if !slicesutils.Contains(tags, tag) {
  248. CreateTag(tag)
  249. }
  250. }
  251. if !slicesutils.Contains(manufacturers, schematic.Manufacturer) {
  252. CreateManufacturer(schematic.Manufacturer)
  253. }
  254. ctx, _ := context.WithTimeout(context.Background(), timeout)
  255. collection := database.Collection(schematicsCollectionName)
  256. result, err := collection.InsertOne(ctx, schematic)
  257. if err != nil {
  258. fmt.Printf("error: %s\n", err.Error())
  259. return "", err
  260. }
  261. filter := bson.M{"_id": result.InsertedID}
  262. err = collection.FindOne(ctx, filter).Decode(&schematic)
  263. if err != nil {
  264. fmt.Printf("error: %s\n", err.Error())
  265. return "", err
  266. }
  267. switch v := result.InsertedID.(type) {
  268. case primitive.ObjectID:
  269. return v.Hex(), nil
  270. }
  271. return "", nil
  272. }
  273. // GetSchematic getting a sdingle schematic
  274. func GetSchematic(schematicID string) (model.Schematic, error) {
  275. ctx, _ := context.WithTimeout(context.Background(), timeout)
  276. schematicCollection := database.Collection(schematicsCollectionName)
  277. objectId, _ := primitive.ObjectIDFromHex(schematicID)
  278. result := schematicCollection.FindOne(ctx, bson.M{"_id": objectId})
  279. var schematic model.Schematic
  280. if err := result.Decode(&schematic); err != nil {
  281. log.Print(err)
  282. return model.Schematic{}, err
  283. } else {
  284. return schematic, nil
  285. }
  286. }
  287. func GetFile(fileid string, stream io.Writer) error {
  288. objectID, err := primitive.ObjectIDFromHex(fileid)
  289. if err != nil {
  290. log.Print(err)
  291. return err
  292. }
  293. dStream, err := bucket.DownloadToStream(objectID, stream)
  294. if err != nil {
  295. log.Print(err)
  296. return err
  297. }
  298. fmt.Printf("File size to download: %v \n", dStream)
  299. return nil
  300. }
  301. // GetSchematics getting a sdingle schematic
  302. func GetSchematics(query string, offset int, limit int) ([]model.Schematic, error) {
  303. ctx, _ := context.WithTimeout(context.Background(), timeout)
  304. schematicCollection := database.Collection(schematicsCollectionName)
  305. queryDoc := bson.M{}
  306. err := bson.UnmarshalExtJSON([]byte(query), false, queryDoc)
  307. if err != nil {
  308. log.Print(err)
  309. return nil, err
  310. }
  311. cursor, err := schematicCollection.Find(ctx, queryDoc, &options.FindOptions{Collation: &options.Collation{Locale: "en", Strength: 2}})
  312. if err != nil {
  313. log.Print(err)
  314. return nil, err
  315. }
  316. defer cursor.Close(ctx)
  317. schematics := make([]model.Schematic, 0)
  318. count := 0
  319. docs := 0
  320. for cursor.Next(ctx) {
  321. if count >= offset {
  322. if docs < limit {
  323. var schematic model.Schematic
  324. if err = cursor.Decode(&schematic); err != nil {
  325. log.Print(err)
  326. return nil, err
  327. } else {
  328. schematics = append(schematics, schematic)
  329. docs++
  330. }
  331. } else {
  332. break
  333. }
  334. }
  335. count++
  336. }
  337. return schematics, nil
  338. }
  339. // CreateTag create a new tag in the storage
  340. func CreateTag(tag string) error {
  341. tag = strings.ToLower(tag)
  342. ctx, _ := context.WithTimeout(context.Background(), timeout)
  343. collection := database.Collection(tagsCollectionName)
  344. tagModel := bson.M{"name": tag}
  345. _, err := collection.InsertOne(ctx, tagModel)
  346. if err != nil {
  347. fmt.Printf("error: %s\n", err.Error())
  348. return err
  349. }
  350. tags = append(tags, tag)
  351. return nil
  352. }
  353. // CreateManufacturer create a new manufacturer in the storage
  354. func CreateManufacturer(manufacturer string) error {
  355. ctx, _ := context.WithTimeout(context.Background(), timeout)
  356. collection := database.Collection(manufacturersCollectionName)
  357. manufacturerModel := bson.M{"name": manufacturer}
  358. _, err := collection.InsertOne(ctx, manufacturerModel)
  359. if err != nil {
  360. fmt.Printf("error: %s\n", err.Error())
  361. return err
  362. }
  363. manufacturers = append(manufacturers, manufacturer)
  364. return nil
  365. }
  366. func GetTags() []string {
  367. return tags
  368. }
  369. func GetManufacturers() []string {
  370. return manufacturers
  371. }
  372. func GetTagsCount() int {
  373. return len(tags)
  374. }
  375. func GetManufacturersCount() int {
  376. return len(manufacturers)
  377. }
  378. func DropAll() {
  379. ctx, _ := context.WithTimeout(context.Background(), timeout)
  380. collectionNames, err := database.ListCollectionNames(ctx, bson.D{}, &options.ListCollectionsOptions{})
  381. if err != nil {
  382. log.Fatal(err)
  383. }
  384. for _, name := range collectionNames {
  385. collection := database.Collection(name)
  386. err = collection.Drop(ctx)
  387. if err != nil {
  388. log.Fatal(err)
  389. }
  390. }
  391. }