古い.NETRemotingIPCコードの一部をWCF名前付きパイプに移植しようとしていますが、おそらくここで愚かなことをしていると思います...
ServiceHost/NetNamedPipeBindingエンドポイントを使用するWindowsサービスがあります。
ChannelFactory/NetNamedPipeBindingを使用してWinFormsUIに接続しています。
Windows Serverが実行されているかどうかに関係なくCreateChannel
、クライアントで実行した後、ChannelFactoryオブジェクトの状態は開かれています。サービスが実行されていない場合は、「オープン」以外のものになると思います。例外を受け取るのは、リモートメソッドを呼び出そうとするまではありません。接続が正常に確立されたかどうかを検出するための最良の方法は何ですか?
サービスが実際に実行されている場合、コードは正常に機能しますが、実行されていない場合(または接続が切断された場合)、クライアントの状態をOpenedから変更できないようです。
これが私のサービスヘルパーコードです:
public class WcfServerHelper
{
public static ServiceHost RemoteServiceHost { get; set; }
public static void Start(Type remotingType)
{
RemoteServiceHost = new ServiceHost(remotingType, new Uri[] {
new Uri("net.pipe://localhost")
});
RemoteServiceHost.AddServiceEndpoint(typeof(ICardPrinterRemoting), new NetNamedPipeBinding(), "LifeMedPrinter");
RemoteServiceHost.Open();
}
public static void Stop()
{
if (RemoteServiceHost != null && RemoteServiceHost.State == CommunicationState.Opened)
{
RemoteServiceHost.Close();
}
}
}
Windowsサービス:
public partial class Service1 : ServiceBase
{
public Service1()
{
InitializeComponent();
}
protected override void OnStart(string[] args)
{
WcfServerHelper.Start(typeof(CardPrinterRemoting));
}
protected override void OnStop()
{
WcfServerHelper.Stop();
}
}
クライアントヘルパーコード:
public class WcfClientHelper
{
public static ICardPrinterRemoting RemotingObject { get; set; }
public static ChannelFactory<ICardPrinterRemoting> PipeFactory { get; set; }
public static void ConnectIpc()
{
PipeFactory = new ChannelFactory<ICardPrinterRemoting>(new NetNamedPipeBinding(), new EndpointAddress("net.pipe://localhost/LifeMedPrinter"));
RemotingObject = PipeFactory.CreateChannel();
}
}
Windows UI(サービスクライアント)コード:
private void btnConnect_Click(object sender, EventArgs e)
{
WcfClientHelper.ConnectIpc();
// The event handlers never get called, even if a successfully connection is made and then the Windows Service is stopped
WcfClientHelper.PipeFactory.Closing += PipeFactory_Closing;
WcfClientHelper.PipeFactory.Faulted += PipeFactory_Faulted;
WcfClientHelper.PipeFactory.Closed += PipeFactory_Closed;
if (WcfClientHelper.PipeFactory.State == CommunicationState.Opened)
{
LifeMedPrinterSettings lifeMedPrinterSettings = new LifeMedPrinterSettings();
lifeMedPrinterSettings.PrinterName = "Hello World";
// It always falls into this IF statement, but will blow up on this method call if the service isn't actually running:
WcfClientHelper.RemotingObject.SaveSettings(lifeMedPrinterSettings);
}
else
{
MessageBox.Show("No can-do");
}
}
そして、サービスが実行されていないときにリモートメソッドを呼び出したときに受け取る例外は次のとおりです。
System.ServiceModel.EndpointNotFoundException was unhandled
HResult=-2146233087
Message=There was no endpoint listening at net.pipe://localhost/LifeMedPrinter that could accept the message. This is often caused by an incorrect address or SOAP action. See InnerException, if present, for more details.
Source=mscorlib
StackTrace:
Server stack trace:
at System.ServiceModel.Channels.PipeConnectionInitiator.GetPipeName(Uri uri, IPipeTransportFactorySettings transportFactorySettings)
at System.ServiceModel.Channels.NamedPipeConnectionPoolRegistry.NamedPipeConnectionPool.GetPoolKey(EndpointAddress address, Uri via)
at System.ServiceModel.Channels.ConnectionPoolHelper.TakeConnection(TimeSpan timeout)
at System.ServiceModel.Channels.ConnectionPoolHelper.EstablishConnection(TimeSpan timeout)
at System.ServiceModel.Channels.ClientFramingDuplexSessionChannel.OnOpen(TimeSpan timeout)
at System.ServiceModel.Channels.CommunicationObject.Open(TimeSpan timeout)
at System.ServiceModel.Channels.ServiceChannel.OnOpen(TimeSpan timeout)
at System.ServiceModel.Channels.CommunicationObject.Open(TimeSpan timeout)
at System.ServiceModel.Channels.ServiceChannel.CallOnceManager.CallOnce(TimeSpan timeout, CallOnceManager cascade)
at System.ServiceModel.Channels.ServiceChannel.EnsureOpened(TimeSpan timeout)
at System.ServiceModel.Channels.ServiceChannel.Call(String action, Boolean oneway, ProxyOperationRuntime operation, Object[] ins, Object[] outs, TimeSpan timeout)
at System.ServiceModel.Channels.ServiceChannelProxy.InvokeService(IMethodCallMessage methodCall, ProxyOperationRuntime operation)
at System.ServiceModel.Channels.ServiceChannelProxy.Invoke(IMessage message)
Exception rethrown at [0]:
at System.Runtime.Remoting.Proxies.RealProxy.HandleReturnMessage(IMessage reqMsg, IMessage retMsg)
at System.Runtime.Remoting.Proxies.RealProxy.PrivateInvoke(MessageData& msgData, Int32 type)
at LifeMed.CardPrinter.Remoting.Interfaces.ICardPrinterRemoting.SaveSettings(LifeMedPrinterSettings printerSettings)
InnerException: System.IO.PipeException
HResult=-2146232800
Message=The pipe endpoint 'net.pipe://localhost/LifeMedPrinter' could not be found on your local machine.
ErrorCode=-2146232800
InnerException:
ありがとうございました!