如何在不成为本地管理员的情况下使用IPC?

时间:2010-06-18 17:50:08

标签: c# .net-3.5 ipc servicehost

我目前在.Net中维护一个内部应用程序。该应用程序使用IPC一次维护一个正在运行的会话;如果另一个会话尝试打开(某人再次点击该图标,一些会打开一个保存的结果文件),第二个会话将此信息传达给第一个会话,然后终止,第一个会话然后启动请求的操作。

我目前使用System.ServiceModel命名空间执行此操作,如下所示:

命名空间EventLogViewer.IPCServer {     //提供流程的通用框架,以便与另一个进行通信。

[ServiceContract(Namespace = "http://ELV.domain")]
interface IELVLauncher
{
    [OperationContract]
    bool LaunchTabs(String[] fileNames);
}
public delegate void TabLauncher(String[] fileNames);
class ELVLauncherServer : IDisposable
{
    ServiceHost ipcService;
    public ELVLauncherServer()
    {
        Uri baseAddress = new Uri("Http://localhost:45600/elvlauncher/service");
        String address = "net.pipe://localhost/elvlauncher/tablauncher";

        ipcService = new ServiceHost(typeof(ELVLauncher), baseAddress);

        NetNamedPipeBinding binding = new NetNamedPipeBinding(NetNamedPipeSecurityMode.None);
        ipcService.AddServiceEndpoint(typeof(IELVLauncher), binding, address);

        ServiceMetadataBehavior behavior = new ServiceMetadataBehavior();
        behavior.HttpGetEnabled = true;
        behavior.HttpGetUrl = new Uri("http://localhost:45601/elvlauncher");
        ipcService.Description.Behaviors.Add(behavior);
        Utilities.WriteMemoryDebugStatement(new DebugStatement(DebugStatement.StatementType.INFO, "Registering IPC Service"));
        ipcService.Open();

    }

    #region IDisposable Members

    public void Dispose()
    {
        ipcService.Close();
    }

    #endregion
}
public class ELVLauncher : IELVLauncher
{
    #region IELVLauncher Members

    public bool LaunchTabs(string[] fileNames)
    {
        try
        {
            Utilities.WriteMemoryDebugStatement(new DebugStatement(DebugStatement.StatementType.INFO, String.Format("IPC request received, files: {0}", String.Join(", ", fileNames))));
            Program.mainWindow.backgroundListener_OnLaunchRequestReceived(fileNames);
        }
        catch (Exception exception)
        {
            Utilities.WriteMemoryDebugStatement(new DebugStatement(exception));
        }
        return (true);
    }

    #endregion

}

此代码最初是在Windows XP下开发的,就像我们没有迁移到Vista的大多数公司一样。由于我们都以本地管理员身份运行,因此这段代码运行没有问题。

但是,我们现在正在转换到Win 7,并且此代码在启动时会产生异常,因为它无法注册IPC端点。更多信息请访问:http://mark.michaelis.net/Blog/WindowsCommunicationFoundationWithWindowsVistaAndUAC.aspx

给出的解决方法是将应用程序设置为需要管理员权限。这不起作用有两个原因:

  1. 此应用不需要管理员权限
  2. 我们使用ClickOnce部署内部工具,ClickOnce除了作为流程启动器运行外,不支持任何其他工具。
  3. 所以在这一点上,我需要找到一个不需要管理员权限的IPC解决方案,让我实现最初的目标:检测已经运行的代码实例并告诉它该做什么,否则启动自己。任何关于该做什么的建议(在.Net框架内,请不要提供第三方解决方案)将不胜感激。

2 个答案:

答案 0 :(得分:3)

确保您的应用的单个会话非常困难(并且您已经发现,有问题的方式)。您应该将该功能重构/重做为use a mutex

这是另一个example. 必要的wiki article about mutexes

修改

要绕过通信位,你可以生成一个带有你自己的特殊前缀的临时文件,或者在OS提供的临时文件夹中扩展,单个应用程序将轮询(过滤你的特殊前缀和/或扩展名)以查看是否有更多请求已经提出。参见:

System.IO.Path.GetTempFileName 

System.IO.Path.GetTempPath

答案 1 :(得分:2)

明确使用named mutex将应用程序限制为单个实例,然后您可以使用NamedPipeServerStreamNamedPipeClientStream来执行不需要管理员权限的IPC。