|
16 | 16 | using System;
|
17 | 17 | using System.Collections.Generic;
|
18 | 18 | using System.Linq;
|
| 19 | +using System.Reflection; |
19 | 20 | using System.Text;
|
20 | 21 | using System.Threading;
|
21 | 22 | using System.Threading.Tasks;
|
@@ -255,6 +256,60 @@ public override IAggregateFluent<TNewResult> Facet<TNewResult>(
|
255 | 256 | return AppendStage<TNewResult>(stage);
|
256 | 257 | }
|
257 | 258 |
|
| 259 | + public override IAggregateFluent<TNewResult> GraphLookup<TNewResult, TFrom, TConnect, TConnectFrom, TStartWith, TAs, TAsEnumerable>( |
| 260 | + IMongoCollection<TFrom> from, |
| 261 | + FieldDefinition<TFrom, TConnectFrom> connectFromField, |
| 262 | + FieldDefinition<TFrom, TConnect> connectToField, |
| 263 | + AggregateExpressionDefinition<TResult, TStartWith> startWith, |
| 264 | + FieldDefinition<TNewResult, TAsEnumerable> @as, |
| 265 | + FieldDefinition<TAs, int> depthField, |
| 266 | + AggregateGraphLookupOptions<TNewResult, TFrom, TConnect, TConnectFrom, TStartWith, TAs, TAsEnumerable> options = null) |
| 267 | + { |
| 268 | + Ensure.IsNotNull(from, nameof(from)); |
| 269 | + Ensure.IsNotNull(connectFromField, nameof(connectFromField)); |
| 270 | + Ensure.IsNotNull(connectToField, nameof(connectToField)); |
| 271 | + Ensure.IsNotNull(startWith, nameof(startWith)); |
| 272 | + Ensure.IsNotNull(@as, nameof(@as)); |
| 273 | + Ensure.That(from.Database.DatabaseNamespace.Equals(_collection.Database.DatabaseNamespace), "From collection must be from the same database.", nameof(from)); |
| 274 | + Ensure.That(IsTConnectOrEnumerableTConnect<TConnectFrom, TConnect>(), "TConnectFrom must be either TConnect or a type that implements IEnumerable<TConnect>.", nameof(TConnectFrom)); |
| 275 | + Ensure.That(IsTConnectOrEnumerableTConnect<TStartWith, TConnect>(), "TStartWith must be either TConnect or a type that implements IEnumerable<TConnect>.", nameof(TStartWith)); |
| 276 | + |
| 277 | + const string operatorName = "$graphLookup"; |
| 278 | + var stage = new DelegatedPipelineStageDefinition<TResult, TNewResult>( |
| 279 | + operatorName, |
| 280 | + (s, sr) => |
| 281 | + { |
| 282 | + var resultSerializer = s; |
| 283 | + var newResultSerializer = options?.NewResultSerializer ?? sr.GetSerializer<TNewResult>(); |
| 284 | + var fromSerializer = options?.FromSerializer ?? sr.GetSerializer<TFrom>(); |
| 285 | + var asSerializer = options?.AsSerializer ?? sr.GetSerializer<TAs>(); |
| 286 | + var renderedConnectToField = connectToField.Render(fromSerializer, sr); |
| 287 | + var renderedStartWith = startWith.Render(resultSerializer, sr); |
| 288 | + var renderedConnectFromField = connectFromField.Render(fromSerializer, sr); |
| 289 | + var renderedAs = @as.Render(newResultSerializer, sr); |
| 290 | + var renderedDepthField = depthField?.Render(asSerializer, sr); |
| 291 | + var renderedRestrictSearchWithMatch = options?.RestrictSearchWithMatch?.Render(fromSerializer, sr); |
| 292 | + var document = new BsonDocument |
| 293 | + { |
| 294 | + { operatorName, new BsonDocument |
| 295 | + { |
| 296 | + { "from", from.CollectionNamespace.CollectionName }, |
| 297 | + { "connectFromField", renderedConnectFromField.FieldName }, |
| 298 | + { "connectToField", renderedConnectToField.FieldName }, |
| 299 | + { "startWith", renderedStartWith }, |
| 300 | + { "as", renderedAs.FieldName }, |
| 301 | + { "depthField", () => renderedDepthField.FieldName, renderedDepthField != null }, |
| 302 | + { "maxDepth", () => options.MaxDepth.Value, options != null && options.MaxDepth.HasValue }, |
| 303 | + { "restrictSearchWithMatch", renderedRestrictSearchWithMatch, renderedRestrictSearchWithMatch != null } |
| 304 | + } |
| 305 | + } |
| 306 | + }; |
| 307 | + return new RenderedPipelineStageDefinition<TNewResult>(operatorName, document, newResultSerializer); |
| 308 | + }); |
| 309 | + |
| 310 | + return AppendStage<TNewResult>(stage); |
| 311 | + } |
| 312 | + |
258 | 313 | public override IAggregateFluent<TNewResult> Group<TNewResult>(ProjectionDefinition<TResult, TNewResult> group)
|
259 | 314 | {
|
260 | 315 | const string operatorName = "$group";
|
@@ -479,5 +534,22 @@ public override string ToString()
|
479 | 534 | sb.Append("])");
|
480 | 535 | return sb.ToString();
|
481 | 536 | }
|
| 537 | + |
| 538 | + // private methods |
| 539 | + private bool IsTConnectOrEnumerableTConnect<TConnectFrom, TConnect>() |
| 540 | + { |
| 541 | + if (typeof(TConnect) == typeof(TConnectFrom)) |
| 542 | + { |
| 543 | + return true; |
| 544 | + } |
| 545 | + |
| 546 | + var ienumerableTConnect = typeof(IEnumerable<>).MakeGenericType(typeof(TConnect)); |
| 547 | + if (typeof(TConnectFrom).GetTypeInfo().GetInterfaces().Contains(ienumerableTConnect)) |
| 548 | + { |
| 549 | + return true; |
| 550 | + } |
| 551 | + |
| 552 | + return false; |
| 553 | + } |
482 | 554 | }
|
483 | 555 | }
|
0 commit comments