mirror of https://github.com/grpc/grpc.git
parent
4d1fc55261
commit
5bd75d789c
7 changed files with 314 additions and 1 deletions
@ -0,0 +1,112 @@ |
|||||||
|
#region Copyright notice and license |
||||||
|
// Copyright 2015, Google Inc. |
||||||
|
// All rights reserved. |
||||||
|
// |
||||||
|
// Redistribution and use in source and binary forms, with or without |
||||||
|
// modification, are permitted provided that the following conditions are |
||||||
|
// met: |
||||||
|
// |
||||||
|
// * Redistributions of source code must retain the above copyright |
||||||
|
// notice, this list of conditions and the following disclaimer. |
||||||
|
// * Redistributions in binary form must reproduce the above |
||||||
|
// copyright notice, this list of conditions and the following disclaimer |
||||||
|
// in the documentation and/or other materials provided with the |
||||||
|
// distribution. |
||||||
|
// * Neither the name of Google Inc. nor the names of its |
||||||
|
// contributors may be used to endorse or promote products derived from |
||||||
|
// this software without specific prior written permission. |
||||||
|
// |
||||||
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||||
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
||||||
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
||||||
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
||||||
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
||||||
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
||||||
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
||||||
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
||||||
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
||||||
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
||||||
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||||
|
#endregion |
||||||
|
using System; |
||||||
|
using System.Runtime.InteropServices; |
||||||
|
using System.Threading; |
||||||
|
using System.Threading.Tasks; |
||||||
|
|
||||||
|
using Grpc.Core.Logging; |
||||||
|
using Grpc.Core.Utils; |
||||||
|
|
||||||
|
namespace Grpc.Core.Internal |
||||||
|
{ |
||||||
|
internal delegate void NativeMetadataInterceptor(IntPtr statePtr, IntPtr serviceUrlPtr, IntPtr callbackPtr, IntPtr userDataPtr, bool isDestroy); |
||||||
|
|
||||||
|
internal class NativeMetadataCredentialsPlugin |
||||||
|
{ |
||||||
|
const string GetMetadataExceptionMsg = "Exception occured in metadata credentials plugin."; |
||||||
|
static readonly ILogger Logger = GrpcEnvironment.Logger.ForType<NativeMetadataCredentialsPlugin>(); |
||||||
|
|
||||||
|
[DllImport("grpc_csharp_ext.dll")] |
||||||
|
static extern CredentialsSafeHandle grpcsharp_metadata_credentials_create_from_plugin(NativeMetadataInterceptor interceptor); |
||||||
|
|
||||||
|
[DllImport("grpc_csharp_ext.dll", CharSet = CharSet.Ansi)] |
||||||
|
static extern void grpcsharp_metadata_credentials_notify_from_plugin(IntPtr callbackPtr, IntPtr userData, MetadataArraySafeHandle metadataArray, StatusCode statusCode, string errorDetails); |
||||||
|
|
||||||
|
AsyncAuthInterceptor interceptor; |
||||||
|
GCHandle gcHandle; |
||||||
|
NativeMetadataInterceptor nativeInterceptor; |
||||||
|
CredentialsSafeHandle credentials; |
||||||
|
|
||||||
|
public NativeMetadataCredentialsPlugin(AsyncAuthInterceptor interceptor) |
||||||
|
{ |
||||||
|
this.interceptor = Preconditions.CheckNotNull(interceptor, "interceptor"); |
||||||
|
this.nativeInterceptor = NativeMetadataInterceptorHandler; |
||||||
|
|
||||||
|
// Make sure the callback doesn't get garbage collected until it is destroyed. |
||||||
|
this.gcHandle = GCHandle.Alloc(this.nativeInterceptor, GCHandleType.Normal); |
||||||
|
this.credentials = grpcsharp_metadata_credentials_create_from_plugin(nativeInterceptor); |
||||||
|
} |
||||||
|
|
||||||
|
public CredentialsSafeHandle Credentials |
||||||
|
{ |
||||||
|
get { return credentials; } |
||||||
|
} |
||||||
|
|
||||||
|
private void NativeMetadataInterceptorHandler(IntPtr statePtr, IntPtr serviceUrlPtr, IntPtr callbackPtr, IntPtr userDataPtr, bool isDestroy) |
||||||
|
{ |
||||||
|
if (isDestroy) |
||||||
|
{ |
||||||
|
gcHandle.Free(); |
||||||
|
return; |
||||||
|
} |
||||||
|
|
||||||
|
try |
||||||
|
{ |
||||||
|
string serviceUrl = Marshal.PtrToStringAnsi(serviceUrlPtr); |
||||||
|
StartGetMetadata(serviceUrl, callbackPtr, userDataPtr); |
||||||
|
} |
||||||
|
catch (Exception e) |
||||||
|
{ |
||||||
|
grpcsharp_metadata_credentials_notify_from_plugin(callbackPtr, userDataPtr, null, StatusCode.Unknown, GetMetadataExceptionMsg); |
||||||
|
Logger.Error(e, GetMetadataExceptionMsg); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
private async void StartGetMetadata(string serviceUrl, IntPtr callbackPtr, IntPtr userDataPtr) |
||||||
|
{ |
||||||
|
try |
||||||
|
{ |
||||||
|
var metadata = new Metadata(); |
||||||
|
await interceptor(serviceUrl, metadata); |
||||||
|
using (var metadataArray = MetadataArraySafeHandle.Create(metadata)) |
||||||
|
{ |
||||||
|
grpcsharp_metadata_credentials_notify_from_plugin(callbackPtr, userDataPtr, metadataArray, StatusCode.OK, null); |
||||||
|
} |
||||||
|
} |
||||||
|
catch (Exception e) |
||||||
|
{ |
||||||
|
grpcsharp_metadata_credentials_notify_from_plugin(callbackPtr, userDataPtr, null, StatusCode.Unknown, GetMetadataExceptionMsg); |
||||||
|
Logger.Error(e, GetMetadataExceptionMsg); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,100 @@ |
|||||||
|
#region Copyright notice and license |
||||||
|
|
||||||
|
// Copyright 2015, Google Inc. |
||||||
|
// All rights reserved. |
||||||
|
// |
||||||
|
// Redistribution and use in source and binary forms, with or without |
||||||
|
// modification, are permitted provided that the following conditions are |
||||||
|
// met: |
||||||
|
// |
||||||
|
// * Redistributions of source code must retain the above copyright |
||||||
|
// notice, this list of conditions and the following disclaimer. |
||||||
|
// * Redistributions in binary form must reproduce the above |
||||||
|
// copyright notice, this list of conditions and the following disclaimer |
||||||
|
// in the documentation and/or other materials provided with the |
||||||
|
// distribution. |
||||||
|
// * Neither the name of Google Inc. nor the names of its |
||||||
|
// contributors may be used to endorse or promote products derived from |
||||||
|
// this software without specific prior written permission. |
||||||
|
// |
||||||
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||||
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
||||||
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
||||||
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
||||||
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
||||||
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
||||||
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
||||||
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
||||||
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
||||||
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
||||||
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||||
|
|
||||||
|
#endregion |
||||||
|
|
||||||
|
using System; |
||||||
|
using System.Collections.Generic; |
||||||
|
using System.IO; |
||||||
|
using System.Linq; |
||||||
|
using System.Threading; |
||||||
|
using System.Threading.Tasks; |
||||||
|
using Grpc.Core; |
||||||
|
using Grpc.Core.Utils; |
||||||
|
using Grpc.Testing; |
||||||
|
using NUnit.Framework; |
||||||
|
|
||||||
|
namespace Grpc.IntegrationTesting |
||||||
|
{ |
||||||
|
public class MetadataCredentialsTest |
||||||
|
{ |
||||||
|
const string Host = "localhost"; |
||||||
|
Server server; |
||||||
|
Channel channel; |
||||||
|
TestService.ITestServiceClient client; |
||||||
|
|
||||||
|
[TestFixtureSetUp] |
||||||
|
public void Init() |
||||||
|
{ |
||||||
|
var serverCredentials = new SslServerCredentials(new[] { new KeyCertificatePair( |
||||||
|
File.ReadAllText(TestCredentials.ServerCertChainPath), |
||||||
|
File.ReadAllText(TestCredentials.ServerPrivateKeyPath)) }); |
||||||
|
server = new Server |
||||||
|
{ |
||||||
|
Services = { TestService.BindService(new TestServiceImpl()) }, |
||||||
|
Ports = { { Host, ServerPort.PickUnused, serverCredentials } } |
||||||
|
}; |
||||||
|
server.Start(); |
||||||
|
|
||||||
|
var options = new List<ChannelOption> |
||||||
|
{ |
||||||
|
new ChannelOption(ChannelOptions.SslTargetNameOverride, TestCredentials.DefaultHostOverride) |
||||||
|
}; |
||||||
|
|
||||||
|
var asyncAuthInterceptor = new AsyncAuthInterceptor(async (authUri, metadata) => |
||||||
|
{ |
||||||
|
await Task.Delay(100); // make sure the operation is asynchronous. |
||||||
|
metadata.Add("authorization", "SECRET_TOKEN"); |
||||||
|
}); |
||||||
|
|
||||||
|
var clientCredentials = CompositeCredentials.Create( |
||||||
|
new SslCredentials(File.ReadAllText(TestCredentials.ClientCertAuthorityPath)), |
||||||
|
new MetadataCredentials(asyncAuthInterceptor) |
||||||
|
); |
||||||
|
channel = new Channel(Host, server.Ports.Single().BoundPort, clientCredentials, options); |
||||||
|
client = TestService.NewClient(channel); |
||||||
|
} |
||||||
|
|
||||||
|
[TestFixtureTearDown] |
||||||
|
public void Cleanup() |
||||||
|
{ |
||||||
|
channel.ShutdownAsync().Wait(); |
||||||
|
server.ShutdownAsync().Wait(); |
||||||
|
} |
||||||
|
|
||||||
|
[Test] |
||||||
|
public void MetadataCredentials() |
||||||
|
{ |
||||||
|
var response = client.UnaryCall(new SimpleRequest { ResponseSize = 10 }); |
||||||
|
Assert.AreEqual(10, response.Payload.Body.Length); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue