| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349 | package daoimport (	"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.Minuteconst attachmentsCollectionName = "attachments"const schematicsCollectionName = "schematics"const tagsCollectionName = "tags"const manufacturersCollectionName = "manufacturers"var client *mongo.Clientvar mongoConfig config.MongoDBvar bucket gridfs.Bucketvar database mongo.Databasevar tags []stringvar manufacturers []string// InitDB initialise the mongodb connection, build up all collections and indexesfunc 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 likefunc 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 databasefunc 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 storagefunc 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 storagefunc 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)		}	}}
 |