mirror of
https://github.com/FabInfra/capnproto-dotnetcore_Runtime.git
synced 2025-03-12 14:51:41 +01:00
error handling tests
This commit is contained in:
parent
e5902bf242
commit
04665f080a
@ -19,6 +19,7 @@
|
||||
<Compile Include="..\Capnp.Net.Runtime.Tests\Issue20.cs" Link="Issue20.cs" />
|
||||
<Compile Include="..\Capnp.Net.Runtime.Tests\Issue25.cs" Link="Issue25.cs" />
|
||||
<Compile Include="..\Capnp.Net.Runtime.Tests\JobUtil.cs" Link="JobUtil.cs" />
|
||||
<Compile Include="..\Capnp.Net.Runtime.Tests\LocalRpc.cs" Link="LocalRpc.cs" />
|
||||
<Compile Include="..\Capnp.Net.Runtime.Tests\MessageBuilderTests.cs" Link="MessageBuilderTests.cs" />
|
||||
<Compile Include="..\Capnp.Net.Runtime.Tests\ProvidedCapabilityMock.cs" Link="ProvidedCapabilityMock.cs" />
|
||||
<Compile Include="..\Capnp.Net.Runtime.Tests\ProvidedCapabilityMultiCallMock.cs" Link="ProvidedCapabilityMultiCallMock.cs" />
|
||||
@ -28,6 +29,7 @@
|
||||
<Compile Include="..\Capnp.Net.Runtime.Tests\SerializationTests.cs" Link="SerializationTests.cs" />
|
||||
<Compile Include="..\Capnp.Net.Runtime.Tests\TcpRpc.cs" Link="TcpRpc.cs" />
|
||||
<Compile Include="..\Capnp.Net.Runtime.Tests\TcpRpcAdvancedStuff.cs" Link="TcpRpcAdvancedStuff.cs" />
|
||||
<Compile Include="..\Capnp.Net.Runtime.Tests\TcpRpcErrorHandling.cs" Link="TcpRpcErrorHandling.cs" />
|
||||
<Compile Include="..\Capnp.Net.Runtime.Tests\TcpRpcInterop.cs" Link="TcpRpcInterop.cs" />
|
||||
<Compile Include="..\Capnp.Net.Runtime.Tests\TcpRpcPorted.cs" Link="TcpRpcPorted.cs" />
|
||||
<Compile Include="..\Capnp.Net.Runtime.Tests\TcpRpcStress.cs" Link="TcpRpcStress.cs" />
|
||||
@ -44,6 +46,7 @@
|
||||
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="16.2.0" />
|
||||
<PackageReference Include="MSTest.TestAdapter" Version="1.4.0" />
|
||||
<PackageReference Include="MSTest.TestFramework" Version="1.4.0" />
|
||||
<PackageReference Include="System.IO.Pipelines" Version="4.7.0" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
|
@ -24,6 +24,7 @@
|
||||
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="16.2.0" />
|
||||
<PackageReference Include="MSTest.TestAdapter" Version="1.4.0" />
|
||||
<PackageReference Include="MSTest.TestFramework" Version="1.4.0" />
|
||||
<PackageReference Include="System.IO.Pipelines" Version="4.7.0" />
|
||||
<PackageReference Include="System.Threading.Tasks.Dataflow" Version="4.10.0" />
|
||||
</ItemGroup>
|
||||
|
||||
|
31
Capnp.Net.Runtime.Tests/LocalRpc.cs
Normal file
31
Capnp.Net.Runtime.Tests/LocalRpc.cs
Normal file
@ -0,0 +1,31 @@
|
||||
using Capnp.Net.Runtime.Tests.GenImpls;
|
||||
using Capnp.Rpc;
|
||||
using Capnproto_test.Capnp.Test;
|
||||
using Microsoft.VisualStudio.TestTools.UnitTesting;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace Capnp.Net.Runtime.Tests
|
||||
{
|
||||
[TestClass]
|
||||
public class LocalRpc
|
||||
{
|
||||
[TestMethod]
|
||||
public void DeferredLocalAnswer()
|
||||
{
|
||||
var tcs = new TaskCompletionSource<int>();
|
||||
var impl = new TestPipelineImpl2(tcs.Task);
|
||||
var bproxy = BareProxy.FromImpl(impl);
|
||||
var proxy = bproxy.Cast<ITestPipeline>(true);
|
||||
var cap = proxy.GetCap(0, null).OutBox_Cap();
|
||||
var foo = cap.Foo(123, true);
|
||||
tcs.SetResult(0);
|
||||
Assert.IsTrue(foo.Wait(TestBase.MediumNonDbgTimeout));
|
||||
Assert.AreEqual("bar", foo.Result);
|
||||
}
|
||||
}
|
||||
}
|
608
Capnp.Net.Runtime.Tests/TcpRpcErrorHandling.cs
Normal file
608
Capnp.Net.Runtime.Tests/TcpRpcErrorHandling.cs
Normal file
@ -0,0 +1,608 @@
|
||||
using Capnp;
|
||||
using Capnp.Net.Runtime.Tests.GenImpls;
|
||||
using Capnp.Rpc;
|
||||
using Capnproto_test.Capnp.Test;
|
||||
using Microsoft.VisualStudio.TestTools.UnitTesting;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.IO;
|
||||
using System.IO.Pipelines;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace Capnp.Net.Runtime.Tests
|
||||
{
|
||||
[TestClass]
|
||||
public class TcpRpcErrorHandling: TestBase
|
||||
{
|
||||
class MemStreamEndpoint : IEndpoint
|
||||
{
|
||||
readonly FramePump _fromEnginePump;
|
||||
readonly BinaryReader _reader;
|
||||
|
||||
public bool Dismissed { get; private set; }
|
||||
|
||||
public MemStreamEndpoint()
|
||||
{
|
||||
var pipe = new Pipe();
|
||||
_fromEnginePump = new FramePump(pipe.Writer.AsStream());
|
||||
_reader = new BinaryReader(pipe.Reader.AsStream());
|
||||
}
|
||||
|
||||
public void Dismiss()
|
||||
{
|
||||
Dismissed = true;
|
||||
}
|
||||
|
||||
public void Forward(WireFrame frame)
|
||||
{
|
||||
_fromEnginePump.Send(frame);
|
||||
}
|
||||
|
||||
public WireFrame ReadNextFrame()
|
||||
{
|
||||
return _reader.ReadWireFrame();
|
||||
}
|
||||
}
|
||||
|
||||
class RpcEngineTester
|
||||
{
|
||||
readonly MemStreamEndpoint _fromEngine;
|
||||
|
||||
public RpcEngineTester()
|
||||
{
|
||||
Engine = new RpcEngine();
|
||||
_fromEngine = new MemStreamEndpoint();
|
||||
RealEnd = Engine.AddEndpoint(_fromEngine);
|
||||
}
|
||||
|
||||
public RpcEngine Engine { get; }
|
||||
public RpcEngine.RpcEndpoint RealEnd { get; }
|
||||
public bool IsDismissed => _fromEngine.Dismissed;
|
||||
|
||||
public void Send(Action<Message.WRITER> build)
|
||||
{
|
||||
var mb = MessageBuilder.Create();
|
||||
mb.InitCapTable();
|
||||
build(mb.BuildRoot<Message.WRITER>());
|
||||
RealEnd.Forward(mb.Frame);
|
||||
}
|
||||
|
||||
public void Recv(Action<Message.READER> verify)
|
||||
{
|
||||
var task = Task.Run(() => DeserializerState.CreateRoot(_fromEngine.ReadNextFrame()));
|
||||
Assert.IsTrue(task.Wait(MediumNonDbgTimeout), "reception timeout");
|
||||
verify(new Message.READER(task.Result));
|
||||
}
|
||||
|
||||
public void ExpectAbort()
|
||||
{
|
||||
Recv(_ => { Assert.AreEqual(Message.WHICH.Abort, _.which); });
|
||||
Assert.IsTrue(IsDismissed);
|
||||
Assert.ThrowsException<InvalidOperationException>(
|
||||
() => Send(_ => { _.which = Message.WHICH.Bootstrap; _.Bootstrap.QuestionId = 33; }));
|
||||
}
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void DuplicateQuestion1()
|
||||
{
|
||||
var tester = new RpcEngineTester();
|
||||
tester.Engine.Main = new TestInterfaceImpl(new Counters());
|
||||
|
||||
uint bootCapId = 0;
|
||||
|
||||
tester.Send(_ => { _.which = Message.WHICH.Bootstrap; _.Bootstrap.QuestionId = 99; });
|
||||
tester.Send(_ => { _.which = Message.WHICH.Bootstrap; _.Bootstrap.QuestionId = 99; });
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Return, _.which);
|
||||
Assert.AreEqual(Return.WHICH.Results, _.Return.which);
|
||||
Assert.AreEqual(1, _.Return.Results.CapTable.Count);
|
||||
bootCapId = _.Return.Results.CapTable[0].SenderHosted;
|
||||
});
|
||||
tester.ExpectAbort();
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void DuplicateQuestion2()
|
||||
{
|
||||
var tester = new RpcEngineTester();
|
||||
tester.Engine.Main = new TestInterfaceImpl(new Counters());
|
||||
|
||||
uint bootCapId = 0;
|
||||
|
||||
tester.Send(_ => { _.which = Message.WHICH.Bootstrap; _.Bootstrap.QuestionId = 99; });
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Return, _.which);
|
||||
Assert.AreEqual(Return.WHICH.Results, _.Return.which);
|
||||
Assert.AreEqual(1, _.Return.Results.CapTable.Count);
|
||||
bootCapId = _.Return.Results.CapTable[0].SenderHosted;
|
||||
});
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Call;
|
||||
_.Call.QuestionId = 99;
|
||||
_.Call.Target.which = MessageTarget.WHICH.ImportedCap;
|
||||
_.Call.Target.ImportedCap = bootCapId;
|
||||
_.Call.InterfaceId = ((TypeIdAttribute)typeof(ITestInterface).GetCustomAttributes(typeof(TypeIdAttribute), false)[0]).Id;
|
||||
_.Call.MethodId = 0;
|
||||
_.Call.Params.Content.Rewrap<TestInterface.Params_foo.WRITER>();
|
||||
});
|
||||
tester.ExpectAbort();
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void DuplicateQuestion3()
|
||||
{
|
||||
var tester = new RpcEngineTester();
|
||||
tester.Engine.Main = new TestInterfaceImpl(new Counters());
|
||||
|
||||
uint bootCapId = 0;
|
||||
|
||||
tester.Send(_ => { _.which = Message.WHICH.Bootstrap; _.Bootstrap.QuestionId = 99; });
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Return, _.which);
|
||||
Assert.AreEqual(Return.WHICH.Results, _.Return.which);
|
||||
Assert.AreEqual(1, _.Return.Results.CapTable.Count);
|
||||
bootCapId = _.Return.Results.CapTable[0].SenderHosted;
|
||||
});
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Call;
|
||||
_.Call.QuestionId = 42;
|
||||
_.Call.Target.which = MessageTarget.WHICH.ImportedCap;
|
||||
_.Call.Target.ImportedCap = bootCapId;
|
||||
_.Call.InterfaceId = ((TypeIdAttribute)typeof(ITestInterface).GetCustomAttributes(typeof(TypeIdAttribute), false)[0]).Id;
|
||||
_.Call.MethodId = 0;
|
||||
var wr = _.Call.Params.Content.Rewrap<TestInterface.Params_foo.WRITER>();
|
||||
wr.I = 123u;
|
||||
wr.J = true;
|
||||
});
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Return, _.which);
|
||||
});
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Call;
|
||||
_.Call.QuestionId = 42;
|
||||
_.Call.Target.which = MessageTarget.WHICH.ImportedCap;
|
||||
_.Call.Target.ImportedCap = bootCapId;
|
||||
_.Call.InterfaceId = ((TypeIdAttribute)typeof(ITestInterface).GetCustomAttributes(typeof(TypeIdAttribute), false)[0]).Id;
|
||||
_.Call.MethodId = 0;
|
||||
_.Call.Params.Content.Rewrap<TestInterface.Params_foo.WRITER>();
|
||||
});
|
||||
tester.ExpectAbort();
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void NoBootstrap()
|
||||
{
|
||||
var tester = new RpcEngineTester();
|
||||
|
||||
tester.Send(_ => { _.which = Message.WHICH.Bootstrap; _.Bootstrap.QuestionId = 0; });
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Return, _.which);
|
||||
Assert.AreEqual(Return.WHICH.Exception, _.Return.which);
|
||||
});
|
||||
Assert.IsFalse(tester.IsDismissed);
|
||||
tester.Engine.Main = new TestInterfaceImpl(new Counters());
|
||||
tester.Send(_ => { _.which = Message.WHICH.Bootstrap; _.Bootstrap.QuestionId = 1; });
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Return, _.which);
|
||||
Assert.AreEqual(Return.WHICH.Results, _.Return.which);
|
||||
});
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void DuplicateFinish()
|
||||
{
|
||||
var tester = new RpcEngineTester();
|
||||
tester.Engine.Main = new TestInterfaceImpl(new Counters());
|
||||
|
||||
uint bootCapId = 0;
|
||||
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Bootstrap; _.Bootstrap.QuestionId = 99; });
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Return, _.which);
|
||||
Assert.AreEqual(Return.WHICH.Results, _.Return.which);
|
||||
Assert.AreEqual(1, _.Return.Results.CapTable.Count);
|
||||
bootCapId = _.Return.Results.CapTable[0].SenderHosted;
|
||||
});
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Finish;
|
||||
_.Finish.QuestionId = 99;
|
||||
});
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Finish;
|
||||
_.Finish.QuestionId = 99;
|
||||
});
|
||||
tester.ExpectAbort();
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void DuplicateAnswer()
|
||||
{
|
||||
var tester = new RpcEngineTester();
|
||||
|
||||
var cap = tester.RealEnd.QueryMain();
|
||||
var proxy = new BareProxy(cap);
|
||||
Assert.IsFalse(proxy.WhenResolved.IsCompleted);
|
||||
uint id = 0;
|
||||
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Bootstrap, _.which);
|
||||
id = _.Bootstrap.QuestionId;
|
||||
});
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Return;
|
||||
_.Return.which = Return.WHICH.Results;
|
||||
_.Return.Results.CapTable.Init(1);
|
||||
_.Return.Results.CapTable[0].which = CapDescriptor.WHICH.SenderHosted;
|
||||
_.Return.Results.CapTable[0].SenderHosted = 1;
|
||||
});
|
||||
Assert.IsTrue(proxy.WhenResolved.IsCompleted);
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Finish, _.which);
|
||||
});
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Return;
|
||||
_.Return.which = Return.WHICH.Results;
|
||||
_.Return.Results.CapTable.Init(1);
|
||||
_.Return.Results.CapTable[0].which = CapDescriptor.WHICH.SenderHosted;
|
||||
_.Return.Results.CapTable[0].SenderHosted = 1;
|
||||
});
|
||||
tester.ExpectAbort();
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void InvalidReceiverHosted()
|
||||
{
|
||||
var tester = new RpcEngineTester();
|
||||
|
||||
var cap = tester.RealEnd.QueryMain();
|
||||
var proxy = new BareProxy(cap);
|
||||
Assert.IsFalse(proxy.WhenResolved.IsCompleted);
|
||||
uint id = 0;
|
||||
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Bootstrap, _.which);
|
||||
id = _.Bootstrap.QuestionId;
|
||||
});
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Return;
|
||||
_.Return.which = Return.WHICH.Results;
|
||||
_.Return.Results.CapTable.Init(1);
|
||||
_.Return.Results.CapTable[0].which = CapDescriptor.WHICH.ReceiverHosted;
|
||||
_.Return.Results.CapTable[0].ReceiverHosted = 0;
|
||||
});
|
||||
Assert.IsTrue(proxy.WhenResolved.IsCompleted);
|
||||
tester.ExpectAbort();
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void InvalidReceiverAnswer()
|
||||
{
|
||||
var tester = new RpcEngineTester();
|
||||
|
||||
var cap = tester.RealEnd.QueryMain();
|
||||
var proxy = new BareProxy(cap);
|
||||
Assert.IsFalse(proxy.WhenResolved.IsCompleted);
|
||||
uint id = 0;
|
||||
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Bootstrap, _.which);
|
||||
id = _.Bootstrap.QuestionId;
|
||||
});
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Return;
|
||||
_.Return.which = Return.WHICH.Results;
|
||||
_.Return.Results.CapTable.Init(1);
|
||||
_.Return.Results.CapTable[0].which = CapDescriptor.WHICH.ReceiverAnswer;
|
||||
_.Return.Results.CapTable[0].ReceiverAnswer.QuestionId = 0;
|
||||
_.Return.Results.CapTable[0].ReceiverAnswer.Transform.Init(1);
|
||||
_.Return.Results.CapTable[0].ReceiverAnswer.Transform[0].which = PromisedAnswer.Op.WHICH.GetPointerField;
|
||||
_.Return.Results.CapTable[0].ReceiverAnswer.Transform[0].GetPointerField = 0;
|
||||
});
|
||||
Assert.IsTrue(proxy.WhenResolved.IsCompleted);
|
||||
tester.ExpectAbort();
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void DuplicateResolve()
|
||||
{
|
||||
var tester = new RpcEngineTester();
|
||||
|
||||
var cap = tester.RealEnd.QueryMain();
|
||||
var proxy = new BareProxy(cap);
|
||||
Assert.IsFalse(proxy.WhenResolved.IsCompleted);
|
||||
uint id = 0;
|
||||
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Bootstrap, _.which);
|
||||
id = _.Bootstrap.QuestionId;
|
||||
});
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Return;
|
||||
_.Return.which = Return.WHICH.Results;
|
||||
_.Return.Results.CapTable.Init(1);
|
||||
_.Return.Results.CapTable[0].which = CapDescriptor.WHICH.SenderPromise;
|
||||
_.Return.Results.CapTable[0].SenderPromise = 0;
|
||||
});
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Resolve;
|
||||
_.Resolve.which = Resolve.WHICH.Cap;
|
||||
_.Resolve.Cap.which = CapDescriptor.WHICH.SenderHosted;
|
||||
_.Resolve.Cap.SenderHosted = 1;
|
||||
});
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Finish, _.which);
|
||||
});
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Resolve;
|
||||
_.Resolve.which = Resolve.WHICH.Exception;
|
||||
_.Resolve.Exception.Reason = "problem";
|
||||
});
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Release, _.which);
|
||||
});
|
||||
tester.ExpectAbort();
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void DuplicateRelease1()
|
||||
{
|
||||
var tester = new RpcEngineTester();
|
||||
tester.Engine.Main = new TestInterfaceImpl(new Counters());
|
||||
|
||||
uint bootCapId = 0;
|
||||
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Bootstrap; _.Bootstrap.QuestionId = 99;
|
||||
});
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Return, _.which);
|
||||
Assert.AreEqual(Return.WHICH.Results, _.Return.which);
|
||||
Assert.AreEqual(1, _.Return.Results.CapTable.Count);
|
||||
bootCapId = _.Return.Results.CapTable[0].SenderHosted;
|
||||
});
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Release;
|
||||
_.Release.Id = bootCapId;
|
||||
_.Release.ReferenceCount = 1;
|
||||
});
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Release;
|
||||
_.Release.Id = bootCapId;
|
||||
_.Release.ReferenceCount = 1;
|
||||
});
|
||||
tester.ExpectAbort();
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void DuplicateRelease2()
|
||||
{
|
||||
var tester = new RpcEngineTester();
|
||||
tester.Engine.Main = new TestInterfaceImpl(new Counters());
|
||||
|
||||
uint bootCapId = 0;
|
||||
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Bootstrap; _.Bootstrap.QuestionId = 99;
|
||||
});
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Return, _.which);
|
||||
Assert.AreEqual(Return.WHICH.Results, _.Return.which);
|
||||
Assert.AreEqual(1, _.Return.Results.CapTable.Count);
|
||||
bootCapId = _.Return.Results.CapTable[0].SenderHosted;
|
||||
});
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Release;
|
||||
_.Release.Id = bootCapId;
|
||||
_.Release.ReferenceCount = 2;
|
||||
});
|
||||
tester.ExpectAbort();
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void UnimplementedAccept()
|
||||
{
|
||||
var tester = new RpcEngineTester();
|
||||
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Accept;
|
||||
_.Accept.Embargo = true;
|
||||
_.Accept.QuestionId = 47;
|
||||
_.Accept.Provision.SetStruct(1, 0);
|
||||
_.Accept.Provision.Allocate();
|
||||
});
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Unimplemented, _.which);
|
||||
Assert.AreEqual(Message.WHICH.Accept, _.Unimplemented.which);
|
||||
Assert.IsTrue(_.Unimplemented.Accept.Embargo);
|
||||
Assert.AreEqual(47u, _.Unimplemented.Accept.QuestionId);
|
||||
Assert.AreEqual(1, _.Unimplemented.Accept.Provision.StructDataCount);
|
||||
Assert.AreEqual(0, _.Unimplemented.Accept.Provision.StructPtrCount);
|
||||
});
|
||||
Assert.IsFalse(tester.IsDismissed);
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void UnimplementedJoin()
|
||||
{
|
||||
var tester = new RpcEngineTester();
|
||||
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Join;
|
||||
_.Join.QuestionId = 74;
|
||||
});
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Unimplemented, _.which);
|
||||
Assert.AreEqual(Message.WHICH.Join, _.Unimplemented.which);
|
||||
Assert.AreEqual(74u, _.Unimplemented.Join.QuestionId);
|
||||
});
|
||||
Assert.IsFalse(tester.IsDismissed);
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void UnimplementedProvide()
|
||||
{
|
||||
var tester = new RpcEngineTester();
|
||||
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Provide;
|
||||
_.Provide.QuestionId = 666;
|
||||
});
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Unimplemented, _.which);
|
||||
Assert.AreEqual(Message.WHICH.Provide, _.Unimplemented.which);
|
||||
Assert.AreEqual(666u, _.Unimplemented.Provide.QuestionId);
|
||||
});
|
||||
Assert.IsFalse(tester.IsDismissed);
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void UnimplementedObsoleteDelete()
|
||||
{
|
||||
var tester = new RpcEngineTester();
|
||||
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.ObsoleteDelete;
|
||||
});
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Unimplemented, _.which);
|
||||
Assert.AreEqual(Message.WHICH.ObsoleteDelete, _.Unimplemented.which);
|
||||
});
|
||||
Assert.IsFalse(tester.IsDismissed);
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void UnimplementedObsoleteSave()
|
||||
{
|
||||
var tester = new RpcEngineTester();
|
||||
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.ObsoleteSave;
|
||||
});
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Unimplemented, _.which);
|
||||
Assert.AreEqual(Message.WHICH.ObsoleteSave, _.Unimplemented.which);
|
||||
});
|
||||
Assert.IsFalse(tester.IsDismissed);
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void UnimplementedUnknown()
|
||||
{
|
||||
var tester = new RpcEngineTester();
|
||||
|
||||
tester.Send(_ => {
|
||||
_.which = (Message.WHICH)123;
|
||||
});
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Unimplemented, _.which);
|
||||
Assert.AreEqual((Message.WHICH)123, _.Unimplemented.which);
|
||||
});
|
||||
Assert.IsFalse(tester.IsDismissed);
|
||||
}
|
||||
|
||||
class TestPipelineImpl3 : ITestPipeline
|
||||
{
|
||||
readonly TestPipelineImpl2 _impl;
|
||||
readonly ITestPipeline _proxy;
|
||||
|
||||
public TestPipelineImpl3(Task complete)
|
||||
{
|
||||
_impl = new TestPipelineImpl2(complete);
|
||||
var bproxy = BareProxy.FromImpl(_impl);
|
||||
_proxy = bproxy.Cast<ITestPipeline>(true);
|
||||
}
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
}
|
||||
|
||||
public bool IsGrandsonCapDisposed => _impl.IsChildCapDisposed;
|
||||
|
||||
public Task<(string, TestPipeline.AnyBox)> GetAnyCap(uint n, BareProxy inCap, CancellationToken cancellationToken_ = default)
|
||||
{
|
||||
throw new NotImplementedException();
|
||||
}
|
||||
|
||||
public Task<(string, TestPipeline.Box)> GetCap(uint n, ITestInterface inCap, CancellationToken cancellationToken_ = default)
|
||||
{
|
||||
return Task.FromResult(("foo", new TestPipeline.Box() { Cap = _proxy.GetCap(0, null).OutBox_Cap() }));
|
||||
}
|
||||
|
||||
public Task TestPointers(ITestInterface cap, object obj, IReadOnlyList<ITestInterface> list, CancellationToken cancellationToken_ = default)
|
||||
{
|
||||
throw new NotImplementedException();
|
||||
}
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void UnimplementedResolve()
|
||||
{
|
||||
var tcs = new TaskCompletionSource<int>();
|
||||
var tester = new RpcEngineTester();
|
||||
var impl = new TestPipelineImpl3(tcs.Task);
|
||||
tester.Engine.Main = impl;
|
||||
|
||||
uint bootCapId = 0;
|
||||
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Bootstrap; _.Bootstrap.QuestionId = 99;
|
||||
});
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Return, _.which);
|
||||
Assert.AreEqual(Return.WHICH.Results, _.Return.which);
|
||||
Assert.AreEqual(1, _.Return.Results.CapTable.Count);
|
||||
bootCapId = _.Return.Results.CapTable[0].SenderHosted;
|
||||
});
|
||||
tester.Send(_ => {
|
||||
_.which = Message.WHICH.Call;
|
||||
_.Call.QuestionId = 42;
|
||||
_.Call.Target.which = MessageTarget.WHICH.ImportedCap;
|
||||
_.Call.Target.ImportedCap = bootCapId;
|
||||
_.Call.InterfaceId = ((TypeIdAttribute)typeof(ITestPipeline).GetCustomAttributes(typeof(TypeIdAttribute), false)[0]).Id;
|
||||
_.Call.MethodId = 0;
|
||||
var wr = _.Call.Params.Content.Rewrap<TestPipeline.Params_getCap.WRITER>();
|
||||
wr.InCap = null;
|
||||
_.Call.Params.CapTable.Init(1);
|
||||
_.Call.Params.CapTable[0].which = CapDescriptor.WHICH.ReceiverHosted;
|
||||
});
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Return, _.which);
|
||||
Assert.AreEqual(Return.WHICH.Results, _.Return.which);
|
||||
Assert.AreEqual(1, _.Return.Results.CapTable.Count);
|
||||
Assert.AreEqual(CapDescriptor.WHICH.SenderPromise, _.Return.Results.CapTable[0].which);
|
||||
});
|
||||
tcs.SetResult(0);
|
||||
tester.Recv(_ => {
|
||||
Assert.AreEqual(Message.WHICH.Resolve, _.which);
|
||||
Assert.AreEqual(Resolve.WHICH.Cap, _.Resolve.which);
|
||||
Assert.AreEqual(CapDescriptor.WHICH.SenderHosted, _.Resolve.Cap.which);
|
||||
|
||||
Assert.IsFalse(impl.IsGrandsonCapDisposed);
|
||||
|
||||
tester.Send(_1 =>
|
||||
{
|
||||
_1.which = Message.WHICH.Unimplemented;
|
||||
_1.Unimplemented.which = Message.WHICH.Resolve;
|
||||
Reserializing.DeepCopy(_, _1.Unimplemented.Resolve);
|
||||
});
|
||||
|
||||
Assert.IsFalse(impl.IsGrandsonCapDisposed);
|
||||
|
||||
tester.Send(_1 =>
|
||||
{
|
||||
_1.which = Message.WHICH.Finish;
|
||||
_1.Finish.QuestionId = 42;
|
||||
_1.Finish.ReleaseResultCaps = true;
|
||||
});
|
||||
|
||||
Assert.IsTrue(impl.IsGrandsonCapDisposed);
|
||||
});
|
||||
Assert.IsFalse(tester.IsDismissed);
|
||||
}
|
||||
}
|
||||
}
|
@ -432,6 +432,33 @@ namespace Capnp.Net.Runtime.Tests.GenImpls
|
||||
}
|
||||
}
|
||||
|
||||
class TestInterfaceImpl2 : ITestInterface
|
||||
{
|
||||
public Task Bar(CancellationToken cancellationToken_ = default)
|
||||
{
|
||||
throw new NotImplementedException();
|
||||
}
|
||||
|
||||
public Task Baz(TestAllTypes s, CancellationToken cancellationToken_ = default)
|
||||
{
|
||||
throw new NotImplementedException();
|
||||
}
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
IsDisposed = true;
|
||||
}
|
||||
|
||||
public bool IsDisposed { get; private set; }
|
||||
|
||||
public Task<string> Foo(uint i, bool j, CancellationToken cancellationToken_ = default)
|
||||
{
|
||||
Assert.AreEqual(123u, i);
|
||||
Assert.IsTrue(j);
|
||||
return Task.FromResult("bar");
|
||||
}
|
||||
}
|
||||
|
||||
#endregion TestInterface
|
||||
|
||||
#region TestExtends
|
||||
@ -506,6 +533,41 @@ namespace Capnp.Net.Runtime.Tests.GenImpls
|
||||
throw new NotImplementedException();
|
||||
}
|
||||
}
|
||||
|
||||
class TestPipelineImpl2 : ITestPipeline
|
||||
{
|
||||
readonly Task _deblock;
|
||||
readonly TestInterfaceImpl2 _timpl2;
|
||||
|
||||
public TestPipelineImpl2(Task deblock)
|
||||
{
|
||||
_deblock = deblock;
|
||||
_timpl2 = new TestInterfaceImpl2();
|
||||
}
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
}
|
||||
|
||||
public bool IsChildCapDisposed => _timpl2.IsDisposed;
|
||||
|
||||
public Task<(string, TestPipeline.AnyBox)> GetAnyCap(uint n, BareProxy inCap, CancellationToken cancellationToken_ = default)
|
||||
{
|
||||
throw new NotImplementedException();
|
||||
}
|
||||
|
||||
public async Task<(string, TestPipeline.Box)> GetCap(uint n, ITestInterface inCap, CancellationToken cancellationToken_ = default)
|
||||
{
|
||||
await _deblock;
|
||||
return ("hello", new TestPipeline.Box() { Cap = _timpl2 });
|
||||
}
|
||||
|
||||
public Task TestPointers(ITestInterface cap, object obj, IReadOnlyList<ITestInterface> list, CancellationToken cancellationToken_ = default)
|
||||
{
|
||||
throw new NotImplementedException();
|
||||
}
|
||||
}
|
||||
|
||||
#endregion TestPipeline
|
||||
|
||||
#region TestCallOrder
|
||||
|
@ -60,8 +60,14 @@ namespace Capnp.Rpc
|
||||
}
|
||||
}
|
||||
|
||||
internal class RpcEndpoint : IEndpoint, IRpcEndpoint
|
||||
public class RpcEndpoint : IEndpoint, IRpcEndpoint
|
||||
{
|
||||
public enum EndpointState
|
||||
{
|
||||
Active,
|
||||
Dismissed
|
||||
}
|
||||
|
||||
static readonly ThreadLocal<Action?> _exportCapTablePostActions = new ThreadLocal<Action?>();
|
||||
static readonly ThreadLocal<PendingQuestion?> _tailCall = new ThreadLocal<PendingQuestion?>();
|
||||
static readonly ThreadLocal<bool> _canDeferCalls = new ThreadLocal<bool>();
|
||||
@ -87,8 +93,11 @@ namespace Capnp.Rpc
|
||||
{
|
||||
_host = host;
|
||||
_tx = tx;
|
||||
State = EndpointState.Active;
|
||||
}
|
||||
|
||||
public EndpointState State { get; private set; }
|
||||
|
||||
public void Dismiss()
|
||||
{
|
||||
lock (_reentrancyBlocker)
|
||||
@ -105,6 +114,8 @@ namespace Capnp.Rpc
|
||||
|
||||
_answerTable.Clear();
|
||||
_pendingDisembargos.Clear();
|
||||
|
||||
State = EndpointState.Dismissed;
|
||||
}
|
||||
|
||||
_tx.Dismiss();
|
||||
@ -112,6 +123,9 @@ namespace Capnp.Rpc
|
||||
|
||||
public void Forward(WireFrame frame)
|
||||
{
|
||||
if (State == EndpointState.Dismissed)
|
||||
throw new InvalidOperationException("Endpoint is in dismissed state and doesn't accept frames anymore");
|
||||
|
||||
Interlocked.Increment(ref _recvCount);
|
||||
ProcessFrame(frame);
|
||||
}
|
||||
@ -327,6 +341,7 @@ namespace Capnp.Rpc
|
||||
if (!added)
|
||||
{
|
||||
Logger.LogWarning("Incoming bootstrap request: Peer specified duplicate (not yet released?) answer ID.");
|
||||
throw new RpcProtocolErrorException("Duplicate question ID");
|
||||
}
|
||||
|
||||
|
||||
@ -390,9 +405,7 @@ namespace Capnp.Rpc
|
||||
pendingAnswer.Cancel();
|
||||
pendingAnswer.Dispose();
|
||||
|
||||
SendAbort($"There is another pending answer for the same question ID {req.QuestionId}.");
|
||||
|
||||
return;
|
||||
throw new RpcProtocolErrorException($"There is another pending answer for the same question ID {req.QuestionId}.");
|
||||
}
|
||||
|
||||
switch (req.SendResultsTo.which)
|
||||
@ -553,8 +566,7 @@ namespace Capnp.Rpc
|
||||
{
|
||||
Logger.LogWarning("Incoming RPC call: Peer asked for invalid (already released?) capability ID.");
|
||||
|
||||
SendAbort($"Requested capability with ID {req.Target.ImportedCap} does not exist.");
|
||||
return;
|
||||
throw new RpcProtocolErrorException($"Requested capability with ID {req.Target.ImportedCap} does not exist.");
|
||||
}
|
||||
}
|
||||
|
||||
@ -606,8 +618,7 @@ namespace Capnp.Rpc
|
||||
else
|
||||
{
|
||||
Logger.LogWarning("Incoming RPC call: Peer asked for non-existing answer ID.");
|
||||
SendAbort($"Did not find a promised answer for given ID {req.Target.PromisedAnswer.QuestionId}");
|
||||
return;
|
||||
throw new RpcProtocolErrorException($"Did not find a promised answer for given ID {req.Target.PromisedAnswer.QuestionId}");
|
||||
}
|
||||
}
|
||||
break;
|
||||
@ -637,7 +648,7 @@ namespace Capnp.Rpc
|
||||
{
|
||||
Logger.LogWarning("Incoming RPC return: Unknown answer ID.");
|
||||
|
||||
return;
|
||||
throw new RpcProtocolErrorException("Unknown answer ID");
|
||||
}
|
||||
}
|
||||
|
||||
@ -722,7 +733,7 @@ namespace Capnp.Rpc
|
||||
if (!_importTable.TryGetValue(resolve.PromiseId, out var rcw))
|
||||
{
|
||||
Logger.LogWarning("Received a resolve message with invalid ID");
|
||||
return;
|
||||
throw new RpcProtocolErrorException($"Invalid ID {resolve.PromiseId}");
|
||||
}
|
||||
|
||||
if (!rcw.Cap.TryGetTarget(out var cap))
|
||||
@ -734,17 +745,17 @@ namespace Capnp.Rpc
|
||||
if (!(cap is PromisedCapability resolvableCap))
|
||||
{
|
||||
Logger.LogWarning("Received a resolve message for a capability which is not a promise");
|
||||
return;
|
||||
throw new RpcProtocolErrorException($"Not a promise {resolve.PromiseId}");
|
||||
}
|
||||
|
||||
try
|
||||
{
|
||||
switch (resolve.which)
|
||||
{
|
||||
case Resolve.WHICH.Cap:
|
||||
lock (_reentrancyBlocker)
|
||||
{
|
||||
var resolvedCap = ImportCap(resolve.Cap);
|
||||
if (resolvedCap == null)
|
||||
resolvedCap = LazyCapability.CreateBrokenCap("Failed to resolve this capability");
|
||||
resolvableCap.ResolveTo(resolvedCap);
|
||||
}
|
||||
break;
|
||||
@ -758,6 +769,11 @@ namespace Capnp.Rpc
|
||||
throw new RpcUnimplementedException();
|
||||
}
|
||||
}
|
||||
catch (InvalidOperationException)
|
||||
{
|
||||
throw new RpcProtocolErrorException($"Capability {resolve.PromiseId} was already resolved");
|
||||
}
|
||||
}
|
||||
|
||||
void ProcessSenderLoopback(Disembargo.READER disembargo)
|
||||
{
|
||||
@ -777,10 +793,7 @@ namespace Capnp.Rpc
|
||||
{
|
||||
Logger.LogWarning("Sender loopback request: Peer asked for invalid (already released?) capability ID.");
|
||||
|
||||
SendAbort("'Disembargo': Invalid capability ID");
|
||||
Dismiss();
|
||||
|
||||
return;
|
||||
throw new RpcProtocolErrorException("'Disembargo': Invalid capability ID");
|
||||
}
|
||||
|
||||
reply.Target.which = MessageTarget.WHICH.ImportedCap;
|
||||
@ -830,8 +843,7 @@ namespace Capnp.Rpc
|
||||
{
|
||||
Logger.LogWarning("Sender loopback request: Peer asked for disembargoing an answer which does not resolve back to the sender.");
|
||||
|
||||
SendAbort("'Disembargo': Answer does not resolve back to me");
|
||||
Dismiss();
|
||||
throw new RpcProtocolErrorException("'Disembargo': Answer does not resolve back to me");
|
||||
}
|
||||
}
|
||||
finally
|
||||
@ -844,9 +856,7 @@ namespace Capnp.Rpc
|
||||
{
|
||||
Logger.LogWarning($"Sender loopback request: Peer asked for disembargoing an answer which either has not yet returned, was canceled, or faulted: {exception.Message}");
|
||||
|
||||
SendAbort($"'Disembargo' failure: {exception}");
|
||||
Dismiss();
|
||||
|
||||
throw new RpcProtocolErrorException($"'Disembargo' failure: {exception}");
|
||||
}
|
||||
});
|
||||
}
|
||||
@ -854,10 +864,7 @@ namespace Capnp.Rpc
|
||||
{
|
||||
Logger.LogWarning("Sender loopback request: Peer asked for non-existing answer ID.");
|
||||
|
||||
SendAbort("'Disembargo': Invalid answer ID");
|
||||
Dismiss();
|
||||
|
||||
return;
|
||||
throw new RpcProtocolErrorException("'Disembargo': Invalid answer ID");
|
||||
}
|
||||
|
||||
break;
|
||||
@ -966,6 +973,8 @@ namespace Capnp.Rpc
|
||||
else
|
||||
{
|
||||
Logger.LogWarning("Peer sent 'finish' message with unknown question ID");
|
||||
|
||||
throw new RpcProtocolErrorException("unknown question ID");
|
||||
}
|
||||
}
|
||||
|
||||
@ -994,6 +1003,8 @@ namespace Capnp.Rpc
|
||||
catch (System.Exception exception)
|
||||
{
|
||||
Logger.LogWarning($"Attempting to release capability with invalid reference count: {exception.Message}");
|
||||
|
||||
throw new RpcProtocolErrorException("Invalid reference count");
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -1001,6 +1012,8 @@ namespace Capnp.Rpc
|
||||
if (!exists)
|
||||
{
|
||||
Logger.LogWarning("Attempting to release unknown capability ID");
|
||||
|
||||
throw new RpcProtocolErrorException("Invalid export ID");
|
||||
}
|
||||
}
|
||||
|
||||
@ -1056,9 +1069,7 @@ namespace Capnp.Rpc
|
||||
//# In cases where there is no sensible way to react to an `unimplemented` message (without
|
||||
//# resource leaks or other serious problems), the connection may need to be aborted. This is
|
||||
//# a gray area; different implementations may take different approaches.
|
||||
SendAbort("It's hopeless if you don't implement the bootstrap message");
|
||||
Dismiss();
|
||||
break;
|
||||
throw new RpcProtocolErrorException("It's hopeless if you don't implement the bootstrap message");
|
||||
|
||||
default:
|
||||
// Looking at the various message types it feels OK to just not care about other 'unimplemented'
|
||||
@ -1156,6 +1167,11 @@ namespace Capnp.Rpc
|
||||
|
||||
Tx(mb.Frame);
|
||||
}
|
||||
catch (RpcProtocolErrorException error)
|
||||
{
|
||||
SendAbort(error.Message);
|
||||
Dismiss();
|
||||
}
|
||||
catch (System.Exception exception)
|
||||
{
|
||||
Logger.LogError(exception, "Uncaught exception during message processing.");
|
||||
@ -1164,12 +1180,11 @@ namespace Capnp.Rpc
|
||||
// First, we send implementation specific details of a - maybe internal - error, not very valuable for the
|
||||
// receiver. But worse: From a security point of view, we might even reveil a secret here.
|
||||
SendAbort("Uncaught exception during RPC processing. This may also happen due to invalid input.");
|
||||
|
||||
Dismiss();
|
||||
}
|
||||
}
|
||||
|
||||
ConsumedCapability? ImportCap(CapDescriptor.READER capDesc)
|
||||
ConsumedCapability ImportCap(CapDescriptor.READER capDesc)
|
||||
{
|
||||
lock (_reentrancyBlocker)
|
||||
{
|
||||
@ -1198,7 +1213,6 @@ namespace Capnp.Rpc
|
||||
rcw = new RefCounted<WeakReference<RemoteCapability>>(
|
||||
new WeakReference<RemoteCapability>(newCap));
|
||||
_importTable.Add(capDesc.SenderHosted, rcw);
|
||||
Debug.Assert(newCap != null);
|
||||
return newCap;
|
||||
}
|
||||
|
||||
@ -1217,7 +1231,6 @@ namespace Capnp.Rpc
|
||||
rcwp.Cap.SetTarget(impCap);
|
||||
}
|
||||
|
||||
Debug.Assert(impCap != null);
|
||||
return impCap;
|
||||
}
|
||||
else
|
||||
@ -1226,7 +1239,6 @@ namespace Capnp.Rpc
|
||||
rcw = new RefCounted<WeakReference<RemoteCapability>>(
|
||||
new WeakReference<RemoteCapability>(newCap));
|
||||
_importTable.Add(capDesc.SenderPromise, rcw);
|
||||
Debug.Assert(newCap != null);
|
||||
return newCap;
|
||||
}
|
||||
|
||||
@ -1238,7 +1250,7 @@ namespace Capnp.Rpc
|
||||
else
|
||||
{
|
||||
Logger.LogWarning("Peer refers to receiver-hosted capability which does not exist.");
|
||||
return null;
|
||||
throw new RpcProtocolErrorException($"Receiver-hosted capability {capDesc.ReceiverHosted} does not exist.");
|
||||
}
|
||||
|
||||
case CapDescriptor.WHICH.ReceiverAnswer:
|
||||
@ -1270,7 +1282,7 @@ namespace Capnp.Rpc
|
||||
else
|
||||
{
|
||||
Logger.LogWarning("Peer refers to pending answer which does not exist.");
|
||||
return null;
|
||||
throw new RpcProtocolErrorException($"Invalid question ID {capDesc.ReceiverAnswer.QuestionId}");
|
||||
}
|
||||
|
||||
case CapDescriptor.WHICH.ThirdPartyHosted:
|
||||
@ -1288,7 +1300,6 @@ namespace Capnp.Rpc
|
||||
rcv.Cap.SetTarget(impCap);
|
||||
}
|
||||
|
||||
Debug.Assert(impCap != null);
|
||||
return impCap;
|
||||
}
|
||||
else
|
||||
@ -1296,7 +1307,6 @@ namespace Capnp.Rpc
|
||||
var newCap = new ImportedCapability(this, capDesc.ThirdPartyHosted.VineId);
|
||||
rcv = new RefCounted<WeakReference<RemoteCapability>>(
|
||||
new WeakReference<RemoteCapability>(newCap));
|
||||
Debug.Assert(newCap != null);
|
||||
return newCap;
|
||||
}
|
||||
|
||||
@ -1307,7 +1317,7 @@ namespace Capnp.Rpc
|
||||
}
|
||||
}
|
||||
|
||||
public IList<ConsumedCapability?> ImportCapTable(Payload.READER payload)
|
||||
internal IList<ConsumedCapability?> ImportCapTable(Payload.READER payload)
|
||||
{
|
||||
var list = new List<ConsumedCapability?>();
|
||||
|
||||
@ -1481,7 +1491,7 @@ namespace Capnp.Rpc
|
||||
|
||||
readonly ConcurrentBag<RpcEndpoint> _inboundEndpoints = new ConcurrentBag<RpcEndpoint>();
|
||||
|
||||
internal RpcEndpoint AddEndpoint(IEndpoint outboundEndpoint)
|
||||
public RpcEndpoint AddEndpoint(IEndpoint outboundEndpoint)
|
||||
{
|
||||
var inboundEndpoint = new RpcEndpoint(this, outboundEndpoint);
|
||||
_inboundEndpoints.Add(inboundEndpoint);
|
||||
@ -1503,5 +1513,14 @@ namespace Capnp.Rpc
|
||||
_bootstrapCap = value;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Sets the bootstrap capability. It must be an object which implements a valid capability interface
|
||||
/// (<see cref="SkeletonAttribute"/>).
|
||||
/// </summary>
|
||||
public object Main
|
||||
{
|
||||
set { BootstrapCap = Skeleton.GetOrCreateSkeleton(value, false); }
|
||||
}
|
||||
}
|
||||
}
|
9
Capnp.Net.Runtime/Rpc/RpcProtocolErrorException.cs
Normal file
9
Capnp.Net.Runtime/Rpc/RpcProtocolErrorException.cs
Normal file
@ -0,0 +1,9 @@
|
||||
namespace Capnp.Rpc
|
||||
{
|
||||
class RpcProtocolErrorException : System.Exception
|
||||
{
|
||||
public RpcProtocolErrorException(string reason): base(reason)
|
||||
{
|
||||
}
|
||||
}
|
||||
}
|
@ -325,7 +325,7 @@ namespace Capnp.Rpc
|
||||
/// </summary>
|
||||
public object Main
|
||||
{
|
||||
set { _rpcEngine.BootstrapCap = Skeleton.GetOrCreateSkeleton(value, false); }
|
||||
set { _rpcEngine.Main = value; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
|
Loading…
x
Reference in New Issue
Block a user