| 
					
				 | 
			
			
				@@ -0,0 +1,137 @@ 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+#region Copyright notice and license 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+ 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+// Copyright 2018 gRPC authors. 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+// 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+// Licensed under the Apache License, Version 2.0 (the "License"); 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+// you may not use this file except in compliance with the License. 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+// You may obtain a copy of the License at 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+// 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+//     http://www.apache.org/licenses/LICENSE-2.0 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+// 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+// Unless required by applicable law or agreed to in writing, software 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+// distributed under the License is distributed on an "AS IS" BASIS, 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+// See the License for the specific language governing permissions and 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+// limitations under the License. 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+ 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+#endregion 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+ 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+using System; 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+using System.Linq; 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+using Grpc.Core.Utils; 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+ 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+namespace Grpc.Core.Interceptors 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+{ 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+    /// <summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+    /// Extends the CallInvoker class to provide the interceptor facility on the client side. 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+    /// This is an EXPERIMENTAL API. 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+    /// </summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+    public static class CallInvokerExtensions 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+    { 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        /// <summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        /// Decorates an underlying <see cref="Grpc.Core.CallInvoker" /> to 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        /// intercept calls through a given interceptor. 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        /// </summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        private class InterceptingCallInvoker : CallInvoker 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        { 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            readonly CallInvoker invoker; 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            readonly Interceptor interceptor; 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+ 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            /// <summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            /// Creates a new instance of <see cref="Grpc.Core.Interceptors.CallInvokerExtensions.InterceptingCallInvoker" /> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            /// with the given underlying invoker and interceptor instances. 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            /// </summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            public InterceptingCallInvoker(CallInvoker invoker, Interceptor interceptor) 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            { 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                this.invoker = GrpcPreconditions.CheckNotNull(invoker, "invoker"); 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                this.interceptor = GrpcPreconditions.CheckNotNull(interceptor, "interceptor"); 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            } 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+ 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            /// <summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            /// Intercepts a simple blocking call with the registered interceptor. 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            /// </summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            public override TResponse BlockingUnaryCall<TRequest, TResponse>(Method<TRequest, TResponse> method, string host, CallOptions options, TRequest request) 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            { 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                return interceptor.BlockingUnaryCall( 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                    request, 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                    new ClientInterceptorContext<TRequest, TResponse>(method, host, options), 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                    (req, ctx) => invoker.BlockingUnaryCall(ctx.Method, ctx.Host, ctx.Options, req)); 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            } 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+ 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            /// <summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            /// Intercepts a simple asynchronous call with the registered interceptor. 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            /// </summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            public override AsyncUnaryCall<TResponse> AsyncUnaryCall<TRequest, TResponse>(Method<TRequest, TResponse> method, string host, CallOptions options, TRequest request) 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            { 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                return interceptor.AsyncUnaryCall( 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                    request, 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                    new ClientInterceptorContext<TRequest, TResponse>(method, host, options), 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                    (req, ctx) => invoker.AsyncUnaryCall(ctx.Method, ctx.Host, ctx.Options, req)); 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            } 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+ 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            /// <summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            /// Intercepts an asynchronous server streaming call with the registered interceptor. 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            /// </summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            public override AsyncServerStreamingCall<TResponse> AsyncServerStreamingCall<TRequest, TResponse>(Method<TRequest, TResponse> method, string host, CallOptions options, TRequest request) 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            { 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                return interceptor.AsyncServerStreamingCall( 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                    request, 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                    new ClientInterceptorContext<TRequest, TResponse>(method, host, options), 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                    (req, ctx) => invoker.AsyncServerStreamingCall(ctx.Method, ctx.Host, ctx.Options, req)); 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            } 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+ 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            /// <summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            /// Intercepts an asynchronous client streaming call with the registered interceptor. 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            /// </summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            public override AsyncClientStreamingCall<TRequest, TResponse> AsyncClientStreamingCall<TRequest, TResponse>(Method<TRequest, TResponse> method, string host, CallOptions options) 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            { 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                return interceptor.AsyncClientStreamingCall( 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                    new ClientInterceptorContext<TRequest, TResponse>(method, host, options), 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                    ctx => invoker.AsyncClientStreamingCall(ctx.Method, ctx.Host, ctx.Options)); 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            } 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+ 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            /// <summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            /// Intercepts an asynchronous duplex streaming call with the registered interceptor. 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            /// </summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            public override AsyncDuplexStreamingCall<TRequest, TResponse> AsyncDuplexStreamingCall<TRequest, TResponse>(Method<TRequest, TResponse> method, string host, CallOptions options) 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            { 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                return interceptor.AsyncDuplexStreamingCall( 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                    new ClientInterceptorContext<TRequest, TResponse>(method, host, options), 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                    ctx => invoker.AsyncDuplexStreamingCall(ctx.Method, ctx.Host, ctx.Options)); 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            } 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        } 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+ 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        /// <summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        /// Returns a <see cref="Grpc.Core.CallInvoker" /> instance that intercepts 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        /// the invoker with the given interceptor. 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        /// </summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        /// <param name="invoker">The underlying invoker to intercept.</param> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        /// <param name="interceptor">The interceptor to intercept calls to the invoker with.</param> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        public static CallInvoker Intercept(this CallInvoker invoker, Interceptor interceptor) 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        { 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            return new InterceptingCallInvoker(invoker, interceptor); 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        } 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+ 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        /// <summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        /// Returns a <see cref="Grpc.Core.CallInvoker" /> instance that intercepts 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        /// the invoker with the given interceptors. 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        /// </summary> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        /// <param name="invoker">The channel to intercept.</param> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        /// <param name="interceptors"> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        /// An array of interceptors to intercept the calls to the invoker with. 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        /// Control is passed to the interceptors in the order specified. 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        /// </param> 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        public static CallInvoker Intercept(this CallInvoker invoker, params Interceptor[] interceptors) 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        { 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            GrpcPreconditions.CheckNotNull(invoker, "invoker"); 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            GrpcPreconditions.CheckNotNull(interceptors, "interceptors"); 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+ 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            foreach (var interceptor in interceptors.Reverse()) 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            { 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+                invoker = Intercept(invoker, interceptor); 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            } 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+ 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+            return invoker; 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+        } 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+    } 
			 | 
		
	
		
			
				 | 
				 | 
			
			
				+} 
			 |