views:

419

answers:

4

Is there a way to debug completely a windows service with Delphi?

+3  A: 

Yes, there is: Debugging services: an easy way

Do you create services with Delphi? Then maybe you are also annoyed at the time consuming way of starting, restarting, killing and attaching to the service process application every time. Well, there is remedy.

You don’t need to do this. Instead run Delphi as a SYSTEM application and do some minor adaptions to the service code.

Mick
RunAsSys sounds good. The example code in the article you linked uses conditional defines to select the "mode". Do you know whether RunAsSys allows for selecting debug mode without having to rebuild the app? Ie using a command line parameter to go into debug mode instead of running as a service?
Marjan Venema
I am not certain, but it seems that you would be able to add that functionality yourself to it.
Mick
+1  A: 

Yes there is.

In your dpr:

Application.CreateForm(TMyService, MyService);

_ServiceInDebugMode := SysUtils.FindCmdLineSwitch('DEBUG', True);
if _ServiceInDebugMode then
  DebugService(MyService)
else
  SvcMgr.Application.Run;

DebugService is a procedure that creates a debug form, a service control thread and kicks off everything by calling Forms.Application.Run.

You can compare the service control thread with the Windows' SCM (Service Control Manager), the debug form with an application that talks to the SCM (such as the services.msc) to start and stop services. The service should also have its own thread (service thread) to respond to control codes coming in from the SCM or our service control thread and one or many more separate threads to do its actual work. You want separate threads for the actual work (instead of coding it in the event handlers of your TService descendant) to ensure that the thread in which the TService itself runs is always free to respond to control codes from the SCM and you can still stop and start the service even when per chance a/the worker thread is frozen.

This approach allows you to debug the service app code as well, but involves a fair amount of code and placing a couple of hooks into windows api functions to work properly. Too much to show here at short notice. Maybe I'll write it up in an article some day.

In the mean time, if you don't want to code it all yourself you have two options. Either go with a library such as SVCOM or the one mentioned by Mick which do it all for you, or when in debug mode by-pass the service code all together and "simply" start your service as a "normal" forms application. You will have to disentangle the real functionality of your service from your TService descendant's code/event handlers, but that is something I'd recommend anyway for the reasons mentioned above.

Marjan Venema
I have the same app with forms and service. With forms it works. With services it doesn't work.
O Engenheiro
+1  A: 

You can use unitDebugService.pas from Colin Wilson's NT Low Level Utilities

and then in the DPR:

begin
  if (paramCount > 0) and (SameText(ParamStr(1), '-DEBUG')) then
  begin
    FreeAndNil (Application);
    Application := TDebugServiceApplication.Create(nil);
  end;

  //... the rest of the normal DPR code
end.

This way you can run from within Delphi with debugging (by setting the project Debugger Parameters), use the EXE as a service, or run from the commandline with the -DEBUG switch, and .

jasonpenny
This works awesome, thanks for the tip!
Mick
+4  A: 

Use Run -> Attach to process. This way you can debug a service without doing any changes to its code. The only tricky part maybe debugging the service start code, because attaching may require some time, and the start must happen in 30s (although you can tweak Windows to allow a longer time). You can use a delay (sleep...) to allow you to attach in time, or if you just need to see what happens you can use OutputDebugString() to print to the debug output (use Delphi Event View to see it).

ldsandon
I've tried this, but only appears the cpu window with assembly codes.
O Engenheiro