mongodao.go 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349
  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/mongo"
  14. "go.mongodb.org/mongo-driver/mongo/gridfs"
  15. "go.mongodb.org/mongo-driver/mongo/options"
  16. )
  17. const timeout = 1 * time.Minute
  18. const attachmentsCollectionName = "attachments"
  19. const schematicsCollectionName = "schematics"
  20. const tagsCollectionName = "tags"
  21. const manufacturersCollectionName = "manufacturers"
  22. var client *mongo.Client
  23. var mongoConfig config.MongoDB
  24. var bucket gridfs.Bucket
  25. var database mongo.Database
  26. var tags []string
  27. var manufacturers []string
  28. // InitDB initialise the mongodb connection, build up all collections and indexes
  29. func InitDB(MongoConfig config.MongoDB) {
  30. mongoConfig = MongoConfig
  31. // uri := fmt.Sprintf("mongodb://%s:%s@%s:%d", mongoConfig.Username, mongoConfig.Password, mongoConfig.Host, mongoConfig.Port)
  32. uri := fmt.Sprintf("mongodb://%s:%d", mongoConfig.Host, mongoConfig.Port)
  33. clientOptions := options.Client()
  34. clientOptions.ApplyURI(uri)
  35. clientOptions.Auth = &options.Credential{Username: mongoConfig.Username, Password: mongoConfig.Password, AuthSource: mongoConfig.AuthDB}
  36. var err error
  37. client, err = mongo.NewClient(clientOptions)
  38. if err != nil {
  39. fmt.Printf("error: %s\n", err.Error())
  40. }
  41. ctx, cancel := context.WithTimeout(context.Background(), timeout)
  42. defer cancel()
  43. err = client.Connect(ctx)
  44. if err != nil {
  45. fmt.Printf("error: %s\n", err.Error())
  46. }
  47. database = *client.Database(mongoConfig.Database)
  48. myBucket, err := gridfs.NewBucket(&database, options.GridFSBucket().SetName(attachmentsCollectionName))
  49. if err != nil {
  50. fmt.Printf("error: %s\n", err.Error())
  51. }
  52. bucket = *myBucket
  53. initIndexSchematics()
  54. initIndexTags()
  55. initIndexManufacturers()
  56. tags = make([]string, 0)
  57. manufacturers = make([]string, 0)
  58. initTags()
  59. initManufacturers()
  60. }
  61. func initIndexSchematics() {
  62. collection := database.Collection(schematicsCollectionName)
  63. indexView := collection.Indexes()
  64. ctx, _ := context.WithTimeout(context.Background(), timeout)
  65. cursor, err := indexView.List(ctx)
  66. if err != nil {
  67. log.Fatal(err)
  68. }
  69. defer cursor.Close(ctx)
  70. myIndexes := make([]string, 0)
  71. for cursor.Next(ctx) {
  72. var index bson.M
  73. if err = cursor.Decode(&index); err != nil {
  74. log.Fatal(err)
  75. }
  76. myIndexes = append(myIndexes, index["name"].(string))
  77. }
  78. for _, name := range myIndexes {
  79. log.Println(name)
  80. }
  81. if !slicesutils.Contains(myIndexes, "manufaturer") {
  82. ctx, _ = context.WithTimeout(context.Background(), timeout)
  83. models := []mongo.IndexModel{
  84. {
  85. Keys: bson.D{{"manufacturer", 1}},
  86. Options: options.Index().SetName("manufacturer").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  87. },
  88. {
  89. Keys: bson.D{{"model", 1}},
  90. Options: options.Index().SetName("model").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  91. },
  92. {
  93. Keys: bson.D{{"tags", 1}},
  94. Options: options.Index().SetName("tags").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  95. },
  96. {
  97. Keys: bson.D{{"subtitle", 1}},
  98. Options: options.Index().SetName("subtitle").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  99. },
  100. }
  101. // Specify the MaxTime option to limit the amount of time the operation can run on the server
  102. opts := options.CreateIndexes().SetMaxTime(2 * time.Second)
  103. names, err := indexView.CreateMany(context.TODO(), models, opts)
  104. if err != nil {
  105. log.Fatal(err)
  106. }
  107. log.Print("create indexes:")
  108. for _, name := range names {
  109. log.Println(name)
  110. }
  111. }
  112. }
  113. func initIndexTags() {
  114. collection := database.Collection(tagsCollectionName)
  115. indexView := collection.Indexes()
  116. ctx, _ := context.WithTimeout(context.Background(), timeout)
  117. cursor, err := indexView.List(ctx)
  118. if err != nil {
  119. log.Fatal(err)
  120. }
  121. defer cursor.Close(ctx)
  122. myIndexes := make([]string, 0)
  123. for cursor.Next(ctx) {
  124. var index bson.M
  125. if err = cursor.Decode(&index); err != nil {
  126. log.Fatal(err)
  127. }
  128. myIndexes = append(myIndexes, index["name"].(string))
  129. }
  130. for _, name := range myIndexes {
  131. log.Println(name)
  132. }
  133. if !slicesutils.Contains(myIndexes, "name") {
  134. ctx, _ = context.WithTimeout(context.Background(), timeout)
  135. models := []mongo.IndexModel{
  136. {
  137. Keys: bson.D{{"name", 1}},
  138. Options: options.Index().SetUnique(true).SetName("name").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  139. },
  140. }
  141. // Specify the MaxTime option to limit the amount of time the operation can run on the server
  142. opts := options.CreateIndexes().SetMaxTime(2 * time.Second)
  143. names, err := indexView.CreateMany(context.TODO(), models, opts)
  144. if err != nil {
  145. log.Fatal(err)
  146. }
  147. log.Print("create indexes:")
  148. for _, name := range names {
  149. log.Println(name)
  150. }
  151. }
  152. }
  153. func initIndexManufacturers() {
  154. collection := database.Collection(manufacturersCollectionName)
  155. indexView := collection.Indexes()
  156. ctx, _ := context.WithTimeout(context.Background(), timeout)
  157. cursor, err := indexView.List(ctx)
  158. if err != nil {
  159. log.Fatal(err)
  160. }
  161. defer cursor.Close(ctx)
  162. myIndexes := make([]string, 0)
  163. for cursor.Next(ctx) {
  164. var index bson.M
  165. if err = cursor.Decode(&index); err != nil {
  166. log.Fatal(err)
  167. }
  168. myIndexes = append(myIndexes, index["name"].(string))
  169. }
  170. for _, name := range myIndexes {
  171. log.Println(name)
  172. }
  173. if !slicesutils.Contains(myIndexes, "name") {
  174. ctx, _ = context.WithTimeout(context.Background(), timeout)
  175. models := []mongo.IndexModel{
  176. {
  177. Keys: bson.D{{"name", 1}},
  178. Options: options.Index().SetUnique(true).SetName("name").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
  179. },
  180. }
  181. // Specify the MaxTime option to limit the amount of time the operation can run on the server
  182. opts := options.CreateIndexes().SetMaxTime(2 * time.Second)
  183. names, err := indexView.CreateMany(context.TODO(), models, opts)
  184. if err != nil {
  185. log.Fatal(err)
  186. }
  187. log.Print("create indexes:")
  188. for _, name := range names {
  189. log.Println(name)
  190. }
  191. }
  192. }
  193. func initTags() {
  194. ctx, _ := context.WithTimeout(context.Background(), timeout)
  195. tagsCollection := database.Collection(tagsCollectionName)
  196. cursor, err := tagsCollection.Find(ctx, bson.M{})
  197. if err != nil {
  198. log.Fatal(err)
  199. }
  200. defer cursor.Close(ctx)
  201. for cursor.Next(ctx) {
  202. var tag bson.M
  203. if err = cursor.Decode(&tag); err != nil {
  204. log.Fatal(err)
  205. } else {
  206. tags = append(tags, tag["name"].(string))
  207. }
  208. }
  209. }
  210. func initManufacturers() {
  211. ctx, _ := context.WithTimeout(context.Background(), timeout)
  212. manufacturersCollection := database.Collection(manufacturersCollectionName)
  213. cursor, err := manufacturersCollection.Find(ctx, bson.M{})
  214. if err != nil {
  215. log.Fatal(err)
  216. }
  217. defer cursor.Close(ctx)
  218. for cursor.Next(ctx) {
  219. var manufacturer bson.M
  220. if err = cursor.Decode(&manufacturer); err != nil {
  221. log.Fatal(err)
  222. } else {
  223. manufacturers = append(manufacturers, manufacturer["name"].(string))
  224. }
  225. }
  226. }
  227. // AddFile adding a file to the storage, stream like
  228. func AddFile(filename string, reader io.Reader) (string, error) {
  229. uploadOpts := options.GridFSUpload().SetMetadata(bson.D{{"tag", "tag"}})
  230. fileID, err := bucket.UploadFromStream(filename, reader, uploadOpts)
  231. if err != nil {
  232. fmt.Printf("error: %s\n", err.Error())
  233. return "", err
  234. }
  235. log.Printf("Write file to DB was successful. File id: %s \n", fileID)
  236. id := fileID.String()
  237. return id, nil
  238. }
  239. // CreateSchematic creating a new schematic in the database
  240. func CreateSchematic(schematic model.Schematic) (string, error) {
  241. for _, tag := range schematic.Tags {
  242. if !slicesutils.Contains(tags, tag) {
  243. CreateTag(tag)
  244. }
  245. }
  246. if !slicesutils.Contains(manufacturers, schematic.Manufacturer) {
  247. CreateManufacturer(schematic.Manufacturer)
  248. }
  249. ctx, _ := context.WithTimeout(context.Background(), timeout)
  250. collection := database.Collection(schematicsCollectionName)
  251. result, err := collection.InsertOne(ctx, schematic)
  252. if err != nil {
  253. fmt.Printf("error: %s\n", err.Error())
  254. return "", err
  255. }
  256. filter := bson.M{"_id": result.InsertedID}
  257. err = collection.FindOne(ctx, filter).Decode(&schematic)
  258. if err != nil {
  259. fmt.Printf("error: %s\n", err.Error())
  260. return "", err
  261. }
  262. return result.InsertedID.(string), nil
  263. }
  264. // CreateTag create a new tag in the storage
  265. func CreateTag(tag string) error {
  266. tag = strings.ToLower(tag)
  267. ctx, _ := context.WithTimeout(context.Background(), timeout)
  268. collection := database.Collection(tagsCollectionName)
  269. tagModel := bson.M{"name": tag}
  270. _, err := collection.InsertOne(ctx, tagModel)
  271. if err != nil {
  272. fmt.Printf("error: %s\n", err.Error())
  273. return err
  274. }
  275. tags = append(tags, tag)
  276. return nil
  277. }
  278. // CreateManufacturer create a new manufacturer in the storage
  279. func CreateManufacturer(manufacturer string) error {
  280. ctx, _ := context.WithTimeout(context.Background(), timeout)
  281. collection := database.Collection(manufacturersCollectionName)
  282. manufacturerModel := bson.M{"name": manufacturer}
  283. _, err := collection.InsertOne(ctx, manufacturerModel)
  284. if err != nil {
  285. fmt.Printf("error: %s\n", err.Error())
  286. return err
  287. }
  288. manufacturers = append(manufacturers, manufacturer)
  289. return nil
  290. }
  291. func GetTags() []string {
  292. return tags
  293. }
  294. func GetManufacturers() []string {
  295. return manufacturers
  296. }
  297. func GetTagsCount() int {
  298. return len(tags)
  299. }
  300. func GetManufacturersCount() int {
  301. return len(manufacturers)
  302. }
  303. func DropAll() {
  304. ctx, _ := context.WithTimeout(context.Background(), timeout)
  305. collectionNames, err := database.ListCollectionNames(ctx, bson.D{}, &options.ListCollectionsOptions{})
  306. if err != nil {
  307. log.Fatal(err)
  308. }
  309. for _, name := range collectionNames {
  310. collection := database.Collection(name)
  311. err = collection.Drop(ctx)
  312. if err != nil {
  313. log.Fatal(err)
  314. }
  315. }
  316. }