如何在Visual Studio中“开始调试”并运行到调试器时运行自定义代码

时间:2018-04-04 20:43:41

标签: visual-studio debugging ide remote-debugging

VS Code似乎有一个名为Launch.json的文件来自定义运行应用程序进行调试时会发生什么。 我们如何为Visual Studio进行此操作? 我的目标是在没有VS Docker Tools的情况下运行docker-compose,然后在容器启动后告诉VS附加到容器。

我的问题并非特定于这种情况,它也可能是另一种使用调试器方案运行的自定义方式。 理想情况下,我可以运行一些命令来构建项目,运行项目,然后告诉VS附加到它并将该逻辑与VS关联。

3 个答案:

答案 0 :(得分:2)

右键单击解决方案资源管理器中的项目;选择“属性”。 选择“调试”节点。填写“Command”,“Command Arguments”(如果需要),并将“Attach”设置为true。

答案 1 :(得分:2)

根据Vlad和Jack的答案,我提出了以下解决方案。 为了在按下“运行”按钮时运行自己的代码,我设置了一个带有自定义启动设置的空白命令行项目。

{
  "profiles": {
    "Build": {
      "commandName": "Executable",
      "executablePath": "powershell",
      "commandLineArgs": ".\\DebugRun.ps1",
      "workingDirectory": "."
    }
  }
}

每当我按Run,它将使用PowerShell运行DebugRun.ps1脚本。 这是我放在DebugRun.ps1中的内容。

docker-compose -f "./docker-compose.debug.yml" --no-ansi up -d --force-recreate --build

Start-Sleep -Seconds 5

$appId = ((docker ps --filter "ancestor=employeemapapp:debug")[1] -split " ")[0]
$apiId = ((docker ps --filter "ancestor=employeemapapi:debug")[1] -split " ")[0]

$appIp = (docker inspect --format '{{range .NetworkSettings.Networks}}{{.IPAddress}}{{end}}' $appId)
$apiIp = (docker inspect --format '{{range .NetworkSettings.Networks}}{{.IPAddress}}{{end}}' $apiId)

docker exec -d $appId C:\\remote_debugger\\x64\\msvsmon.exe /noauth /anyuser /silent /nostatus /noclrwarn /nosecuritywarn /nofirewallwarn /nowowwarn /timeout:214748364
docker exec -d $apiId C:\\remote_debugger\\x64\\msvsmon.exe /noauth /anyuser /silent /nostatus /noclrwarn /nosecuritywarn /nofirewallwarn /nowowwarn /timeout:214748364

$appTarget = $appId + ":4022"
$apiTarget = $apiId + ":4022"

#Parameters
## 1: Solution Name is used to run the code only to the VS instance that has the Solution open
## 2: Transportation method for remote debugger
## 3: Target is the hostname/IP/... to the target where remote debugging is running
## 4: The process name you want to attach tos
./RemoteDebugAttach.exe "EmployeeMap.sln" "Remote (no authentication)" $appTarget "dotnet.exe"
./RemoteDebugAttach.exe "EmployeeMap.sln" "Remote (no authentication)" $apiTarget "dotnet.exe"

Write-Host "Api:" $apiIp
Write-Host "App:" $appIp
$apiUrl = "http://$apiIp/api/employees"
$appUrl = "http://$appIp"

start $apiUrl
start $appUrl

Read-Host "Press any key to quit"

./RemoteDebugDetach.exe EmployeeMap.sln "dotnet.exe"

docker exec -d $appId C:\\remote_debugger\\x64\\utils\\KillProcess.exe msvsmon.exe
docker exec -d $apiId C:\\remote_debugger\\x64\\utils\\KillProcess.exe msvsmon.exe

docker-compose -f "./docker-compose.debug.yml" --no-ansi down 

此脚本执行以下操作:

  • 通过docker-compose
  • 打开我的docker容器
  • 抓住容器ID,ip等等
  • 启动容器内的远程调试器实例
  • 使用自定义可执行文件(RemoteDebugAttach)附加到容器内的dotnet.exe进程
  • 打开浏览器以浏览从容器提供的网站
  • 等待任何按键降低基础设施
  • 使用自定义可执行文件从进程中分离
  • 终止Docker容器上的远程调试器
  • 放下容器

自定义可执行文件来自一个单独的解决方案,我刚刚构建了一些命令行应用程序。 我使用了此source中的代码来获取计算机上运行的所有VS实例。 并将其与此代码结合起来以附加:

class Program
{
    [STAThread]
    static void Main(string[] args)
    {
        string solutionName = Ask(args, 0, "Solution name?");
        string transportName = Ask(args, 1, "Transport name?");
        string target = Ask(args, 2, "Target machine?");
        string processName = Ask(args, 3, "Process Name?");


        var instances = Msdev.GetIDEInstances(true);
        var dte = (DTE2)instances.Find(d => d.Solution.FullName.EndsWith(solutionName, StringComparison.InvariantCultureIgnoreCase));
        var debugger = dte.Debugger as Debugger2;
        var transports = debugger.Transports;
        Transport transport = null;
        foreach(Transport loopTransport in transports)
        {
            if(loopTransport.Name.Equals(transportName, StringComparison.InvariantCultureIgnoreCase)) // "Remote (no authentication)")
            {
                transport = loopTransport;
                break;
            }
        }

        Processes processes = debugger.GetProcesses(transport, target); // "172.24.50.15:4022");
        foreach(Process process in processes)
        {
            if(process.Name.EndsWith(processName, StringComparison.InvariantCultureIgnoreCase))
            {
                process.Attach();
            }
        }
    }

    static string Ask(string[] args, int index, string question)
    {
        if(args.Length <= index)
        {
            Console.WriteLine(question);
            return Console.ReadLine();
        }

        return args[index];
    }
}

以下分离:

class Program
{
    [STAThread]
    static void Main(string[] args)
    {
        string solutionName = Ask(args, 0, "Solution name");
        string processName = Ask(args, 1, "Process Name?");

        var instances = Msdev.GetIDEInstances(true);

        var dte = (DTE2)instances.Find(d => d.Solution.FullName.EndsWith(solutionName, StringComparison.InvariantCultureIgnoreCase));
        var debugger = dte.Debugger as Debugger2;

        Processes processes = debugger.DebuggedProcesses;
        foreach (Process2 process in processes)
        {
            if (process.Name.EndsWith(processName, StringComparison.InvariantCultureIgnoreCase))
            {
                process.Detach(false);
            }
        }
    }

    static string Ask(string[] args, int index, string question)
    {
        if (args.Length <= index)
        {
            Console.WriteLine(question);
            return Console.ReadLine();
        }

        return args[index];
    }
}

我宁愿在PowerShell中使用此代码,因为复制到其他项目+编辑会更容易。虽然在PowerShell中,我根本无法获得正确的代码来执行,即使使用反射将某些代码应用于COM对象也是如此。

我希望它可以帮助一些想要将自己的自定义流构建到VS中的人。 谢谢弗拉德和杰克。

答案 2 :(得分:1)

如果要使用脚本附加到进程调试,可以考虑使用powershell。

或者据我所知,我们可以自定义附加到流程的代码。

Attach debugger in C# to another process

但是如果你必须使用VS IDE调试功能,我们经常使用远程调试工具:https://msdn.microsoft.com/library/y7f5zaaa.aspx