Can I do a text query with the mongodb c# driver

前端 未结 5 2061
萌比男神i
萌比男神i 2020-12-12 17:52

Is there a way to submit a query that is expressed in the shell query syntax to the mongo c# driver

For example Something like

Coll.find { \"myrecs\"         


        
相关标签:
5条回答
  • 2020-12-12 18:05

    The QueryComplete class seems to have been deprecated. Use MongoDB.Driver.QueryDocument instead. As below:

    BsonDocument document = MongoDB.Bson.Serialization.BsonSerializer.Deserialize<BsonDocument>("{ name : value }");
    QueryDocument queryDoc = new QueryDocument(document);
    MongoCursor toReturn = collection.Find(queryDoc);
    
    0 讨论(0)
  • 2020-12-12 18:06

    There is no exact same functionality you want.

    But you can create BsonDocument from json for query:

    var jsonQuery = "{ x : 3, y : 'abc' }";
    BsonDocument doc = MongoDB.Bson.Serialization
                       .BsonSerializer.Deserialize<BsonDocument>(jsonQuery);
    

    And after that you can create query from BsonDocument:

    var query = new QueryComplete(doc); // or probably Query.Wrap(doc);
    

    The same you can do for the sort expression:

    var jsonOrder = "{ x : 1 }";
    BsonDocument orderDoc = BsonSerializer.Deserialize<BsonDocument>(jsonQuery);
    
    var sortExpr = new SortByWrapper(orderDoc);
    

    Also you can create extension method for the MongoCollection like this:

    public static List<T> GetItems<T>(this MongoCollection collection, string queryString, string orderString) where T : class 
    {
        var queryDoc = BsonSerializer.Deserialize<BsonDocument>(queryString);
        var orderDoc = BsonSerializer.Deserialize<BsonDocument>(orderString);
    
        //as of version 1.8 you should use MongoDB.Driver.QueryDocument instead (thanks to @Erik Hunter)
        var query = new QueryComplete(queryDoc);
        var order = new SortByWrapper(orderDoc);
    
        var cursor = collection.FindAs<T>(query);
        cursor.SetSortOrder(order);
    
        return cursor.ToList();
    }
    

    I didn't test the code above. Will do it later if need.

    Update:

    Just tested the code above, it's working!

    You can use it like this:

    var server = MongoServer.Create("mongodb://localhost:27020");
    var collection= server.GetDatabase("examples").GetCollection("SO");
    
    var items = collection.GetItems<DocType>("{ x : 3, y : 'abc' }", "{ x : 1 }");
    
    0 讨论(0)
  • 2020-12-12 18:06

    Here are a few routines I use for converting from string and from .NET objects to BSON queries (this is part of business object wrapper so a couple of refs to that class):

        public QueryDocument GetQueryFromString(string jsonQuery)
        {
            return new QueryDocument(BsonSerializer.Deserialize<BsonDocument>(jsonQuery));
        }
    
        public IEnumerable<T> QueryFromString<T>(string jsonQuery, string collectionName = null)
        {
            if (string.IsNullOrEmpty(collectionName))
                collectionName = this.CollectionName;
    
            var query = GetQueryFromString(jsonQuery);            
            var items = Database.GetCollection<T>(collectionName).Find(query);
    
            return items as IEnumerable<T>;
        }
    
    
        public IEnumerable<T> QueryFromObject<T>(object queryObject, string collectionName = null)
        {
            if (string.IsNullOrEmpty(collectionName))
                collectionName = this.CollectionName;
    
            var query = new QueryDocument(queryObject.ToBsonDocument());
            var items = Database.GetCollection<T>(collectionName).Find(query);
    
            return items as IEnumerable<T>;
        }
    

    Using these it's pretty easy to query via string or object parms:

    var questionBus = new busQuestion();           
    var json = "{ QuestionText: /elimination/, GroupName: \"Elimination\" }";
    var questions = questionBus.QueryFromString<Question>(json);
    
    foreach(var question in questions) { ... }
    

    or using object syntax:

    var questionBus = new busQuestion();            
    var query = new {QuestionText = new BsonRegularExpression("/elimination/"), 
                     GroupName = "Elimination"};
    var questions = questionBus.QueryFromObject<Question>(query);
    
    foreach(var question in questions) { ... }
    

    I like the object syntax simply because it's a bit easier to write out in C# code than dealing with embedded quotes in JSON strings if they are handcoded.

    0 讨论(0)
  • 2020-12-12 18:09

    Using the official C# driver, you'd do something like this:

    var server = MongoServer.Create("mongodb://localhost:27017");
    var db = server.GetDatabase("mydb");
    var col = db.GetCollection("col");
    
    var query = Query.And(Query.EQ("x", 3), Query.EQ("y", "abc"));
    var resultsCursor = col.Find(query).SetSortOrder("x");
    var results = resultsCursor.ToList();
    

    The equivalent query from the shell would be:

    col.find({ x: 3, y: "abc" }).sort({ x: 1 })
    
    0 讨论(0)
  • 2020-12-12 18:15

    Here is a web service function I wrote which you can send in a filter query, limit, and skip for pagination and a sort query for any collection you want! It's generic and fast.

        /// <summary>
        /// This method returns data from a collection specified by data type
        /// </summary>
        /// <param name="dataType"></param>
        /// <param name="filter">filter is a json specified filter. one or more separated by commas.  example: { "value":"23" }  example: { "enabled":true, "startdate":"2015-10-10"}</param>
        /// <param name="limit">limit and skip are for pagination, limit is the number of results per page</param>
        /// <param name="skip">skip is is the page size * page. so limit of 100 should use skip 0,100,200,300,400, etc. which represent page 1,2,3,4,5, etc</param>
        /// <param name="sort">specify which fields to sort and direction example:  { "value":1 }  for ascending, {"value:-1} for descending</param>
        /// <returns></returns>
        [WebMethod]
        public string GetData(string dataType, string filter, int limit, int skip, string sort) {
            //example: limit of 100 and skip of 0 returns the first 100 records
    
            //get bsondocument from a collection dynamically identified by datatype
            try {
                MongoCollection<BsonDocument> col = MongoDb.GetConnection("qis").GetCollection<BsonDocument>(dataType);
                if (col == null) {
                    return "Error: Collection Not Found";
                }
    
                MongoCursor cursor = null;
                SortByWrapper sortExpr = null;
    
                //calc sort order
                try {
                    BsonDocument orderDoc = BsonSerializer.Deserialize<BsonDocument>(sort);
                    sortExpr = new SortByWrapper(orderDoc);
                } catch { }
    
                //create a query from the filter if one is specified
                try {
                    if (filter != "") {
                        //sample filter: "{tags:'dog'},{enabled:true}"
                        BsonDocument query = BsonSerializer.Deserialize<BsonDocument>(filter);
                        QueryDocument queryDoc = new QueryDocument(query);
                        cursor = col.Find(queryDoc).SetSkip(skip).SetLimit(limit);
    
                        if (sortExpr != null) {
                            cursor.SetSortOrder(sortExpr);
                        }
    
                        return cursor.ToJson();
                    }
                } catch{}
    
    
                //if no filter specified or the filter failed just return all
                cursor = col.FindAll().SetSkip(skip).SetLimit(limit);
    
                if (sortExpr != null) {
                    cursor.SetSortOrder(sortExpr);
                }
    
                return cursor.ToJson();
            } catch(Exception ex) {
                return "Exception: " + ex.Message;
            }
        }
    

    Assuming I had these records in my collection called "mytest2":

    [{ "_id" : ObjectId("54ff7b1e5cc61604f0bc3016"), "timestamp" : "2015-01-10 10:10:10", "value" : "23" }, 
     { "_id" : ObjectId("54ff7b415cc61604f0bc3017"), "timestamp" : "2015-01-10 10:10:11", "value" : "24" }, 
     { "_id" : ObjectId("54ff7b485cc61604f0bc3018"), "timestamp" : "2015-01-10 10:10:12", "value" : "25" }, 
     { "_id" : ObjectId("54ff7b4f5cc61604f0bc3019"), "timestamp" : "2015-01-10 10:10:13", "value" : "26" }]
    

    I could make the web service call with the following parameters to return 100 records starting with the first page where value >= 23 and value <= 26, in descending order

    dataType: mytest2
    filter: { value: {$gte: 23}, value: {$lte: 26} }
    limit: 100
    skip: 0
    sort: { "value": -1 }
    

    Enjoy!

    0 讨论(0)
提交回复
热议问题