VS Code似乎有一个名为Launch.json的文件来自定义运行应用程序进行调试时会发生什么。 我们如何为Visual Studio进行此操作? 我的目标是在没有VS Docker Tools的情况下运行docker-compose,然后在容器启动后告诉VS附加到容器。
我的问题并非特定于这种情况,它也可能是另一种使用调试器方案运行的自定义方式。 理想情况下,我可以运行一些命令来构建项目,运行项目,然后告诉VS附加到它并将该逻辑与VS关联。
答案 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
此脚本执行以下操作:
dotnet.exe
进程自定义可执行文件来自一个单独的解决方案,我刚刚构建了一些命令行应用程序。 我使用了此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。