Remove deprecated Yin/YangStatementSourceImpl
[yangtools.git] / yang / yang-parser-impl / src / main / java / org / opendaylight / yangtools / yang / parser / rfc6020 / repo / YangStatementStreamSource.java
1 /*
2  * Copyright (c) 2017 Pantheon Technologies, s.r.o. and others.  All rights reserved.
3  *
4  * This program and the accompanying materials are made available under the
5  * terms of the Eclipse Public License v1.0 which accompanies this distribution,
6  * and is available at http://www.eclipse.org/legal/epl-v10.html
7  */
8 package org.opendaylight.yangtools.yang.parser.rfc6020.repo;
9
10 import com.google.common.annotations.Beta;
11 import com.google.common.base.MoreObjects;
12 import com.google.common.base.Preconditions;
13 import com.google.common.collect.ImmutableList;
14 import java.io.IOException;
15 import java.io.InputStream;
16 import org.antlr.v4.runtime.CharStreams;
17 import org.antlr.v4.runtime.CommonTokenStream;
18 import org.antlr.v4.runtime.ParserRuleContext;
19 import org.antlr.v4.runtime.tree.ErrorNode;
20 import org.antlr.v4.runtime.tree.ParseTreeListener;
21 import org.antlr.v4.runtime.tree.ParseTreeWalker;
22 import org.antlr.v4.runtime.tree.TerminalNode;
23 import org.opendaylight.yangtools.antlrv4.code.gen.YangStatementLexer;
24 import org.opendaylight.yangtools.antlrv4.code.gen.YangStatementParser;
25 import org.opendaylight.yangtools.antlrv4.code.gen.YangStatementParser.StatementContext;
26 import org.opendaylight.yangtools.concepts.Identifiable;
27 import org.opendaylight.yangtools.yang.common.YangVersion;
28 import org.opendaylight.yangtools.yang.model.parser.api.YangSyntaxErrorException;
29 import org.opendaylight.yangtools.yang.model.repo.api.SourceIdentifier;
30 import org.opendaylight.yangtools.yang.model.repo.api.YangTextSchemaSource;
31 import org.opendaylight.yangtools.yang.parser.impl.YangStatementParserListenerImpl;
32 import org.opendaylight.yangtools.yang.parser.spi.source.PrefixToModule;
33 import org.opendaylight.yangtools.yang.parser.spi.source.QNameToStatementDefinition;
34 import org.opendaylight.yangtools.yang.parser.spi.source.StatementStreamSource;
35 import org.opendaylight.yangtools.yang.parser.spi.source.StatementWriter;
36
37 /**
38  * This class represents implementation of StatementStreamSource in order to emit YANG statements using supplied
39  * StatementWriter.
40  *
41  * @author Robert Varga
42  */
43 @Beta
44 public final class YangStatementStreamSource implements Identifiable<SourceIdentifier>, StatementStreamSource {
45     private static final ParseTreeListener MAKE_IMMUTABLE_LISTENER = new ParseTreeListener() {
46         @Override
47         public void enterEveryRule(final ParserRuleContext ctx) {
48             // No-op
49         }
50
51         @Override
52         public void exitEveryRule(final ParserRuleContext ctx) {
53             ctx.children = ctx.children == null ? ImmutableList.of() : ImmutableList.copyOf(ctx.children);
54         }
55
56         @Override
57         public void visitTerminal(final TerminalNode node) {
58             // No-op
59         }
60
61         @Override
62         public void visitErrorNode(final ErrorNode node) {
63             // No-op
64         }
65     };
66
67     private final YangStatementParserListenerImpl yangStatementModelParser;
68     private final SourceIdentifier identifier;
69     private final StatementContext context;
70
71     private YangStatementStreamSource(final SourceIdentifier identifier, final YangStatementParserListenerImpl parser,
72             final StatementContext context) {
73         this.identifier = Preconditions.checkNotNull(identifier);
74         this.yangStatementModelParser = Preconditions.checkNotNull(parser);
75         this.context = Preconditions.checkNotNull(context);
76     }
77
78     public static YangStatementStreamSource create(final YangTextSchemaSource source) throws IOException,
79             YangSyntaxErrorException {
80         final StatementContext context;
81         try (final InputStream stream = source.openStream()) {
82             context = parseYangSource(stream);
83         }
84
85         final String sourceName = source.getSymbolicName().orElse(null);
86         final YangStatementParserListenerImpl parser = new YangStatementParserListenerImpl(sourceName);
87         return new YangStatementStreamSource(source.getIdentifier(), parser, context);
88     }
89
90     public static YangStatementStreamSource create(final SourceIdentifier identifier, final StatementContext context,
91         final String symbolicName) {
92         return new YangStatementStreamSource(identifier, new YangStatementParserListenerImpl(symbolicName), context);
93     }
94
95     @Override
96     public void writePreLinkage(final StatementWriter writer, final QNameToStatementDefinition stmtDef) {
97         yangStatementModelParser.setAttributes(writer, stmtDef);
98         ParseTreeWalker.DEFAULT.walk(yangStatementModelParser, context);
99     }
100
101     @Override
102     public void writeLinkage(final StatementWriter writer, final QNameToStatementDefinition stmtDef,
103             final PrefixToModule preLinkagePrefixes) {
104         writeLinkage(writer, stmtDef, preLinkagePrefixes, YangVersion.VERSION_1);
105     }
106
107     @Override
108     public void writeLinkage(final StatementWriter writer, final QNameToStatementDefinition stmtDef,
109             final PrefixToModule preLinkagePrefixes, final YangVersion yangVersion) {
110         yangStatementModelParser.setAttributes(writer, stmtDef, preLinkagePrefixes, yangVersion);
111         ParseTreeWalker.DEFAULT.walk(yangStatementModelParser, context);
112     }
113
114     @Override
115     public void writeLinkageAndStatementDefinitions(final StatementWriter writer,
116             final QNameToStatementDefinition stmtDef, final PrefixToModule prefixes) {
117         writeLinkageAndStatementDefinitions(writer, stmtDef, prefixes, YangVersion.VERSION_1);
118     }
119
120     @Override
121     public void writeLinkageAndStatementDefinitions(final StatementWriter writer,
122             final QNameToStatementDefinition stmtDef, final PrefixToModule prefixes, final YangVersion yangVersion) {
123         yangStatementModelParser.setAttributes(writer, stmtDef, prefixes, yangVersion);
124         ParseTreeWalker.DEFAULT.walk(yangStatementModelParser, context);
125     }
126
127     @Override
128     public void writeFull(final StatementWriter writer, final QNameToStatementDefinition stmtDef,
129             final PrefixToModule prefixes) {
130         writeFull(writer, stmtDef, prefixes, YangVersion.VERSION_1);
131     }
132
133     @Override
134     public void writeFull(final StatementWriter writer, final QNameToStatementDefinition stmtDef,
135             final PrefixToModule prefixes, final YangVersion yangVersion) {
136         yangStatementModelParser.setAttributes(writer, stmtDef, prefixes, yangVersion);
137         ParseTreeWalker.DEFAULT.walk(yangStatementModelParser, context);
138     }
139
140     @Override
141     public SourceIdentifier getIdentifier() {
142         return identifier;
143     }
144
145     public ParserRuleContext getYangAST() {
146         return context;
147     }
148
149     /**
150      * @deprecated Provided for migration purposes only. Do not use.
151      */
152     @Deprecated
153     public static StatementContext parseYangSource(final InputStream stream) throws IOException,
154             YangSyntaxErrorException {
155         final YangStatementLexer lexer = new YangStatementLexer(CharStreams.fromStream(stream));
156         final CommonTokenStream tokens = new CommonTokenStream(lexer);
157         final YangStatementParser parser = new YangStatementParser(tokens);
158         //disconnect from console error output
159         parser.removeErrorListeners();
160
161         final YangErrorListener errorListener = new YangErrorListener();
162         parser.addErrorListener(errorListener);
163
164         final StatementContext result = parser.statement();
165         errorListener.validate();
166
167         // Walk the resulting tree and replace each children with an immutable list, lowering memory requirements
168         // and making sure the resulting tree will not get accidentally modified. An alternative would be to use
169         // org.antlr.v4.runtime.Parser.TrimToSizeListener, but that does not make the tree immutable.
170         ParseTreeWalker.DEFAULT.walk(MAKE_IMMUTABLE_LISTENER, result);
171
172         return result;
173     }
174
175     @Override
176     public String toString() {
177         return MoreObjects.toStringHelper(this).add("identifier", getIdentifier()).toString();
178     }
179 }