|
@@ -30,6 +30,8 @@ const schematicsCollectionName = "schematics"
|
|
|
const tagsCollectionName = "tags"
|
|
|
const manufacturersCollectionName = "manufacturers"
|
|
|
const usersCollectionName = "users"
|
|
|
+const effectsCollectionName = "effects"
|
|
|
+const effectTypesCollectionName = "effectTypes"
|
|
|
|
|
|
// MongoDAO a mongodb based dao
|
|
|
type MongoDAO struct {
|
|
@@ -76,6 +78,8 @@ func (m *MongoDAO) InitDAO(MongoConfig config.MongoDB) {
|
|
|
m.initIndexSchematics()
|
|
|
m.initIndexTags()
|
|
|
m.initIndexManufacturers()
|
|
|
+ m.initIndexEffectTypes()
|
|
|
+ m.initIndexEffects()
|
|
|
|
|
|
m.tags = model.NewTags()
|
|
|
m.manufacturers = model.NewManufacturers()
|
|
@@ -222,6 +226,104 @@ func (m *MongoDAO) initIndexManufacturers() {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
+func (m *MongoDAO) initIndexEffects() {
|
|
|
+ collection := m.database.Collection(effectsCollectionName)
|
|
|
+ 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))
|
|
|
+ }
|
|
|
+ if !slicesutils.Contains(myIndexes, "effectType") {
|
|
|
+ ctx, _ = context.WithTimeout(context.Background(), timeout)
|
|
|
+ models := []mongo.IndexModel{
|
|
|
+ {
|
|
|
+ Keys: bson.D{{"effectType", 1}},
|
|
|
+ Options: options.Index().SetName("effectType").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
|
|
|
+ },
|
|
|
+ {
|
|
|
+ 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{{"comment", 1}},
|
|
|
+ Options: options.Index().SetName("comment").SetCollation(&options.Collation{Locale: "en", Strength: 2}),
|
|
|
+ },
|
|
|
+ {
|
|
|
+ Keys: bson.D{{"effectType", "text"}, {"manufacturer", "text"}, {"model", "text"}, {"tags", "text"}, {"comment", "text"}, {"connector", "text"}, {"current", "text"}, {"voltage", "text"}},
|
|
|
+ Options: options.Index().SetName("$text"),
|
|
|
+ },
|
|
|
+ }
|
|
|
+
|
|
|
+ // 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 (m *MongoDAO) initIndexEffectTypes() {
|
|
|
+ collection := m.database.Collection(effectTypesCollectionName)
|
|
|
+ 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))
|
|
|
+ }
|
|
|
+ if !slicesutils.Contains(myIndexes, "typeName") {
|
|
|
+ ctx, _ = context.WithTimeout(context.Background(), timeout)
|
|
|
+ models := []mongo.IndexModel{
|
|
|
+ {
|
|
|
+ Keys: bson.D{{"typeName", 1}},
|
|
|
+ Options: options.Index().SetName("typeName").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 (m *MongoDAO) initTags() {
|
|
|
m.reloadTags()
|
|
|
}
|
|
@@ -669,6 +771,211 @@ func (m *MongoDAO) GetManufacturersCount() int {
|
|
|
return len(m.manufacturers.List)
|
|
|
}
|
|
|
|
|
|
+//CreateEffectType cerating a new effect type
|
|
|
+func (m *MongoDAO) CreateEffectType(effectType model.EffectType) (string, error) {
|
|
|
+ ctx, _ := context.WithTimeout(context.Background(), timeout)
|
|
|
+ collection := m.database.Collection(effectTypesCollectionName)
|
|
|
+ result, err := collection.InsertOne(ctx, effectType)
|
|
|
+ if err != nil {
|
|
|
+ fmt.Printf("error: %s\n", err.Error())
|
|
|
+ return "", err
|
|
|
+ }
|
|
|
+ filter := bson.M{"_id": result.InsertedID}
|
|
|
+ err = collection.FindOne(ctx, filter).Decode(&effectType)
|
|
|
+ if err != nil {
|
|
|
+ fmt.Printf("error: %s\n", err.Error())
|
|
|
+ return "", err
|
|
|
+ }
|
|
|
+ switch v := result.InsertedID.(type) {
|
|
|
+ case primitive.ObjectID:
|
|
|
+ return v.Hex(), nil
|
|
|
+ }
|
|
|
+ return "", nil
|
|
|
+}
|
|
|
+
|
|
|
+// GetEffectTypes getting a sdingle schematic
|
|
|
+func (m *MongoDAO) GetEffectTypes() ([]model.EffectType, error) {
|
|
|
+ ctx, _ := context.WithTimeout(context.Background(), timeout)
|
|
|
+ collection := m.database.Collection(effectTypesCollectionName)
|
|
|
+ findOptions := options.Find()
|
|
|
+ // Sort by `price` field descending
|
|
|
+ findOptions.SetSort(bson.D{{"typeName", 1}})
|
|
|
+ queryDoc := bson.M{}
|
|
|
+ cursor, err := collection.Find(ctx, queryDoc, &options.FindOptions{Collation: &options.Collation{Locale: "en", Strength: 2}, Sort: bson.D{{"typeName", 1}}})
|
|
|
+ if err != nil {
|
|
|
+ log.Print(err)
|
|
|
+ return nil, err
|
|
|
+ }
|
|
|
+ defer cursor.Close(ctx)
|
|
|
+ effectTypes := make([]model.EffectType, 0)
|
|
|
+ for cursor.Next(ctx) {
|
|
|
+ var effectType model.EffectType
|
|
|
+ if err = cursor.Decode(&effectType); err != nil {
|
|
|
+ log.Print(err)
|
|
|
+ return nil, err
|
|
|
+ } else {
|
|
|
+ effectTypes = append(effectTypes, effectType)
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return effectTypes, nil
|
|
|
+}
|
|
|
+
|
|
|
+//CreateEffect cerating a new effect type
|
|
|
+func (m *MongoDAO) CreateEffect(effect model.Effect) (string, error) {
|
|
|
+
|
|
|
+ for _, tag := range effect.Tags {
|
|
|
+ if !m.tags.Contains(tag) {
|
|
|
+ m.CreateTag(tag)
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ if !m.manufacturers.Contains(effect.Manufacturer) {
|
|
|
+ m.CreateManufacturer(effect.Manufacturer)
|
|
|
+ }
|
|
|
+
|
|
|
+ ctx, _ := context.WithTimeout(context.Background(), timeout)
|
|
|
+ collection := m.database.Collection(effectsCollectionName)
|
|
|
+ result, err := collection.InsertOne(ctx, effect)
|
|
|
+ if err != nil {
|
|
|
+ fmt.Printf("error: %s\n", err.Error())
|
|
|
+ return "", err
|
|
|
+ }
|
|
|
+ filter := bson.M{"_id": result.InsertedID}
|
|
|
+ err = collection.FindOne(ctx, filter).Decode(&effect)
|
|
|
+ if err != nil {
|
|
|
+ fmt.Printf("error: %s\n", err.Error())
|
|
|
+ return "", err
|
|
|
+ }
|
|
|
+ switch v := result.InsertedID.(type) {
|
|
|
+ case primitive.ObjectID:
|
|
|
+ return v.Hex(), nil
|
|
|
+ }
|
|
|
+ return "", nil
|
|
|
+}
|
|
|
+
|
|
|
+//GetEffectsCount getting the count of effects regarding to the query
|
|
|
+func (m *MongoDAO) GetEffectsCount(query string) (int, error) {
|
|
|
+ ctx, _ := context.WithTimeout(context.Background(), timeout)
|
|
|
+ collection := m.database.Collection(effectsCollectionName)
|
|
|
+ queryDoc := bson.M{}
|
|
|
+ if query != "" {
|
|
|
+ var queryM map[string]interface{}
|
|
|
+ err := json.Unmarshal([]byte(query), &queryM)
|
|
|
+ if err != nil {
|
|
|
+ log.Print(err)
|
|
|
+ return 0, err
|
|
|
+ }
|
|
|
+ for k, v := range queryM {
|
|
|
+ if k == "$fulltext" {
|
|
|
+ queryDoc["$text"] = bson.M{"$search": v}
|
|
|
+ } else {
|
|
|
+ switch v := v.(type) {
|
|
|
+ // case float64:
|
|
|
+ // case int:
|
|
|
+ // case bool:
|
|
|
+ case string:
|
|
|
+ queryDoc[k] = bson.M{"$regex": v}
|
|
|
+ }
|
|
|
+ //queryDoc[k] = v
|
|
|
+ }
|
|
|
+ }
|
|
|
+ data, _ := json.Marshal(queryDoc)
|
|
|
+ log.Printf("mongoquery: %s\n", string(data))
|
|
|
+ }
|
|
|
+ n, err := collection.CountDocuments(ctx, queryDoc, &options.CountOptions{Collation: &options.Collation{Locale: "en", Strength: 2}})
|
|
|
+ if err != nil {
|
|
|
+ log.Print(err)
|
|
|
+ return 0, err
|
|
|
+ }
|
|
|
+ return int(n), nil
|
|
|
+}
|
|
|
+
|
|
|
+//GetEffects getting a list of effects regarding to the query
|
|
|
+func (m *MongoDAO) GetEffects(query string, offset int, limit int) (int, []model.Effect, error) {
|
|
|
+ ctx, _ := context.WithTimeout(context.Background(), timeout)
|
|
|
+ collection := m.database.Collection(effectsCollectionName)
|
|
|
+ queryDoc := bson.M{}
|
|
|
+ if query != "" {
|
|
|
+ var queryM map[string]interface{}
|
|
|
+ err := json.Unmarshal([]byte(query), &queryM)
|
|
|
+ if err != nil {
|
|
|
+ log.Print(err)
|
|
|
+ return 0, nil, err
|
|
|
+ }
|
|
|
+ for k, v := range queryM {
|
|
|
+ if k == "$fulltext" {
|
|
|
+ queryDoc["$text"] = bson.M{"$search": v}
|
|
|
+ } else {
|
|
|
+ switch v := v.(type) {
|
|
|
+ // case float64:
|
|
|
+ // case int:
|
|
|
+ // case bool:
|
|
|
+ case string:
|
|
|
+ queryDoc[k] = bson.M{"$regex": v}
|
|
|
+ }
|
|
|
+ //queryDoc[k] = v
|
|
|
+ }
|
|
|
+ }
|
|
|
+ data, _ := json.Marshal(queryDoc)
|
|
|
+ log.Printf("mongoquery: %s\n", string(data))
|
|
|
+ }
|
|
|
+ n, err := collection.CountDocuments(ctx, queryDoc, &options.CountOptions{Collation: &options.Collation{Locale: "en", Strength: 2}})
|
|
|
+ if err != nil {
|
|
|
+ log.Print(err)
|
|
|
+ return 0, nil, err
|
|
|
+ }
|
|
|
+ cursor, err := collection.Find(ctx, queryDoc, &options.FindOptions{Collation: &options.Collation{Locale: "en", Strength: 2}, Sort: bson.D{{"manufacturer", 1}, {"model", 1}}})
|
|
|
+ if err != nil {
|
|
|
+ log.Print(err)
|
|
|
+ return 0, nil, err
|
|
|
+ }
|
|
|
+ defer cursor.Close(ctx)
|
|
|
+ effects := make([]model.Effect, 0)
|
|
|
+ count := 0
|
|
|
+ docs := 0
|
|
|
+ for cursor.Next(ctx) {
|
|
|
+ if count >= offset {
|
|
|
+ if docs < limit {
|
|
|
+ var effect model.Effect
|
|
|
+ if err = cursor.Decode(&effect); err != nil {
|
|
|
+ log.Print(err)
|
|
|
+ return 0, nil, err
|
|
|
+ } else {
|
|
|
+ effects = append(effects, effect)
|
|
|
+ docs++
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ break
|
|
|
+ }
|
|
|
+ }
|
|
|
+ count++
|
|
|
+ }
|
|
|
+ return int(n), effects, nil
|
|
|
+}
|
|
|
+
|
|
|
+func (m *MongoDAO) GetEffect(effectID string) (model.Effect, error) {
|
|
|
+ ctx, _ := context.WithTimeout(context.Background(), timeout)
|
|
|
+ collection := m.database.Collection(effectsCollectionName)
|
|
|
+ objectID, _ := primitive.ObjectIDFromHex(effectID)
|
|
|
+ result := collection.FindOne(ctx, bson.M{"_id": objectID})
|
|
|
+ err := result.Err()
|
|
|
+ if err == mongo.ErrNoDocuments {
|
|
|
+ log.Print(err)
|
|
|
+ return model.Effect{}, ErrNoDocument
|
|
|
+ }
|
|
|
+ if err != nil {
|
|
|
+ log.Print(err)
|
|
|
+ return model.Effect{}, err
|
|
|
+ }
|
|
|
+ var effect model.Effect
|
|
|
+ if err := result.Decode(&effect); err != nil {
|
|
|
+ log.Print(err)
|
|
|
+ return model.Effect{}, err
|
|
|
+ } else {
|
|
|
+ return effect, nil
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
// CheckUser checking username and password... returns true if the user is active and the password for this user is correct
|
|
|
func (m *MongoDAO) CheckUser(username string, password string) bool {
|
|
|
username = strings.ToLower(username)
|