JavadocTool.java revision 3904:7486e172ca65
1/*
2 * Copyright (c) 2001, 2017, Oracle and/or its affiliates. All rights reserved.
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4 *
5 * This code is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License version 2 only, as
7 * published by the Free Software Foundation.  Oracle designates this
8 * particular file as subject to the "Classpath" exception as provided
9 * by Oracle in the LICENSE file that accompanied this code.
10 *
11 * This code is distributed in the hope that it will be useful, but WITHOUT
12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
14 * version 2 for more details (a copy is included in the LICENSE file that
15 * accompanied this code).
16 *
17 * You should have received a copy of the GNU General Public License version
18 * 2 along with this work; if not, write to the Free Software Foundation,
19 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
20 *
21 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
22 * or visit www.oracle.com if you need additional information or have any
23 * questions.
24 */
25
26package jdk.javadoc.internal.tool;
27
28
29import java.io.File;
30import java.util.ArrayList;
31import java.util.HashSet;
32import java.util.LinkedHashSet;
33import java.util.List;
34import java.util.Map;
35import java.util.Set;
36
37import javax.tools.JavaFileObject;
38import javax.tools.StandardJavaFileManager;
39
40import com.sun.tools.javac.code.ClassFinder;
41import com.sun.tools.javac.code.Symbol.Completer;
42import com.sun.tools.javac.code.Symbol.CompletionFailure;
43import com.sun.tools.javac.comp.Enter;
44import com.sun.tools.javac.tree.JCTree;
45import com.sun.tools.javac.tree.JCTree.JCClassDecl;
46import com.sun.tools.javac.tree.JCTree.JCCompilationUnit;
47import com.sun.tools.javac.util.Abort;
48import com.sun.tools.javac.util.Context;
49import com.sun.tools.javac.util.ListBuffer;
50import com.sun.tools.javac.util.Position;
51import jdk.javadoc.doclet.DocletEnvironment;
52
53import static jdk.javadoc.internal.tool.Main.Result.*;
54
55/**
56 *  This class could be the main entry point for Javadoc when Javadoc is used as a
57 *  component in a larger software system. It provides operations to
58 *  construct a new javadoc processor, and to run it on a set of source
59 *  files.
60 *
61 *  <p><b>This is NOT part of any supported API.
62 *  If you write code that depends on this, you do so at your own risk.
63 *  This code and its internal interfaces are subject to change or
64 *  deletion without notice.</b>
65 *
66 *  @author Neal Gafter
67 */
68public class JavadocTool extends com.sun.tools.javac.main.JavaCompiler {
69    ToolEnvironment toolEnv;
70
71    final Messager messager;
72    final ClassFinder javadocFinder;
73    final Enter javadocEnter;
74    final Set<JavaFileObject> uniquefiles;
75
76    /**
77     * Construct a new JavaCompiler processor, using appropriately
78     * extended phases of the underlying compiler.
79     */
80    protected JavadocTool(Context context) {
81        super(context);
82        messager = Messager.instance0(context);
83        javadocFinder = JavadocClassFinder.instance(context);
84        javadocEnter = JavadocEnter.instance(context);
85        uniquefiles = new HashSet<>();
86    }
87
88    /**
89     * For javadoc, the parser needs to keep comments. Overrides method from JavaCompiler.
90     */
91    @Override
92    protected boolean keepComments() {
93        return true;
94    }
95
96    /**
97     *  Construct a new javadoc tool.
98     */
99    public static JavadocTool make0(Context context) {
100        Messager messager = null;
101        try {
102            // force the use of Javadoc's class finder
103            JavadocClassFinder.preRegister(context);
104
105            // force the use of Javadoc's own enter phase
106            JavadocEnter.preRegister(context);
107
108            // force the use of Javadoc's own member enter phase
109            JavadocMemberEnter.preRegister(context);
110
111            // force the use of Javadoc's own todo phase
112            JavadocTodo.preRegister(context);
113
114            // force the use of Messager as a Log
115            messager = Messager.instance0(context);
116
117            return new JavadocTool(context);
118        } catch (CompletionFailure ex) {
119            messager.error(Position.NOPOS, ex.getMessage());
120            return null;
121        }
122    }
123
124    public DocletEnvironment getEnvironment(Map<ToolOption,
125            Object> jdtoolOpts,
126            List<String> javaNames,
127            Iterable<? extends JavaFileObject> fileObjects) throws ToolException {
128        toolEnv = ToolEnvironment.instance(context);
129        toolEnv.initialize(jdtoolOpts);
130        ElementsTable etable = new ElementsTable(context, jdtoolOpts);
131        javadocFinder.sourceCompleter = etable.xclasses
132                ? Completer.NULL_COMPLETER
133                : sourceCompleter;
134
135        if (etable.xclasses) {
136            // If -Xclasses is set, the args should be a list of class names
137            for (String arg: javaNames) {
138                if (!isValidPackageName(arg)) { // checks
139                    String text = messager.getText("main.illegal_class_name", arg);
140                    throw new ToolException(CMDERR, text);
141                }
142            }
143            if (messager.hasErrors()) {
144                return null;
145            }
146            etable.setClassArgList(javaNames);
147            // prepare, force the data structures to be analyzed
148            etable.analyze();
149            return new DocEnvImpl(toolEnv, etable);
150        }
151
152        ListBuffer<JCCompilationUnit> classTrees = new ListBuffer<>();
153
154        try {
155            StandardJavaFileManager fm = toolEnv.fileManager instanceof StandardJavaFileManager
156                    ? (StandardJavaFileManager) toolEnv.fileManager
157                    : null;
158            Set<String> packageNames = new LinkedHashSet<>();
159            // Normally, the args should be a series of package names or file names.
160            // Parse the files and collect the package names.
161            for (String arg: javaNames) {
162                if (fm != null && arg.endsWith(".java") && new File(arg).exists()) {
163                    if (new File(arg).getName().equals("module-info.java")) {
164                        messager.printWarningUsingKey("main.file_ignored", arg);
165                    } else {
166                        parse(fm.getJavaFileObjects(arg), classTrees, true);
167                    }
168                } else if (isValidPackageName(arg)) {
169                    packageNames.add(arg);
170                } else if (arg.endsWith(".java")) {
171                    if (fm == null) {
172                        String text = messager.getText("main.assertion.error", "fm == null");
173                        throw new ToolException(ABNORMAL, text);
174                    } else {
175                        String text = messager.getText("main.file_not_found", arg);
176                        throw new ToolException(ERROR, text);
177                    }
178                } else {
179                    String text = messager.getText("main.illegal_package_name", arg);
180                    throw new ToolException(CMDERR, text);
181                }
182            }
183
184            // Parse file objects provide via the DocumentationTool API
185            parse(fileObjects, classTrees, true);
186
187            etable.packages(packageNames)
188                    .classTrees(classTrees.toList())
189                    .scanSpecifiedItems();
190
191            // Parse the files in the packages and subpackages to be documented
192            ListBuffer<JCCompilationUnit> packageTrees = new ListBuffer<>();
193            parse(etable.getFilesToParse(), packageTrees, false);
194            modules.enter(packageTrees.toList(), null);
195
196            if (messager.hasErrors()) {
197                return null;
198            }
199
200            // Enter symbols for all files
201            toolEnv.notice("main.Building_tree");
202            javadocEnter.main(classTrees.toList().appendList(packageTrees));
203            etable.setClassDeclList(listClasses(classTrees.toList()));
204
205            etable.analyze();
206        } catch (CompletionFailure cf) {
207            throw new ToolException(ABNORMAL, cf.getMessage(), cf);
208        } catch (Abort abort) {
209            if (messager.hasErrors()) {
210                // presumably a message has been emitted, keep silent
211                throw new ToolException(ABNORMAL, "", abort);
212            } else {
213                String text = messager.getText("main.internal.error");
214                Throwable t = abort.getCause() == null ? abort : abort.getCause();
215                throw new ToolException(ABNORMAL, text, t);
216            }
217        }
218
219        if (messager.hasErrors())
220            return null;
221
222        toolEnv.docEnv = new DocEnvImpl(toolEnv, etable);
223        return toolEnv.docEnv;
224    }
225
226    /** Is the given string a valid package name? */
227    boolean isValidPackageName(String s) {
228        if (s.contains("/")) {
229            String[] a = s.split("/");
230            if (a.length == 2) {
231                 return isValidPackageName0(a[0]) && isValidPackageName0(a[1]);
232            }
233            return false;
234        }
235        return isValidPackageName0(s);
236    }
237
238    private boolean isValidPackageName0(String s) {
239        for (int index = s.indexOf('.') ; index != -1; index = s.indexOf('.')) {
240            if (!isValidClassName(s.substring(0, index))) {
241                return false;
242            }
243            s = s.substring(index + 1);
244        }
245        return isValidClassName(s);
246    }
247
248    private void parse(Iterable<? extends JavaFileObject> files, ListBuffer<JCCompilationUnit> trees,
249                       boolean trace) {
250        for (JavaFileObject fo: files) {
251            if (uniquefiles.add(fo)) { // ignore duplicates
252                if (trace)
253                    toolEnv.notice("main.Loading_source_file", fo.getName());
254                trees.append(parse(fo));
255            }
256        }
257    }
258
259    /** Are surrogates supported? */
260    final static boolean surrogatesSupported = surrogatesSupported();
261    private static boolean surrogatesSupported() {
262        try {
263            boolean b = Character.isHighSurrogate('a');
264            return true;
265        } catch (NoSuchMethodError ex) {
266            return false;
267        }
268    }
269
270    /**
271     * Return true if given file name is a valid class name
272     * (including "package-info").
273     * @param s the name of the class to check.
274     * @return true if given class name is a valid class name
275     * and false otherwise.
276     */
277    public static boolean isValidClassName(String s) {
278        if (s.length() < 1) return false;
279        if (s.equals("package-info")) return true;
280        if (surrogatesSupported) {
281            int cp = s.codePointAt(0);
282            if (!Character.isJavaIdentifierStart(cp))
283                return false;
284            for (int j = Character.charCount(cp); j < s.length(); j += Character.charCount(cp)) {
285                cp = s.codePointAt(j);
286                if (!Character.isJavaIdentifierPart(cp))
287                    return false;
288            }
289        } else {
290            if (!Character.isJavaIdentifierStart(s.charAt(0)))
291                return false;
292            for (int j = 1; j < s.length(); j++)
293                if (!Character.isJavaIdentifierPart(s.charAt(j)))
294                    return false;
295        }
296        return true;
297    }
298
299    /**
300     * From a list of top level trees, return the list of contained class definitions
301     */
302    List<JCClassDecl> listClasses(List<JCCompilationUnit> trees) {
303        List<JCClassDecl> result = new ArrayList<>();
304        for (JCCompilationUnit t : trees) {
305            for (JCTree def : t.defs) {
306                if (def.hasTag(JCTree.Tag.CLASSDEF))
307                    result.add((JCClassDecl)def);
308            }
309        }
310        return result;
311    }
312}
313