/** This tool produces gsdoc files from source files. Autogsdoc ... a tool to make documentation from source code Copyright (C) 2001 Free Software Foundation, Inc. Written by: richard@brainstorm.co.uk Created: October 2001 This file is part of the GNUstep Project This library is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. You should have received a copy of the GNU General Public License along with this library; see the file COPYING.LIB. If not, write to the Free Software Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. The autogsdoc tool

The autogsdoc tool is a command-line utility for parsing ObjectiveC source code (header files and optionally source files) in order to generate documentation covering the public interface of the various classes, categories, and protocols in the source.

The simple way to use this is to run the command with one or more header file names as arguments ... the tool will automatically parse corresponding source files in the same directory, and produce gsdoc files as output. You may also supply source file names (in which case documentation will be produced for the private methods within the source files), and the names of existing gsdoc documentation files (in which case their contents will be indexed).

Even without any human assistance, this tool will produce skeleton documents listing the methods in the classes found in the source files, but more importantly it can take specially formatted comments from the source files and insert those comments into the gsdoc output.

Any comment beginning with slash and two asterisks rather than the common slash and single asterisk, is taken to be gsdoc markup, to be use as the description of the class or method following it. This comment text is reformatted and then inserted into the output.

There are some cases where special extra processing is performed, predominantly in the first comment found in the source file, from which various chunks of gsdoc markup may be extracted and placed into appropriate locations in the output document -

<abstract> An abstract of the content of the document ... placed in the head of the gsdoc output. <author> A description of the author of the code - may be repeated to handle the case where a document has multiple authors. Placed in the head of the gsdoc output. <back> Placed in the gsdoc output just before the end of the body of the document - intended to be used for appendices, index etc. <chapter> Placed immediately before any generated class documentation ... intended to be used to provide overall description of how the code being documented works.
<copy> Copyright of the content of the document ... placed in the head of the gsdoc output. <date> Date of the revision of the document ... placed in the head of the gsdoc output. If this is omitted the tool will try to construct a value from the RCS Date tag (if available). <front> Inserted into the document at the start of the body ... intended to provide for introduction or contents pages etc. <title> Title of the document ... placed in the head of the gsdoc output. If this is omitted the tool will generate a (probably poor) title of its own. NBThis markup may be used within class, category, or protocol documentation ... if so, it is extracted and wrapped round the rest of the documentation for the class as the classes chapter. The rest of the class documentation is normally inserted at the end of the chapter, but may instead be sbstituted in in place of the <unit /> pseudo-element within the <chapter> element. <version> Version identifier of the document ... placed in the head of the gsdoc output. If this is omitted the tool will try to construct a value from the RCS Revision tag (if available).

In comments being used to provide text for a method description, the following markup is removed from the text and handled specially -

<init /> The method is marked as being the designated initialiser for the class. <override-subclass /> The method is marked as being one which subclasses must override (eg an abstract method). <override-never /> The method is marked as being one which subclasses should NOT override. <standards> ... </standards> The markup is removed from the description and placed after it in the gsdoc output - so that the method is described as conforming (or not conforming) to the specified standards.

Generally, the text in comments is reformatted to standardise and indent it nicely ... the reformatting is not performed on any text inside an <example> element.
When the text is reformatted, it is broken into whitespace separated 'words' which are then subjected to some extra processing ...

Certain well known constants such as YES, NO, and nil are enclosed in <code> ... </code> markup. The names of method arguments within method descriptions are enclosed in <var> ... </var> markup. Method names (beginning with a plus or minus) are enclosed in <ref...> ... </ref> markup.
eg. -init
Method specifiers including class names (beginning and ending with square brackets) are enclosed in <ref...> ... </ref> markup.
eg. [NSObject -init]

The tools accepts certain user defaults (which can of course be supplied as command-line arguments as usual) -

Declared Specify where headers are to be documented as being found.
The actual name produced in the documentation is formed by appending the last component of the header file name to the value of this default.
If this default is not specified, the full name of the header file (as supplied on the command line), with the HeaderDirectory default prepended, is used.
DocumentationDirectory May be used to specify the directory in which generated documentation is to be placed. If this is not set, output is placed in the current directory. HeaderDirectory May be used to specify the directory to be searched for header files. If this is not specified, headers are looked for relative to the current directory or using absolute path names if given. Project May be used to specify the name of this project ... determines the name of the index reference file produced as part of the documentation to provide information enabling other projects to cross-reference to items in this project. SourceDirectory May be used to specify the directory to be searched for header files. If this is not specified, headers are looked for relative to the current directory or using absolute path names if given.
Inter-document linkage

When supplied with a list of documents to process, the tool will set up linkage between documents using the gsdoc 'prev', 'next', and 'up' attributes.

The first document processed will be the 'up' link for all subsequent documents.

The 'prev' and 'next' links will be set up to link the documents in the order in which they are processed.

*/ #include #include "AGSParser.h" #include "AGSOutput.h" #include "AGSIndex.h" #include "AGSHtml.h" int main(int argc, char **argv, char **env) { NSProcessInfo *proc; NSArray *args; unsigned i; NSUserDefaults *defs; NSFileManager *mgr; NSString *documentationDirectory; NSString *declared; NSString *headerDirectory; NSString *sourceDirectory; NSString *projectName; NSString *refsFile; AGSIndex *prjRefs; AGSIndex *indexer; AGSParser *parser; AGSOutput *output; NSString *up = nil; NSString *prev = nil; unsigned pass; CREATE_AUTORELEASE_POOL(pool); #ifdef GS_PASS_ARGUMENTS [NSProcessInfo initializeWithArguments: argv count: argc environment: env]; #endif defs = [NSUserDefaults standardUserDefaults]; [defs registerDefaults: [NSDictionary dictionaryWithObjectsAndKeys: @"Untitled", @"ProjectName", nil]]; projectName = [defs stringForKey: @"ProjectName"]; declared = [defs stringForKey: @"Declared"]; headerDirectory = [defs stringForKey: @"HeaderDirectory"]; if (headerDirectory == nil) { headerDirectory = @"."; } sourceDirectory = [defs stringForKey: @"SourceDirectory"]; if (sourceDirectory == nil) { sourceDirectory = headerDirectory; } documentationDirectory = [defs stringForKey: @"DocumentationDirectory"]; if (documentationDirectory == nil) { documentationDirectory = @"."; } proc = [NSProcessInfo processInfo]; if (proc == nil) { NSLog(@"unable to get process information!"); exit(1); } mgr = [NSFileManager defaultManager]; prjRefs = [AGSIndex new]; indexer = [AGSIndex new]; parser = [AGSParser new]; output = [AGSOutput new]; args = [proc arguments]; /* * On the initial pass (pass0), we parse all files, produce indexes, * and write gsdoc output, but not html output. * * On the next pass, we have all the indexing info, so we can use it * to produce html output. */ for (pass = 0; pass < 2; pass++) { CREATE_AUTORELEASE_POOL(arp); for (i = 1; i < [args count]; i++) { NSString *arg = [args objectAtIndex: i]; if ([arg hasPrefix: @"-"]) { i++; // Skip next value ... it is a default. } else if ([arg hasSuffix: @".h"] == YES || [arg hasSuffix: @".m"] == YES || [arg hasSuffix: @".gsdoc"]== YES) { NSString *gsdocfile; NSString *htmlfile; NSString *hfile; NSString *sfile; NSString *ddir; NSString *hdir; NSString *sdir; NSString *file; NSString *generated; BOOL isSource = [arg hasSuffix: @".m"]; BOOL isDocumentation = [arg hasSuffix: @".gsdoc"]; file = [[arg lastPathComponent] stringByDeletingPathExtension]; hdir = [arg stringByDeletingLastPathComponent]; if ([hdir length] == 0) { hdir = headerDirectory; sdir = sourceDirectory; } else if ([hdir isAbsolutePath] == YES) { sdir = hdir; } else { hdir = [headerDirectory stringByAppendingPathComponent: hdir]; sdir = [sourceDirectory stringByAppendingPathComponent: sdir]; } ddir = documentationDirectory; hfile = [hdir stringByAppendingPathComponent: file]; hfile = [hfile stringByAppendingPathExtension: @"h"]; sfile = [sdir stringByAppendingPathComponent: file]; sfile = [sfile stringByAppendingPathExtension: @"m"]; gsdocfile = [ddir stringByAppendingPathComponent: file]; gsdocfile = [gsdocfile stringByAppendingPathExtension: @"gsdoc"]; htmlfile = [ddir stringByAppendingPathComponent: file]; htmlfile = [htmlfile stringByAppendingPathExtension: @"html"]; if (pass == 0) { /* * We perform parsing of source code in pass 0 only. */ [parser reset]; if (isSource == NO && isDocumentation == NO) { /* * Try to parse header to see what needs documenting. */ if ([mgr isReadableFileAtPath: hfile] == NO) { NSLog(@"No readable header at '%@' ... skipping", hfile); continue; } if (declared != nil) { [parser setDeclared: [declared stringByAppendingPathComponent: [hfile lastPathComponent]]]; } [parser parseFile: hfile isSource: NO]; } else if (isSource == YES) { /* * Try to parse source *as-if-it-was-a-header* * to see what needs documenting. */ if ([mgr isReadableFileAtPath: sfile] == NO) { NSLog(@"No readable source at '%@' ... skipping", sfile); continue; } if (declared != nil) { [parser setDeclared: [declared stringByAppendingPathComponent: [sfile lastPathComponent]]]; } [parser parseFile: sfile isSource: NO]; } if (isDocumentation == NO) { /* * If we can read a source file, parse it for any * additional information on items found in the header. */ if ([mgr isReadableFileAtPath: sfile] == YES) { [parser parseFile: sfile isSource: YES]; } /* * Set up linkage for this file. */ [[parser info] setObject: file forKey: @"base"]; if (up == nil) { up = file; } else { [[parser info] setObject: up forKey: @"up"]; } if (prev != nil) { [[parser info] setObject: prev forKey: @"prev"]; } prev = file; if (i < [args count] - 1) { unsigned j = i + 1; while (j < [args count]) { NSString *name = [args objectAtIndex: j++]; if ([name hasSuffix: @".h"] || [name hasSuffix: @".m"] || [name hasSuffix: @".gsdoc"]) { name = [[name lastPathComponent] stringByDeletingPathExtension]; [[parser info] setObject: name forKey: @"next"]; break; } } } generated = [output output: [parser info]]; if ([generated writeToFile: gsdocfile atomically: YES] == NO) { NSLog(@"Sorry unable to write %@", gsdocfile); } } } if ([mgr isReadableFileAtPath: gsdocfile] == YES) { CREATE_AUTORELEASE_POOL(pool); GSXMLParser *parser; AGSIndex *locRefs; AGSHtml *html; NSString *result; parser = [GSXMLParser parserWithContentsOfFile: gsdocfile]; [parser substituteEntities: YES]; [parser doValidityChecking: YES]; if ([parser parse] == NO) { NSLog(@"WARNING %@ is not a valid document", gsdocfile); } if (![[[[parser doc] root] name] isEqualToString: @"gsdoc"]) { NSLog(@"not a gsdoc document - because name node is %@", [[[parser doc] root] name]); return 1; } locRefs = AUTORELEASE([AGSIndex new]); [locRefs makeRefs: [[parser doc] root]]; if (pass == 1) { /* * We only perform final outpu in pass 1 */ html = AUTORELEASE([AGSHtml new]); [html setGlobalRefs: prjRefs]; [html setLocalRefs: locRefs]; result = [html outputDocument: [[parser doc] root]]; if ([result writeToFile: htmlfile atomically: YES] == NO) { NSLog(@"Sorry unable to write %@", htmlfile); } } else { /* * We only accumulate index info in pass 0 */ [indexer mergeRefs: [locRefs refs]]; [prjRefs mergeRefs: [locRefs refs]]; } RELEASE(pool); } else if (isDocumentation) { NSLog(@"No readable documentation at '%@' ... skipping", gsdocfile); } } else { NSLog(@"Unknown argument '%@' ... ignored", arg); } } RELEASE(arp); } /* * Save references. */ refsFile = [documentationDirectory stringByAppendingPathComponent: projectName]; refsFile = [refsFile stringByAppendingPathExtension: @"gsdocidx"]; if ([[prjRefs refs] writeToFile: refsFile atomically: YES] == NO) { NSLog(@"Sorry unable to write %@", refsFile); } RELEASE(pool); return 0; }