1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 #include <precomp.h>
23 #include <adc_cl.hxx>
24 
25 
26 // NOT FULLY DEFINED SERVICES
27 #include <algorithm>
28 #include <cosv/x.hxx>
29 #include <cosv/file.hxx>
30 #include <cosv/tpl/tpltools.hxx>
31 #include <ary/ary.hxx>
32 #include <tools/tkpchars.hxx>
33 #include <adc_msg.hxx>
34 #include "adc_cmds.hxx"
35 #include "adc_cmd_parse.hxx"
36 #include "cmd_sincedata.hxx"
37 
38 
39 namespace autodoc
40 {
41 
42 CommandLine * CommandLine::pTheInstance_ = 0;
43 
44 const char * const C_sUserGuide =
45 "\n\n\n"
46 "               General Use of Autodoc\n"
47 "               ----------------------\n"
48 "\n"
49 "   Example for C++:\n"
50 "\n"
51 "   -html <OutputDirectory> -name \"UDK 3.x anything\" -lg c++\n"
52 "        -p <ProjName> <ProjectRootDirectory>\n"
53 "            -t <SourceDir_relativeToProjectRoot>\n"
54 "\n"
55 "   There may be several projects specified by -p.\n"
56 "\n"
57 "\n"
58 "   Example for IDL:\n"
59 "\n"
60 "   -html <OutputDirectory> -name \"UDK 3.x anything\" -lg idl\n"
61 "         -t <SourceDir1> <SourceDir2>\n"
62 "\n"
63 "   For both languages, instead of or in addition to -t may be\n"
64 "   used -d (no subdirectories) or -f (just one file). There can\n"
65 "   be multiple arguments after each of these options (-t -d -f).\n"
66 "\n"
67 "\n"
68 "           Replacing @since Tag Content\n"
69 "           ----------------------------\n"
70 "\n"
71 "   In both languages you can give a transformation file to replace\n"
72 "   entries in @since tags by different entries.\n"
73 "   This file is given by the option\n"
74 "       -sincefile <TransformationFilePath>\n"
75 "   This option has to appear between the -html and the -lg option.\n"
76 "   Example:\n"
77 "   -html <OutputDirectory> -sincefile replacesince.txt\n"
78 "       -name \"UDK 3.x anything\" -lg idl -t <SourceDir>\n"
79 "\n"
80 "\n";
81 
82 
83 #if 0   // FUTURE
84 "\n\n\n"
85 "               Use of Autodoc\n"
86 "               --------------\n"
87 "\n"
88 "   Basics:\n"
89 "\n"
90 "   Autodoc may perform different tasks.\n"
91 "\n"
92 "   Possible tasks are\n"
93 "       - parsing source code\n"
94 "       - creating HTML-output.\n"
95 "   On the command line each task starts with a specific\n"
96 "   option:\n"
97 "       '-parse' for parsing source code,\n"
98 "       '-html' for creating HTML.\n"
99 "   All command line options, related to one task, have to follow before\n"
100 "   the starting option of the next task.\n"
101 "\n"
102 "   Within the task '-parse', there may be defined different projects.\n"
103 "   A project definition is started with '-p'.\n"
104 "   All not project specific options within the task '-parse' have to\n"
105 "   appear in front of the first '-p'.\n"
106 "   There can be no project at all. Then all options, available for\n"
107 "   projects, can be used like for one nameless default project, without using\n"
108 "   '-p', but these options still have to appear behind all other\n"
109 "   options of the task '-parse'.\n"
110 "\n"
111 "\n"
112 "   Legend:\n"
113 "\n"
114 "       <SomeText>      Describes an argument.\n"
115 "       'SomeText'      Within '...' is the literal value of an argument.\n"
116 "       +               There can be multiple arguments.\n"
117 "       |               Separator for alternative literal values of an argument.\n"
118 "\n"
119 "\n"
120 "   Syntax:\n"
121 "\n"
122 "   -parse\n"
123 "       -name <RepositoryName>]\n"
124 "       -lg 'c++'|'idl'\n"
125 "       -extg <AdditonalExtensions>+\n"
126 "       -docg 'usehtml'\n"
127 "       -p <ProjectName> <ProjectRootDir>\n"
128 "           -l 'c++'|'idl'\n"
129 "           -ext <AdditonalExtensions>+\n"
130 "           -doc 'usehtml'\n"
131 "           -d <SourceDir_relative2ProjectRootDir_nosubdirs>+\n"
132 "           -t <SourceTree_relative2ProjectRootDir>+\n"
133 "           -f <SourceFile_relative2ProjectRootDir>+\n"
134 "   -html <OutputDir>\n"
135 "       -xlinks <Namespace> <ExternLinksRootDir>\n"
136 "   -i <CommandFilePath>\n"
137 "   -v <VerboseNr>\n"
138 "\n"
139 "\n"
140 "   Detailed Options Description:\n"
141 "\n"
142 "   Option      Arguments\n"
143 "   ----------------------------------------------------------\n"
144 "\n"
145 "   -parse      \n\n"
146 "               Starts the task \"Parse source code\".\n"
147 "               May be omitted, if it would be the first option on the\n"
148 "               command line.\n"
149 "\n"
150 "   -name       <RepositoryName>\n\n"
151 "               This name is used for naming the repository in\n"
152 "               human readable output. In future it may be used also for\n"
153 "               identifiing a repository in searches.\n"
154 "\n"
155 "   -lg         'c++|'idl'\n\n"
156 "               Identifies the programming language to be parsed.\n"
157 "                   'c++':  C++\n"
158 "                           Files with extensions '.h', '.hxx' are parsed.\n"
159 "                   'idl':  UNO-IDL\n"
160 "                           Files with extensions '.idl' are parsed.\n"
161 "               Here the language is set globally for all projects.\n"
162 "               A project can override this by using '-l'.\n"
163 "\n"
164 "   -extg       <.AdditionalExtension>+\n\n"
165 "               Has to follow immediately behind '-lg'.\n"
166 "               Specifies additional extensions, that will be recognised as\n"
167 "               source code files of the previously specified programming\n"
168 "               language.  Each extension has to start with '.'.\n"
169 "               It is possible to include extensionless files, too,\n"
170 "               by the argument '.'\n"
171 "               Here these extensions are set globally for all projects.\n"
172 "               A project can override this by using '-l' and '-ext'.\n"
173 "\n"
174 "   -docg       'html'|'nohtml'\n\n"
175 "               Specifies the default for all comments in source code, so \n"
176 "               that HTML-tags are interpreted as such or else treated as\n"
177 "               regular text.\n"
178 "               Without this option, the default is 'nohtml'.\n"
179 "               Here the default is set globally for all projects.\n"
180 "               A project can override this by using '-doc'.\n"
181 "\n"
182 "   -p          <ProjectName> <ProjectRootDirectory>\n\n"
183 "               ProjectName is used in output as human readable identifier\n"
184 "               for the project. ProjectRootDirectory is the path,\n"
185 "               where the arguments of '-d', '-t' and '-f' are relative to.\n"
186 "               This option can be omitted, then there is no project name\n"
187 "               and all paths are relative to the current working directory.\n"
188 "\n"
189 "   -l          'c++|'idl'\n\n"
190 "               Overrides -lg and -extg for the current project, which is\n"
191 "               specified by the last previous '-p'.\n"
192 "               For details see at option '-lg'.\n"
193 "\n"
194 "   -ext        <.AdditionalExtension>+\n\n"
195 "               Can be used only immediately behind '-l'.\n"
196 "               Overrides -extg for the current project, which is\n"
197 "               specified by the last previous '-p'.\n"
198 "               For details see at option '-extg'.\n"
199 "\n"
200 "   -doc        'html'|'nohtml'\n\n"
201 "               Overrides -docg for the current project, which is\n"
202 "               specified by the last previous '-p'.\n"
203 "               For details see at option '-docg'.\n"
204 "\n"
205 "   -d          <SourceDir_relative2ProjectRootDir_nosubdirs>+\n\n"
206 "               For the current project all files in the given\n"
207 "               directories are parsed, which have valid extensions.\n"
208 "               Subdirectories are NOT parsed.\n"
209 "\n"
210 "   -t          <SourceTree_relative2ProjectRootDir>+\n\n"
211 "               For the current project all files in the given\n"
212 "               directories AND its subdirectories are parsed, which\n"
213 "               have valid extensions.\n"
214 "\n"
215 "   -f          <SourceFile_relative2ProjectRootDir>+\n\n"
216 "               For the current project and language the given files\n"
217 "               are parsed. It doesn't matter, if their extensions match\n"
218 "               the valid extensions.\n"
219 "\n"
220 "   -html       <OutputRootDir>\n\n"
221 "               Starts the task \"Create HTML output\".\n"
222 "\n"
223 "   -xlinks     <Namespace> <ExternLinksRootDir>\n\n"
224 "               This option allows, to create links to external\n"
225 "               HTML-documents.\n"
226 "               For all source code objects (like classes or functions)\n"
227 "               which belong in the given namespace, the given root\n"
228 "               directory is used as a base for links to them.\n"
229 "               Presently, this works only for C++-mappings of IDL-items.\n"
230 "               The given namespace has to be absolute.\n"
231 "\n"
232 "   -i          <CommandFilePath>\n\n"
233 "               This option is replaced by the contents of the given\n"
234 "               file. The file has to be ASCII and each option\n"
235 "               has to start in the first column of a new line.\n"
236 "               So each valid line starts with a '-'.\n"
237 "               Empty lines are allowed.\n"
238 "               Comment lines have to start with '#'\n"
239 "\n"
240 "   -v          <VerboseNumber>\n\n"
241 "               Show details during parsing:\n"
242 "                   2    shows each parsed letter,\n"
243 "                   4    shows stored objects.\n"
244 "                   1    shows recognised tokens.\n"
245 "               These bit-values can be combined.\n"
246 "               This option suppresses errors, because of\n"
247 "               missing output options (no '-html').\n";
248 #endif // 0, FUTURE
249 
250 
CommandLine()251 CommandLine::CommandLine()
252     :   nDebugStyle(0),
253         pSinceTransformator(new command::SinceTagTransformationData),
254         aCommands(),
255         bInitOk(false),
256         pCommand_CreateHtml(0),
257         pReposy( & ary::Repository::Create_() ),
258         bCpp(false),
259         bIdl(false)
260 {
261     csv_assert(pTheInstance_ == 0);
262     pTheInstance_ = this;
263 }
264 
~CommandLine()265 CommandLine::~CommandLine()
266 {
267     csv::erase_container_of_heap_ptrs(aCommands);
268     pTheInstance_ = 0;
269 }
270 
271 int
Run() const272 CommandLine::Run() const
273 {
274     Cout() << "\nAutodoc version 2.2.5"
275            << "\n---------------------"
276            << "\n" << Endl();
277 
278     bool
279         ok = true;
280     for ( CommandList::const_iterator it = aCommands.begin();
281           ok AND it != aCommands.end();
282           ++it )
283     {
284         ok = (*it)->Run();
285     }
286 
287     if (pCommand_CreateHtml != 0)
288     {
289         StreamStr aDiagnosticMessagesFile(700);
290         aDiagnosticMessagesFile
291             << pCommand_CreateHtml->OutputDir()
292             << csv::ploc::Delimiter()
293             << "Autodoc_DiagnosticMessages.txt";
294         TheMessages().WriteFile(aDiagnosticMessagesFile.c_str());
295     }
296 
297     return ok ? 0 : 1;
298 }
299 
300 CommandLine &
Get_()301 CommandLine::Get_()
302 {
303     csv_assert(pTheInstance_ != 0);
304     return *pTheInstance_;
305 }
306 
307 bool
DoesTransform_SinceTag() const308 CommandLine::DoesTransform_SinceTag() const
309 {
310     return pSinceTransformator->DoesTransform();
311 }
312 
313 //bool
314 //CommandLine::Strip_SinceTagText( String & io_sSinceTagValue ) const
315 //{
316 //    return pSinceTransformator->StripSinceTagText(io_sSinceTagValue);
317 //}
318 
319 const String &
DisplayOf_SinceTagValue(const String & i_sVersionNumber) const320 CommandLine::DisplayOf_SinceTagValue( const String & i_sVersionNumber ) const
321 {
322     return pSinceTransformator->DisplayOf(i_sVersionNumber);
323 }
324 
325 void
do_Init(int argc,char * argv[])326 CommandLine::do_Init( int                 argc,
327                       char *              argv[] )
328 {
329   try
330   {
331     bInitOk = false;
332     StringVector    aParameters;
333 
334     char * * itpEnd = &argv[0] + argc;
335     for ( char * * itp = &argv[1]; itp != itpEnd; ++itp )
336     {
337      	if ( strncmp(*itp, "-I:", 3) != 0 )
338             aParameters.push_back(String(*itp));
339         else
340             load_IncludedCommands(aParameters, (*itp)+3);
341     }
342 
343     StringVector::const_iterator itEnd = aParameters.end();
344     for ( StringVector::const_iterator it = aParameters.begin();
345           it != itEnd;
346         )
347     {
348         if ( *it == command::C_opt_Verbose )
349             do_clVerbose(it,itEnd);
350         else if ( *it == command::C_opt_LangAll
351                   OR *it == command::C_opt_Name
352                   OR *it == command::C_opt_DevmanFile )
353             do_clParse(it,itEnd);
354         else if (*it == command::C_opt_CreateHtml)
355             do_clCreateHtml(it,itEnd);
356         else if (*it == command::C_opt_SinceFile)
357             do_clSinceFile(it,itEnd);
358         else if (*it == command::C_opt_IgnoreDefine)
359         {
360             AddIgnoreDefine(*(++it));
361             ++it;
362         }
363         else if (*it == command::C_opt_ExternNamespace)
364         {
365             sExternNamespace = *(++it);
366             ++it;
367             if ( strncmp(sExternNamespace.c_str(), "::", 2) != 0)
368             {
369              	throw command::X_CommandLine(
370                         "-extnsp needs an absolute qualified namespace, starting with \"::\"."
371                         );
372             }
373         }
374         else if (*it == command::C_opt_ExternRoot)
375         {
376             ++it;
377             StreamLock sl(1000);
378             if ( csv::compare(*it, 0, "http://", 7) != 0 )
379             {
380                 sl() << "http://" << *it;
381             }
382             if ( *(sl().end()-1) != '/')
383                 sl() << '/';
384             sExternRoot = sl().c_str();
385 
386             ++it;
387         }
388 //        else if (*it == command::C_opt_CreateXml)
389 //            do_clCreateXml(it,itEnd);
390 //        else if (command::C_opt_Load)
391 //            do_clLoad(it,itEnd);
392 //        else if (*it == command::C_opt_Save)
393 //            do_clSave(it,itEnd);
394         else if (*it == "-h" OR *it == "-?" OR *it == "?")
395             // Leads to displaying help, because bInitOk stays on false.
396          	return;
397         else if ( *it == command::C_opt_Parse )
398             // Only for backwards compatibility.
399             //   Just ignore "-parse".
400             ++it;
401         else
402         {
403             StreamLock sl(200);
404          	throw command::X_CommandLine(
405                             sl() << "Unknown commandline option \""
406                                  << *it
407                                  << "\"."
408                                  << c_str );
409         }
410     }   // end for
411     sort_Commands();
412 
413     bInitOk = true;
414 
415   }   // end try
416   catch ( command::X_CommandLine & xxx )
417   {
418     xxx.Report( Cerr() );
419   }
420   catch ( csv::Exception & xxx )
421   {
422     xxx.GetInfo( Cerr() );
423   }
424 }
425 
426 void
do_PrintUse() const427 CommandLine::do_PrintUse() const
428 {
429     Cout() << C_sUserGuide << Endl();
430 }
431 
432 bool
inq_CheckParameters() const433 CommandLine::inq_CheckParameters() const
434 {
435     if (NOT bInitOk OR aCommands.size() == 0)
436         return false;
437     return true;
438 }
439 
440 void
load_IncludedCommands(StringVector & out,const char * i_filePath)441 CommandLine::load_IncludedCommands( StringVector &      out,
442                                     const char *        i_filePath )
443 {
444     CharacterSource
445         aIncludedCommands;
446     csv::File
447         aFile(i_filePath, csv::CFM_READ);
448     if (NOT aFile.open())
449     {
450      	Cerr() << "Command include file \""
451                << i_filePath
452                << "\" not found."
453                << Endl();
454         throw command::X_CommandLine("Invalid file in option -I:<command-file>.");
455     }
456     aIncludedCommands.LoadText(aFile);
457     aFile.close();
458 
459     bool bInToken = false;
460     StreamLock aTransmit(200);
461     for ( ; NOT aIncludedCommands.IsFinished(); aIncludedCommands.MoveOn() )
462     {
463         if (bInToken)
464         {
465             if (aIncludedCommands.CurChar() <= 32)
466             {
467                 const char *
468                     pToken = aIncludedCommands.CutToken();
469                 bInToken = false;
470 
471              	if ( strncmp(pToken, "-I:", 3) != 0 )
472              	{
473              	    aTransmit().seekp(0);
474              	    aTransmit() << pToken;
475              	    aTransmit().replace_all('\\', *csv::ploc::Delimiter());
476              	    aTransmit().replace_all('/', *csv::ploc::Delimiter());
477                     out.push_back(String(aTransmit().c_str()));
478              	}
479                 else
480                     load_IncludedCommands(out, pToken+3);
481             }
482         }
483         else
484         {
485             if (aIncludedCommands.CurChar() > 32)
486             {
487                 aIncludedCommands.CutToken();
488                 bInToken = true;
489             }
490         }   // endif (bInToken) else
491 
492     }   // end while()
493 }
494 
495 namespace
496 {
497 inline int
v_nr(StringVector::const_iterator it)498 v_nr(StringVector::const_iterator it)
499 {
500  	return int( *(*it).c_str() ) - int('0');
501 }
502 }   // anonymous namespace
503 
504 void
do_clVerbose(opt_iter & it,opt_iter itEnd)505 CommandLine::do_clVerbose(  opt_iter &          it,
506                             opt_iter            itEnd )
507 {
508     ++it;
509     if ( it == itEnd ? true : v_nr(it) < 0 OR v_nr(it) > 7 )
510         throw command::X_CommandLine( "Missing or invalid number in -v option." );
511     nDebugStyle = v_nr(it);
512     ++it;
513 }
514 
515 void
do_clParse(opt_iter & it,opt_iter itEnd)516 CommandLine::do_clParse( opt_iter &          it,
517                          opt_iter            itEnd )
518 {
519     command::Command *
520         pCmd_Parse = new command::Parse;
521     pCmd_Parse->Init(it, itEnd);
522     aCommands.push_back(pCmd_Parse);
523 }
524 
525 void
do_clCreateHtml(opt_iter & it,opt_iter itEnd)526 CommandLine::do_clCreateHtml( opt_iter &          it,
527                               opt_iter            itEnd )
528 {
529     pCommand_CreateHtml = new command::CreateHtml;
530     pCommand_CreateHtml->Init(it, itEnd);
531     aCommands.push_back(pCommand_CreateHtml);
532 }
533 
534 void
do_clSinceFile(opt_iter & it,opt_iter itEnd)535 CommandLine::do_clSinceFile( opt_iter &          it,
536                              opt_iter            itEnd )
537 {
538     pSinceTransformator->Init(it, itEnd);
539 }
540 
541 
542 namespace
543 {
544 
545 struct Less_RunningRank
546 {
operator ()autodoc::__anon5eb9f8a90211::Less_RunningRank547     bool                operator()(
548                             const command::Command * const &
549                                                 i1,
550                             const command::Command * const &
551                                                 i2 ) const
552                         { return i1->RunningRank() < i2->RunningRank(); }
553 };
554 
555 }   // anonymous namespace
556 
557 
558 
559 void
sort_Commands()560 CommandLine::sort_Commands()
561 {
562     std::sort( aCommands.begin(),
563                aCommands.end(),
564                Less_RunningRank() );
565 }
566 
567 }   // namespace autodoc
568