To change parse-log-file so it establishes a restart instead of a condition handler, you can change the **HANDLER-CASE** to a **RESTART-CASE**. The form of **RESTART-CASE** is quite similar to a **HANDLER-CASE** except the names of restarts are just names, not necessarily the names of condition types. In general, a restart name should describe the action the restart takes. In parse-log-file, you can call the restart skip-log-entry since that’s what it does. The new version will look like this:

    If you invoke this version of parse-log-file on a log file containing corrupted entries, it won’t handle the error directly; you’ll end up in the debugger. However, there among the various restarts presented by the debugger will be one called skip-log-entry, which, if you choose it, will cause parse-log-file to continue on its way as before. To avoid ending up in the debugger, you can establish a condition handler that invokes the skip-log-entry restart automatically.

    The advantage of establishing a restart rather than having parse-log-file handle the error directly is it makes parse-log-file usable in more situations. The higher-level code that invokes parse-log-file doesn’t have to invoke the skip-log-entry restart. It can choose to handle the error at a higher level. Or, as I’ll show in the next section, you can add restarts to parse-log-entry to provide other recovery strategies, and then condition handlers can choose which strategy they want to use.

    But before I can talk about that, you need to see how to set up a condition handler that will invoke the skip-log-entry restart. You can set up the handler anywhere in the chain of calls leading to parse-log-file. This may be quite high up in your application, not necessarily in parse-log-file‘s direct caller. For instance, suppose the main entry point to your application is a function, log-analyzer, that finds a bunch of logs and analyzes them with the function analyze-log, which eventually leads to a call to parse-log-file. Without any error handling, it might look like this:

    1. (dolist (log (find-all-logs))

    where the function analyze-entry is presumably responsible for extracting whatever information you care about from each log entry and stashing it away somewhere.

    Thus, the path from the top-level function, log-analyzer, to parse-log-entry, which actually signals an error, is as follows:

    Assuming you always want to skip malformed log entries, you could change this function to establish a condition handler that invokes the skip-log-entry restart for you. However, you can’t use **HANDLER-CASE** to establish the condition handler because then the stack would be unwound to the function where the **HANDLER-CASE** appears. Instead, you need to use the lower-level macro **HANDLER-BIND**. The basic form of **HANDLER-BIND** is as follows:

    1. (handler-bind (binding*) form*)

    In this , the handler function is an anonymous function that invokes the restart skip-log-entry. You could also define a named function that does the same thing and bind it instead. In fact, a common practice when defining a restart is to define a function, with the same name and taking a single argument, the condition, that invokes the eponymous restart. Such functions are called restart functions. You could define a restart function for skip-log-entry like this:

    1. (defun skip-log-entry (c)

    Then you could change the definition of log-analyzer to this:

    As written, the skip-log-entry restart function assumes that a skip-log-entry restart has been established. If a malformed-log-entry-error is ever signaled by code called from log-analyzer without a skip-log-entry having been established, the call to **INVOKE-RESTART** will signal a **CONTROL-ERROR** when it fails to find the skip-log-entry restart. If you want to allow for the possibility that a malformed-log-entry-error might be signaled from code that doesn’t have a skip-log-entry restart established, you could change the skip-log-entry function to this:

    1. (defun skip-log-entry (c)
    2. (let ((restart (find-restart 'skip-log-entry)))

    **FIND-RESTART** looks for a restart with a given name and returns an object representing the restart if the restart is found and **NIL** if not. You can invoke the restart by passing the restart object to **INVOKE-RESTART**. Thus, when skip-log-entry is bound with , it will handle the condition by invoking the skip-log-entry restart if one is available and otherwise will return normally, giving other condition handlers, bound higher on the stack, a chance to handle the condition.