123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349 |
- package dao
- import (
- "context"
- "fmt"
- "io"
- "log"
- "strings"
- "time"
- "github.com/willie68/schematic-service-go/config"
- slicesutils "github.com/willie68/schematic-service-go/internal"
- "github.com/willie68/schematic-service-go/model"
- "go.mongodb.org/mongo-driver/bson"
- "go.mongodb.org/mongo-driver/mongo"
- "go.mongodb.org/mongo-driver/mongo/gridfs"
- "go.mongodb.org/mongo-driver/mongo/options"
- )
- const timeout = 1 * time.Minute
- const attachmentsCollectionName = "attachments"
- const schematicsCollectionName = "schematics"
- const tagsCollectionName = "tags"
- const manufacturersCollectionName = "manufacturers"
- var client *mongo.Client
- var mongoConfig config.MongoDB
- var bucket gridfs.Bucket
- var database mongo.Database
- var tags []string
- var manufacturers []string
- // InitDB initialise the mongodb connection, build up all collections and indexes
- func InitDB(MongoConfig config.MongoDB) {
- mongoConfig = MongoConfig
- // uri := fmt.Sprintf("mongodb://%s:%s@%s:%d", mongoConfig.Username, mongoConfig.Password, mongoConfig.Host, mongoConfig.Port)
- uri := fmt.Sprintf("mongodb://%s:%d", mongoConfig.Host, mongoConfig.Port)
- clientOptions := options.Client()
- clientOptions.ApplyURI(uri)
- clientOptions.Auth = &options.Credential{Username: mongoConfig.Username, Password: mongoConfig.Password, AuthSource: mongoConfig.AuthDB}
- var err error
- client, err = mongo.NewClient(clientOptions)
- if err != nil {
- fmt.Printf("error: %s\n", err.Error())
- }
- ctx, cancel := context.WithTimeout(context.Background(), timeout)
- defer cancel()
- err = client.Connect(ctx)
- if err != nil {
- fmt.Printf("error: %s\n", err.Error())
- }
- database = *client.Database(mongoConfig.Database)
- myBucket, err := gridfs.NewBucket(&database, options.GridFSBucket().SetName(attachmentsCollectionName))
- if err != nil {
- fmt.Printf("error: %s\n", err.Error())
- }
- bucket = *myBucket
- initIndexSchematics()
- initIndexTags()
- initIndexManufacturers()
- tags = make([]string, 0)
- manufacturers = make([]string, 0)
- initTags()
- initManufacturers()
- }
- func initIndexSchematics() {
- collection := database.Collection(schematicsCollectionName)
- indexView := collection.Indexes()
- ctx, _ := context.WithTimeout(context.Background(), timeout)
- cursor, err := indexView.List(ctx)
- if err != nil {
- log.Fatal(err)
- }
- defer cursor.Close(ctx)
- myIndexes := make([]string, 0)
- for cursor.Next(ctx) {
- var index bson.M
- if err = cursor.Decode(&index); err != nil {
- log.Fatal(err)
- }
- myIndexes = append(myIndexes, index["name"].(string))
- }
- for _, name := range myIndexes {
- log.Println(name)
- }
- if !slicesutils.Contains(myIndexes, "manufaturer") {
- ctx, _ = context.WithTimeout(context.Background(), timeout)
- models := []mongo.IndexModel{
- {
- Keys: bson.D{{"manufacturer", 1}},
- Options: options.Index().SetName("manufacturer").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
- },
- {
- Keys: bson.D{{"model", 1}},
- Options: options.Index().SetName("model").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
- },
- {
- Keys: bson.D{{"tags", 1}},
- Options: options.Index().SetName("tags").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
- },
- {
- Keys: bson.D{{"subtitle", 1}},
- Options: options.Index().SetName("subtitle").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
- },
- }
- // Specify the MaxTime option to limit the amount of time the operation can run on the server
- opts := options.CreateIndexes().SetMaxTime(2 * time.Second)
- names, err := indexView.CreateMany(context.TODO(), models, opts)
- if err != nil {
- log.Fatal(err)
- }
- log.Print("create indexes:")
- for _, name := range names {
- log.Println(name)
- }
- }
- }
- func initIndexTags() {
- collection := database.Collection(tagsCollectionName)
- indexView := collection.Indexes()
- ctx, _ := context.WithTimeout(context.Background(), timeout)
- cursor, err := indexView.List(ctx)
- if err != nil {
- log.Fatal(err)
- }
- defer cursor.Close(ctx)
- myIndexes := make([]string, 0)
- for cursor.Next(ctx) {
- var index bson.M
- if err = cursor.Decode(&index); err != nil {
- log.Fatal(err)
- }
- myIndexes = append(myIndexes, index["name"].(string))
- }
- for _, name := range myIndexes {
- log.Println(name)
- }
- if !slicesutils.Contains(myIndexes, "name") {
- ctx, _ = context.WithTimeout(context.Background(), timeout)
- models := []mongo.IndexModel{
- {
- Keys: bson.D{{"name", 1}},
- Options: options.Index().SetUnique(true).SetName("name").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
- },
- }
- // Specify the MaxTime option to limit the amount of time the operation can run on the server
- opts := options.CreateIndexes().SetMaxTime(2 * time.Second)
- names, err := indexView.CreateMany(context.TODO(), models, opts)
- if err != nil {
- log.Fatal(err)
- }
- log.Print("create indexes:")
- for _, name := range names {
- log.Println(name)
- }
- }
- }
- func initIndexManufacturers() {
- collection := database.Collection(manufacturersCollectionName)
- indexView := collection.Indexes()
- ctx, _ := context.WithTimeout(context.Background(), timeout)
- cursor, err := indexView.List(ctx)
- if err != nil {
- log.Fatal(err)
- }
- defer cursor.Close(ctx)
- myIndexes := make([]string, 0)
- for cursor.Next(ctx) {
- var index bson.M
- if err = cursor.Decode(&index); err != nil {
- log.Fatal(err)
- }
- myIndexes = append(myIndexes, index["name"].(string))
- }
- for _, name := range myIndexes {
- log.Println(name)
- }
- if !slicesutils.Contains(myIndexes, "name") {
- ctx, _ = context.WithTimeout(context.Background(), timeout)
- models := []mongo.IndexModel{
- {
- Keys: bson.D{{"name", 1}},
- Options: options.Index().SetUnique(true).SetName("name").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
- },
- }
- // Specify the MaxTime option to limit the amount of time the operation can run on the server
- opts := options.CreateIndexes().SetMaxTime(2 * time.Second)
- names, err := indexView.CreateMany(context.TODO(), models, opts)
- if err != nil {
- log.Fatal(err)
- }
- log.Print("create indexes:")
- for _, name := range names {
- log.Println(name)
- }
- }
- }
- func initTags() {
- ctx, _ := context.WithTimeout(context.Background(), timeout)
- tagsCollection := database.Collection(tagsCollectionName)
- cursor, err := tagsCollection.Find(ctx, bson.M{})
- if err != nil {
- log.Fatal(err)
- }
- defer cursor.Close(ctx)
- for cursor.Next(ctx) {
- var tag bson.M
- if err = cursor.Decode(&tag); err != nil {
- log.Fatal(err)
- } else {
- tags = append(tags, tag["name"].(string))
- }
- }
- }
- func initManufacturers() {
- ctx, _ := context.WithTimeout(context.Background(), timeout)
- manufacturersCollection := database.Collection(manufacturersCollectionName)
- cursor, err := manufacturersCollection.Find(ctx, bson.M{})
- if err != nil {
- log.Fatal(err)
- }
- defer cursor.Close(ctx)
- for cursor.Next(ctx) {
- var manufacturer bson.M
- if err = cursor.Decode(&manufacturer); err != nil {
- log.Fatal(err)
- } else {
- manufacturers = append(manufacturers, manufacturer["name"].(string))
- }
- }
- }
- // AddFile adding a file to the storage, stream like
- func AddFile(filename string, reader io.Reader) (string, error) {
- uploadOpts := options.GridFSUpload().SetMetadata(bson.D{{"tag", "tag"}})
- fileID, err := bucket.UploadFromStream(filename, reader, uploadOpts)
- if err != nil {
- fmt.Printf("error: %s\n", err.Error())
- return "", err
- }
- log.Printf("Write file to DB was successful. File id: %s \n", fileID)
- id := fileID.String()
- return id, nil
- }
- // CreateSchematic creating a new schematic in the database
- func CreateSchematic(schematic model.Schematic) (string, error) {
- for _, tag := range schematic.Tags {
- if !slicesutils.Contains(tags, tag) {
- CreateTag(tag)
- }
- }
- if !slicesutils.Contains(manufacturers, schematic.Manufacturer) {
- CreateManufacturer(schematic.Manufacturer)
- }
- ctx, _ := context.WithTimeout(context.Background(), timeout)
- collection := database.Collection(schematicsCollectionName)
- result, err := collection.InsertOne(ctx, schematic)
- if err != nil {
- fmt.Printf("error: %s\n", err.Error())
- return "", err
- }
- filter := bson.M{"_id": result.InsertedID}
- err = collection.FindOne(ctx, filter).Decode(&schematic)
- if err != nil {
- fmt.Printf("error: %s\n", err.Error())
- return "", err
- }
- return result.InsertedID.(string), nil
- }
- // CreateTag create a new tag in the storage
- func CreateTag(tag string) error {
- tag = strings.ToLower(tag)
- ctx, _ := context.WithTimeout(context.Background(), timeout)
- collection := database.Collection(tagsCollectionName)
- tagModel := bson.M{"name": tag}
- _, err := collection.InsertOne(ctx, tagModel)
- if err != nil {
- fmt.Printf("error: %s\n", err.Error())
- return err
- }
- tags = append(tags, tag)
- return nil
- }
- // CreateManufacturer create a new manufacturer in the storage
- func CreateManufacturer(manufacturer string) error {
- ctx, _ := context.WithTimeout(context.Background(), timeout)
- collection := database.Collection(manufacturersCollectionName)
- manufacturerModel := bson.M{"name": manufacturer}
- _, err := collection.InsertOne(ctx, manufacturerModel)
- if err != nil {
- fmt.Printf("error: %s\n", err.Error())
- return err
- }
- manufacturers = append(manufacturers, manufacturer)
- return nil
- }
- func GetTags() []string {
- return tags
- }
- func GetManufacturers() []string {
- return manufacturers
- }
- func GetTagsCount() int {
- return len(tags)
- }
- func GetManufacturersCount() int {
- return len(manufacturers)
- }
- func DropAll() {
- ctx, _ := context.WithTimeout(context.Background(), timeout)
- collectionNames, err := database.ListCollectionNames(ctx, bson.D{}, &options.ListCollectionsOptions{})
- if err != nil {
- log.Fatal(err)
- }
- for _, name := range collectionNames {
- collection := database.Collection(name)
- err = collection.Drop(ctx)
- if err != nil {
- log.Fatal(err)
- }
- }
- }
|