mongodao.go 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379
  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. // Specify the MaxTime option to limit the amount of time the operation can run on the server
  103. opts := options.CreateIndexes().SetMaxTime(2 * time.Second)
  104. names, err := indexView.CreateMany(context.TODO(), models, opts)
  105. if err != nil {
  106. log.Fatal(err)
  107. }
  108. log.Print("create indexes:")
  109. for _, name := range names {
  110. log.Println(name)
  111. }
  112. }
  113. }
  114. func initIndexTags() {
  115. collection := database.Collection(tagsCollectionName)
  116. indexView := collection.Indexes()
  117. ctx, _ := context.WithTimeout(context.Background(), timeout)
  118. cursor, err := indexView.List(ctx)
  119. if err != nil {
  120. log.Fatal(err)
  121. }
  122. defer cursor.Close(ctx)
  123. myIndexes := make([]string, 0)
  124. for cursor.Next(ctx) {
  125. var index bson.M
  126. if err = cursor.Decode(&index); err != nil {
  127. log.Fatal(err)
  128. }
  129. myIndexes = append(myIndexes, index["name"].(string))
  130. }
  131. for _, name := range myIndexes {
  132. log.Println(name)
  133. }
  134. if !slicesutils.Contains(myIndexes, "name") {
  135. ctx, _ = context.WithTimeout(context.Background(), timeout)
  136. models := []mongo.IndexModel{
  137. {
  138. Keys: bson.D{{"name", 1}},
  139. Options: options.Index().SetUnique(true).SetName("name").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  140. },
  141. }
  142. // Specify the MaxTime option to limit the amount of time the operation can run on the server
  143. opts := options.CreateIndexes().SetMaxTime(2 * time.Second)
  144. names, err := indexView.CreateMany(context.TODO(), models, opts)
  145. if err != nil {
  146. log.Fatal(err)
  147. }
  148. log.Print("create indexes:")
  149. for _, name := range names {
  150. log.Println(name)
  151. }
  152. }
  153. }
  154. func initIndexManufacturers() {
  155. collection := database.Collection(manufacturersCollectionName)
  156. indexView := collection.Indexes()
  157. ctx, _ := context.WithTimeout(context.Background(), timeout)
  158. cursor, err := indexView.List(ctx)
  159. if err != nil {
  160. log.Fatal(err)
  161. }
  162. defer cursor.Close(ctx)
  163. myIndexes := make([]string, 0)
  164. for cursor.Next(ctx) {
  165. var index bson.M
  166. if err = cursor.Decode(&index); err != nil {
  167. log.Fatal(err)
  168. }
  169. myIndexes = append(myIndexes, index["name"].(string))
  170. }
  171. for _, name := range myIndexes {
  172. log.Println(name)
  173. }
  174. if !slicesutils.Contains(myIndexes, "name") {
  175. ctx, _ = context.WithTimeout(context.Background(), timeout)
  176. models := []mongo.IndexModel{
  177. {
  178. Keys: bson.D{{"name", 1}},
  179. Options: options.Index().SetUnique(true).SetName("name").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  180. },
  181. }
  182. // Specify the MaxTime option to limit the amount of time the operation can run on the server
  183. opts := options.CreateIndexes().SetMaxTime(2 * time.Second)
  184. names, err := indexView.CreateMany(context.TODO(), models, opts)
  185. if err != nil {
  186. log.Fatal(err)
  187. }
  188. log.Print("create indexes:")
  189. for _, name := range names {
  190. log.Println(name)
  191. }
  192. }
  193. }
  194. func initTags() {
  195. ctx, _ := context.WithTimeout(context.Background(), timeout)
  196. tagsCollection := database.Collection(tagsCollectionName)
  197. cursor, err := tagsCollection.Find(ctx, bson.M{})
  198. if err != nil {
  199. log.Fatal(err)
  200. }
  201. defer cursor.Close(ctx)
  202. for cursor.Next(ctx) {
  203. var tag bson.M
  204. if err = cursor.Decode(&tag); err != nil {
  205. log.Fatal(err)
  206. } else {
  207. tags = append(tags, tag["name"].(string))
  208. }
  209. }
  210. }
  211. func initManufacturers() {
  212. ctx, _ := context.WithTimeout(context.Background(), timeout)
  213. manufacturersCollection := database.Collection(manufacturersCollectionName)
  214. cursor, err := manufacturersCollection.Find(ctx, bson.M{})
  215. if err != nil {
  216. log.Fatal(err)
  217. }
  218. defer cursor.Close(ctx)
  219. for cursor.Next(ctx) {
  220. var manufacturer bson.M
  221. if err = cursor.Decode(&manufacturer); err != nil {
  222. log.Fatal(err)
  223. } else {
  224. manufacturers = append(manufacturers, manufacturer["name"].(string))
  225. }
  226. }
  227. }
  228. // AddFile adding a file to the storage, stream like
  229. func AddFile(filename string, reader io.Reader) (string, error) {
  230. uploadOpts := options.GridFSUpload().SetMetadata(bson.D{{"tag", "tag"}})
  231. fileID, err := bucket.UploadFromStream(filename, reader, uploadOpts)
  232. if err != nil {
  233. fmt.Printf("error: %s\n", err.Error())
  234. return "", err
  235. }
  236. log.Printf("Write file to DB was successful. File id: %s \n", fileID)
  237. id := fileID.Hex()
  238. return id, nil
  239. }
  240. // CreateSchematic creating a new schematic in the database
  241. func CreateSchematic(schematic model.Schematic) (string, error) {
  242. for _, tag := range schematic.Tags {
  243. if !slicesutils.Contains(tags, tag) {
  244. CreateTag(tag)
  245. }
  246. }
  247. if !slicesutils.Contains(manufacturers, schematic.Manufacturer) {
  248. CreateManufacturer(schematic.Manufacturer)
  249. }
  250. ctx, _ := context.WithTimeout(context.Background(), timeout)
  251. collection := database.Collection(schematicsCollectionName)
  252. result, err := collection.InsertOne(ctx, schematic)
  253. if err != nil {
  254. fmt.Printf("error: %s\n", err.Error())
  255. return "", err
  256. }
  257. filter := bson.M{"_id": result.InsertedID}
  258. err = collection.FindOne(ctx, filter).Decode(&schematic)
  259. if err != nil {
  260. fmt.Printf("error: %s\n", err.Error())
  261. return "", err
  262. }
  263. return result.InsertedID.(string), nil
  264. }
  265. // GetSchematic getting a sdingle schematic
  266. func GetSchematic(schematicID string) (model.Schematic, error) {
  267. ctx, _ := context.WithTimeout(context.Background(), timeout)
  268. schematicCollection := database.Collection(schematicsCollectionName)
  269. result := schematicCollection.FindOne(ctx, bson.M{"_id": schematicID})
  270. var schematic model.Schematic
  271. if err := result.Decode(&schematic); err != nil {
  272. log.Print(err)
  273. return model.Schematic{}, err
  274. } else {
  275. return schematic, nil
  276. }
  277. }
  278. func GetFile(fileid string, stream io.Writer) error {
  279. objectID, err := primitive.ObjectIDFromHex(fileid)
  280. if err != nil {
  281. log.Print(err)
  282. return err
  283. }
  284. dStream, err := bucket.DownloadToStream(objectID, stream)
  285. if err != nil {
  286. log.Print(err)
  287. return err
  288. }
  289. fmt.Printf("File size to download: %v \n", dStream)
  290. return nil
  291. }
  292. // CreateTag create a new tag in the storage
  293. func CreateTag(tag string) error {
  294. tag = strings.ToLower(tag)
  295. ctx, _ := context.WithTimeout(context.Background(), timeout)
  296. collection := database.Collection(tagsCollectionName)
  297. tagModel := bson.M{"name": tag}
  298. _, err := collection.InsertOne(ctx, tagModel)
  299. if err != nil {
  300. fmt.Printf("error: %s\n", err.Error())
  301. return err
  302. }
  303. tags = append(tags, tag)
  304. return nil
  305. }
  306. // CreateManufacturer create a new manufacturer in the storage
  307. func CreateManufacturer(manufacturer string) error {
  308. ctx, _ := context.WithTimeout(context.Background(), timeout)
  309. collection := database.Collection(manufacturersCollectionName)
  310. manufacturerModel := bson.M{"name": manufacturer}
  311. _, err := collection.InsertOne(ctx, manufacturerModel)
  312. if err != nil {
  313. fmt.Printf("error: %s\n", err.Error())
  314. return err
  315. }
  316. manufacturers = append(manufacturers, manufacturer)
  317. return nil
  318. }
  319. func GetTags() []string {
  320. return tags
  321. }
  322. func GetManufacturers() []string {
  323. return manufacturers
  324. }
  325. func GetTagsCount() int {
  326. return len(tags)
  327. }
  328. func GetManufacturersCount() int {
  329. return len(manufacturers)
  330. }
  331. func DropAll() {
  332. ctx, _ := context.WithTimeout(context.Background(), timeout)
  333. collectionNames, err := database.ListCollectionNames(ctx, bson.D{}, &options.ListCollectionsOptions{})
  334. if err != nil {
  335. log.Fatal(err)
  336. }
  337. for _, name := range collectionNames {
  338. collection := database.Collection(name)
  339. err = collection.Drop(ctx)
  340. if err != nil {
  341. log.Fatal(err)
  342. }
  343. }
  344. }