Home Backend Development C#.Net Tutorial Extending QueryBuilder of MongoDB C# Driver

Extending QueryBuilder of MongoDB C# Driver

Feb 28, 2017 am 11:53 AM

Extending the MongoDB C# Driver's QueryBuilder

Since I don't want to directly hardcode strings like "ClassA.MemberA.MemberB", I wrote the following classes for the following common scenarios:
1. Convert expression to string function: ExpToStr()
2. Collection function: When there are collection members, you can use this class, which will return a QueryCollection object. The code of this class is appended
3. CollectionAs function: When inheritance is used, you want to convert the base class to a subclass and return the QueryCollection of the subclass
Usage example:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

//获得表达式的字符串形式

1. QueryEx<ClassA>.ExpToStr ((ClassA m)=> m.MemberA.MemberB.MemberC)

 

 

//集合.成员.字段

//PoppedSegments为集合,AssignedNetwork.Name为成员

//将返回PoppedSegments.AssignedNetwork.Name

2. QueryEx<MDDelivery>.Collection(x => x.PoppedSegments).Matches(p => p.AssignedNetwork.Name, bsonRegex),

 

 

//子类集合.成员.字段

//STPaymentTransaction为基类,STPaymentCompanyCredit为子类,Company字段在子类中

//将返回Payments.Company.Name

3. QueryEx<MDDelivery>.CollectionAs<STPaymentTransaction, STPaymentCompanyCredit>(x=>x.Payments).Matches(p=>p.Company.Name, bsonRegex)

 

 

//集合.集合.成员.字段

//Parcels为集合,STCustomPropertyRuntime为基类,STNumericPropertyRuntime为子类,CustomProps为STNumericPropertyRuntime中成员,Value为CustomProp中成员

//将返回Parcels.CustomProps.Value

4. QueryEx<MDDelivery>.Collection(x=>x.Parcels).CollectionMemberAs<STCustomPropertyRuntime, STNumericPropertyRuntime>(p=>p.CustomProps).Matches(p=>p.Value, bsonRegex),

Copy after login


Implementation code:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

84

85

86

87

88

89

90

91

92

93

94

95

96

97

98

99

100

101

102

103

104

105

106

107

108

109

110

111

112

113

114

115

116

117

118

119

120

121

122

123

124

125

126

127

128

129

130

131

132

133

134

135

136

137

138

139

140

141

142

143

144

145

146

147

148

149

150

151

152

153

154

155

156

157

158

159

160

161

162

163

164

165

166

167

168

169

170

171

172

173

174

175

176

177

178

179

180

181

182

183

184

185

186

187

188

189

190

191

192

193

194

195

196

197

198

199

200

201

202

203

204

205

206

207

208

209

210

211

212

213

214

215

216

217

218

219

220

221

222

223

224

225

226

227

228

229

230

231

232

233

234

235

236

237

238

239

240

241

242

243

244

245

246

247

248

249

250

251

252

253

254

255

256

257

258

259

260

261

262

263

264

265

266

267

268

269

270

271

272

273

274

275

276

277

278

279

280

281

282

283

284

285

286

287

288

289

290

291

292

293

294

295

296

297

298

299

300

301

302

303

304

305

306

307

308

309

310

311

312

313

314

315

316

317

318

319

320

321

322

public class QueryEx<TDocument>

    {

        public static QueryCollection<TDocument, TCollection> Collection<TCollection>(

            Expression<Func<TDocument, IEnumerable<TCollection>>> collectionExpression)

        {

            return new QueryCollection<TDocument, TCollection>(collectionExpression);

        }

 

 

        //for those cases using inheritance

        //e.g STPaymentTransaction

        //Payments will return STPaymentTransaction

        //need to cast to sub classes(STPaymentCompanyCredit) so that be able to filter by child members (e.g. Company)

        public static QueryCollection<TDocument, TSub> CollectionAs<TCollection, TSub>(

            Expression<Func<TDocument, IEnumerable<TCollection>>> collectionExpression)

            where TSub : TCollection

        {

            var argParam = Expression.Parameter(typeof (TDocument), "x");

            var memberStr = ExpToStr(collectionExpression);

            MemberExpression nameProperty = Expression.Property(argParam, memberStr);

 

 

            var subExp = Expression.Convert(nameProperty, typeof(IEnumerable<TSub>));

 

 

            var exp = Expression.Lambda<Func<TDocument, IEnumerable<TSub>>>(

                subExp,

                argParam);

 

 

            return new QueryCollection<TDocument, TSub>(exp);

        }

 

 

        /// <summary>

        /// return string value for a expression:

        /// for s.Name.Val1.Val2 will return Name.Val1.Val2

        /// </summary>

        /// <typeparam name="MDClass"></typeparam>

        /// <typeparam name="Member"></typeparam>

        /// <param name="exp"></param>

        /// <returns></returns>

        public static string ExpToStr<TDocument, Member>(Expression<Func<TDocument, Member>> exp)

        {

            return new QueryExpressionHelper().MemberExpression(exp);

        }

    }

 

 

 

 

 

 

 

 

public class QueryCollection<TDocument, TCollection>

    {

        private readonly QueryExpressionHelper _queryExpression;

        private string _collectionName;

 

 

        public string Context

        {

            get { return _collectionName; }

        }

 

 

        public QueryCollection(Expression<Func<TDocument, IEnumerable<TCollection>>> collectionExpression)

        {

            _queryExpression = new QueryExpressionHelper();

            _collectionName = _queryExpression.MemberExpression(collectionExpression);

        }

 

 

        public QueryMember<TCollection, TMember> Member<TMember>(Expression<Func<TCollection, TMember>> exp)

        {

            var expStr = QueryEx<TCollection>.ExpToStr(exp);

            var context = string.Format("{0}.{1}", _collectionName, expStr);

            var obj = new QueryMember<TCollection, TMember>(context);

 

 

            return obj;

        }

 

 

        public QueryCollection<TCollection, TMember> CollectionMember<TMember>(

            Expression<Func<TCollection, IEnumerable<TMember>>> exp)

        {

            var expStr = QueryEx<TCollection>.ExpToStr(exp);

            var obj = new QueryCollection<TCollection, TMember>(exp)

            {

                _collectionName = string.Format("{0}.{1}", _collectionName, expStr)

            };

 

 

            return obj;

        }

 

 

        /// <summary>

        /// this method only support 1 layer nested(not for Query Collection.Collection , but for Collection.Member)

        /// if member is collection and need convert to sub class

        /// </summary>

        /// <typeparam name="TMember">Base Type</typeparam>

        /// <typeparam name="TMemberSub">Child Class Type</typeparam>

        /// <param name="collectionExpression"></param>

        /// <returns></returns>

        public QueryCollection<TCollection, TMemberSub> CollectionMemberAs<TMember, TMemberSub>(

            Expression<Func<TCollection, IEnumerable<TMember>>> collectionExpression)

            where TMemberSub : TMember

        {

            var obj = QueryEx<TCollection>.CollectionAs<TMember, TMemberSub>(collectionExpression);

            obj._collectionName = string.Format("{0}.{1}", _collectionName, obj._collectionName);

 

 

            return obj;

        }

 

 

        public IMongoQuery LT<TMember>(Expression<Func<TCollection, TMember>> memberExpression, TMember value)

        {

            var memberName = _queryExpression.MemberExpression(memberExpression);

 

 

            return Query.LT(string.Format("{0}.{1}", _collectionName, memberName), BsonValue.Create(value));

        }

 

 

        public IMongoQuery LT<TValue>(Expression<Func<TCollection, IEnumerable<TValue>>> memberExpression, TValue value)

        {

            var memberName = _queryExpression.MemberExpression(memberExpression);

 

 

            return Query.LT(string.Format("{0}.{1}", _collectionName, memberName), BsonValue.Create(value));

        }

 

 

        public IMongoQuery EQ<TMember>(Expression<Func<TCollection, TMember>> memberExpression, TMember value)

        {

            var memberName = _queryExpression.MemberExpression(memberExpression);

 

 

            return Query.EQ(string.Format("{0}.{1}", _collectionName, memberName), BsonValue.Create(value));

        }

 

 

        public IMongoQuery EQ<TValue>(Expression<Func<TCollection, IEnumerable<TValue>>> memberExpression, TValue value)

        {

            var memberName = _queryExpression.MemberExpression(memberExpression);

 

 

            return Query.EQ(string.Format("{0}.{1}", _collectionName, memberName), BsonValue.Create(value));

        }

 

 

        public IMongoQuery NE<TMember>(Expression<Func<TCollection, TMember>> memberExpression, TMember value)

        {

            var memberName = _queryExpression.MemberExpression(memberExpression);

 

 

            return Query.NE(string.Format("{0}.{1}", _collectionName, memberName), BsonValue.Create(value));

        }

 

 

        public IMongoQuery NE<TValue>(Expression<Func<TCollection, IEnumerable<TValue>>> memberExpression, TValue value)

        {

            var memberName = _queryExpression.MemberExpression(memberExpression);

 

 

            return Query.NE(string.Format("{0}.{1}", _collectionName, memberName), BsonValue.Create(value));

        }

 

 

        public IMongoQuery In<TMember>(Expression<Func<TCollection, TMember>> memberExpression, params TMember[] values)

        {

            return In<TMember>(memberExpression, new List<TMember>(values));

        }

 

 

        public IMongoQuery In<TMember>(Expression<Func<TCollection, TMember>> memberExpression,

            IEnumerable<TMember> values)

        {

            var memberName = _queryExpression.MemberExpression(memberExpression);

            return Query.In(string.Format("{0}.{1}", _collectionName, memberName), values.Select(x => BsonValue.Create(x)));

        }

 

 

        public IMongoQuery In<TCastC, TMember>(Expression<Func<TCastC, TMember>> memberExpression,

            IEnumerable<TMember> values) where TCastC : TCollection

        {

            var memberName = _queryExpression.MemberExpression(memberExpression);

            return Query.In(string.Format("{0}.{1}", _collectionName, memberName), values.Select(x => BsonValue.Create(x)));

        }

 

 

        public IMongoQuery In<TValue>(Expression<Func<TCollection, IEnumerable<TValue>>> memberExpression, IEnumerable<TValue> values)

        {

            var memberName = _queryExpression.MemberExpression(memberExpression);

            return Query.In(string.Format("{0}.{1}", _collectionName, memberName), values.Select(x => BsonValue.Create(x)));

        }

 

 

        public IMongoQuery In<TCastC, TValue>(Expression<Func<TCastC, IEnumerable<TValue>>> memberExpression, IEnumerable<TValue> values) where TCastC : TCollection

        {

            var memberName = _queryExpression.MemberExpression(memberExpression);

            return Query.In(string.Format("{0}.{1}", _collectionName, memberName), values.Select(x => BsonValue.Create(x)));

        }

 

 

 

 

        public IMongoQuery Matches<TMember>(Expression<Func<TCollection, TMember>> memberExpression, BsonRegularExpression value)

        {

            var memberName = _queryExpression.MemberExpression(memberExpression);

 

 

            return Query.Matches(string.Format("{0}.{1}", _collectionName, memberName), value);

        }

 

 

        public IMongoQuery Matches<TValue>(Expression<Func<TCollection, IEnumerable<TValue>>> memberExpression, BsonRegularExpression value)

        {

            var memberName = _queryExpression.MemberExpression(memberExpression);

 

 

            return Query.Matches(string.Format("{0}.{1}", _collectionName, memberName), value);

        }

    }

 

 

 

 

 

 

 

 

public class QueryMember<TDocument, TCollection>

    {

        private readonly QueryExpressionHelper _queryExpression;

        private string _collectionName;

 

 

        public string Context

        {

            get { return _collectionName; }

        }

 

 

        public QueryMember(Expression<Func<TDocument, TCollection>> exp)

        {

            _queryExpression = new QueryExpressionHelper();

            _collectionName = _queryExpression.MemberExpression(exp);

        }

 

 

        public QueryMember(string context)

        {

            _collectionName = context;

        }

 

 

    }

 

 

 

 

public class QueryExpressionHelper

    {

        public string Context;

 

 

        public string MemberExpression<TMember>(Expression<TMember> expression)

        {

            MemberExpression me;

            switch (expression.Body.NodeType)

            {

                case ExpressionType.MemberAccess:

                    me = expression.Body as MemberExpression;

                    break;

                case ExpressionType.Convert:

                    dynamic convertedBody = expression.Body;

                    me = convertedBody.Operand as MemberExpression;

                    break;

 

 

                default:

                    throw new NotSupportedException(string.Format("Member with node type {0} is not supported. expression {1}",

                    expression.Body.NodeType, expression));

            }

            var stack = new Stack<string>();

 

 

            while (me != null)

            {

                stack.Push(me.Member.Name);

                me = me.Expression as MemberExpression;

            }

 

 

            var expStr = string.Join(".", stack.ToArray());

            return expStr;

 

 

            

        }

    }

 

 

 

 

public static class QueryMoney

    {

        public static IMongoQuery Value(string name, double val)

        {

            var accuracy = 0.005;

 

 

            return Query.And(

                Query.LT(name, new BsonDouble(val + accuracy)),

                Query.GT(name, new BsonDouble(val - accuracy)));

        }

    }

Copy after login

The above is the content of extending QueryBuilder of MongoDB C# Driver. For more related content, please pay attention to the PHP Chinese website (www.php.cn)!


Statement of this Website
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn

Hot AI Tools

Undresser.AI Undress

Undresser.AI Undress

AI-powered app for creating realistic nude photos

AI Clothes Remover

AI Clothes Remover

Online AI tool for removing clothes from photos.

Undress AI Tool

Undress AI Tool

Undress images for free

Clothoff.io

Clothoff.io

AI clothes remover

AI Hentai Generator

AI Hentai Generator

Generate AI Hentai for free.

Hot Article

R.E.P.O. Energy Crystals Explained and What They Do (Yellow Crystal)
2 weeks ago By 尊渡假赌尊渡假赌尊渡假赌
Repo: How To Revive Teammates
4 weeks ago By 尊渡假赌尊渡假赌尊渡假赌
Hello Kitty Island Adventure: How To Get Giant Seeds
3 weeks ago By 尊渡假赌尊渡假赌尊渡假赌

Hot Tools

Notepad++7.3.1

Notepad++7.3.1

Easy-to-use and free code editor

SublimeText3 Chinese version

SublimeText3 Chinese version

Chinese version, very easy to use

Zend Studio 13.0.1

Zend Studio 13.0.1

Powerful PHP integrated development environment

Dreamweaver CS6

Dreamweaver CS6

Visual web development tools

SublimeText3 Mac version

SublimeText3 Mac version

God-level code editing software (SublimeText3)

Active Directory with C# Active Directory with C# Sep 03, 2024 pm 03:33 PM

Guide to Active Directory with C#. Here we discuss the introduction and how Active Directory works in C# along with the syntax and example.

Access Modifiers in C# Access Modifiers in C# Sep 03, 2024 pm 03:24 PM

Guide to the Access Modifiers in C#. We have discussed the Introduction Types of Access Modifiers in C# along with examples and outputs.

Random Number Generator in C# Random Number Generator in C# Sep 03, 2024 pm 03:34 PM

Guide to Random Number Generator in C#. Here we discuss how Random Number Generator work, concept of pseudo-random and secure numbers.

C# Data Grid View C# Data Grid View Sep 03, 2024 pm 03:32 PM

Guide to C# Data Grid View. Here we discuss the examples of how a data grid view can be loaded and exported from the SQL database or an excel file.

C# StringReader C# StringReader Sep 03, 2024 pm 03:23 PM

Guide to C# StringReader. Here we discuss a brief overview on C# StringReader and its working along with different Examples and Code.

Patterns in C# Patterns in C# Sep 03, 2024 pm 03:33 PM

Guide to Patterns in C#. Here we discuss the introduction and top 3 types of Patterns in C# along with its examples and code implementation.

C# Serialization C# Serialization Sep 03, 2024 pm 03:30 PM

Guide to C# Serialization. Here we discuss the introduction, steps of C# serialization object, working, and example respectively.

C# StringWriter C# StringWriter Sep 03, 2024 pm 03:23 PM

Guide to C# StringWriter. Here we discuss a brief overview on C# StringWriter Class and its working along with different Examples and Codes.

See all articles