001/** 002 * Logback: the reliable, generic, fast and flexible logging framework. 003 * Copyright (C) 1999-2015, QOS.ch. All rights reserved. 004 * 005 * This program and the accompanying materials are dual-licensed under 006 * either the terms of the Eclipse Public License v1.0 as published by 007 * the Eclipse Foundation 008 * 009 * or (per the licensee's choosing) 010 * 011 * under the terms of the GNU Lesser General Public License version 2.1 012 * as published by the Free Software Foundation. 013 */ 014package ch.qos.logback.core.pattern.parser; 015 016import java.util.HashMap; 017import java.util.List; 018import java.util.Map; 019 020import ch.qos.logback.core.CoreConstants; 021import ch.qos.logback.core.pattern.Converter; 022import ch.qos.logback.core.pattern.FormatInfo; 023import ch.qos.logback.core.pattern.IdentityCompositeConverter; 024import ch.qos.logback.core.pattern.ReplacingCompositeConverter; 025import ch.qos.logback.core.pattern.util.IEscapeUtil; 026import ch.qos.logback.core.pattern.util.RegularEscapeUtil; 027import ch.qos.logback.core.spi.ContextAwareBase; 028import ch.qos.logback.core.spi.ScanException; 029 030// ~=lamda 031// E = TE|T 032 033// Left factorization 034// E = T(E|~) 035// Eopt = E|~ 036// replace E|~ with Eopt in E 037// E = TEopt 038 039// T = LITERAL | '%' C | '%' FORMAT_MODIFIER C 040// C = SIMPLE_KEYWORD OPTION | COMPOSITE_KEYWORD COMPOSITE 041// OPTION = {...} | ~ 042// COMPOSITE = E ')' OPTION 043 044public class Parser<E> extends ContextAwareBase { 045 046 public final static String MISSING_RIGHT_PARENTHESIS = CoreConstants.CODES_URL + "#missingRightParenthesis"; 047 public final static Map<String, String> DEFAULT_COMPOSITE_CONVERTER_MAP = new HashMap<String, String>(); 048 public final static String REPLACE_CONVERTER_WORD = "replace"; 049 static { 050 DEFAULT_COMPOSITE_CONVERTER_MAP.put(Token.BARE_COMPOSITE_KEYWORD_TOKEN.getValue().toString(), 051 IdentityCompositeConverter.class.getName()); 052 DEFAULT_COMPOSITE_CONVERTER_MAP.put(REPLACE_CONVERTER_WORD, ReplacingCompositeConverter.class.getName()); 053 } 054 055 final List<Token> tokenList; 056 int pointer = 0; 057 058 Parser(TokenStream ts) throws ScanException { 059 this.tokenList = ts.tokenize(); 060 } 061 062 public Parser(String pattern) throws ScanException { 063 this(pattern, new RegularEscapeUtil()); 064 } 065 066 public Parser(String pattern, IEscapeUtil escapeUtil) throws ScanException { 067 try { 068 TokenStream ts = new TokenStream(pattern, escapeUtil); 069 this.tokenList = ts.tokenize(); 070 } catch (IllegalArgumentException npe) { 071 throw new ScanException("Failed to initialize Parser", npe); 072 } 073 } 074 075 /** 076 * When the parsing step is done, the Node list can be transformed into a 077 * converter chain. 078 * 079 * @param top 080 * @param converterMap 081 * @return 082 */ 083 public Converter<E> compile(final Node top, Map<String, String> converterMap) { 084 Compiler<E> compiler = new Compiler<E>(top, converterMap); 085 compiler.setContext(context); 086 // compiler.setStatusManager(statusManager); 087 return compiler.compile(); 088 } 089 090 public Node parse() throws ScanException { 091 return E(); 092 } 093 094 // E = TEopt 095 Node E() throws ScanException { 096 Node t = T(); 097 if (t == null) { 098 return null; 099 } 100 Node eOpt = Eopt(); 101 if (eOpt != null) { 102 t.setNext(eOpt); 103 } 104 return t; 105 } 106 107 // Eopt = E|~ 108 Node Eopt() throws ScanException { 109 // System.out.println("in Eopt()"); 110 Token next = getCurentToken(); 111 // System.out.println("Current token is " + next); 112 if (next == null) { 113 return null; 114 } else { 115 return E(); 116 } 117 } 118 119 // T = LITERAL | '%' C | '%' FORMAT_MODIFIER C 120 Node T() throws ScanException { 121 Token t = getCurentToken(); 122 expectNotNull(t, "a LITERAL or '%'"); 123 124 switch (t.getType()) { 125 case Token.LITERAL: 126 advanceTokenPointer(); 127 return new Node(Node.LITERAL, t.getValue()); 128 case Token.PERCENT: 129 advanceTokenPointer(); 130 // System.out.println("% token found"); 131 FormatInfo fi; 132 Token u = getCurentToken(); 133 FormattingNode c; 134 expectNotNull(u, "a FORMAT_MODIFIER, SIMPLE_KEYWORD or COMPOUND_KEYWORD"); 135 if (u.getType() == Token.FORMAT_MODIFIER) { 136 fi = FormatInfo.valueOf((String) u.getValue()); 137 advanceTokenPointer(); 138 c = C(); 139 c.setFormatInfo(fi); 140 } else { 141 c = C(); 142 } 143 return c; 144 145 default: 146 return null; 147 148 } 149 150 } 151 152 FormattingNode C() throws ScanException { 153 Token t = getCurentToken(); 154 // System.out.println("in C()"); 155 // System.out.println("Current token is " + t); 156 expectNotNull(t, "a LEFT_PARENTHESIS or KEYWORD"); 157 int type = t.getType(); 158 switch (type) { 159 case Token.SIMPLE_KEYWORD: 160 return SINGLE(); 161 case Token.COMPOSITE_KEYWORD: 162 advanceTokenPointer(); 163 return COMPOSITE(t.getValue().toString()); 164 default: 165 throw new IllegalStateException("Unexpected token " + t); 166 } 167 } 168 169 FormattingNode SINGLE() throws ScanException { 170 // System.out.println("in SINGLE()"); 171 Token t = getNextToken(); 172 // System.out.println("==" + t); 173 SimpleKeywordNode keywordNode = new SimpleKeywordNode(t.getValue()); 174 175 Token ot = getCurentToken(); 176 if (ot != null && ot.getType() == Token.OPTION) { 177 List<String> optionList = ot.getOptionsList(); 178 keywordNode.setOptions(optionList); 179 advanceTokenPointer(); 180 } 181 return keywordNode; 182 } 183 184 FormattingNode COMPOSITE(String keyword) throws ScanException { 185 CompositeNode compositeNode = new CompositeNode(keyword); 186 187 Node childNode = E(); 188 compositeNode.setChildNode(childNode); 189 190 Token t = getNextToken(); 191 192 if (t == null || t.getType() != Token.RIGHT_PARENTHESIS) { 193 String msg = "Expecting RIGHT_PARENTHESIS token but got " + t; 194 addError(msg); 195 addError("See also " + MISSING_RIGHT_PARENTHESIS); 196 throw new ScanException(msg); 197 } 198 Token ot = getCurentToken(); 199 if (ot != null && ot.getType() == Token.OPTION) { 200 List<String> optionList = ot.getOptionsList(); 201 compositeNode.setOptions(optionList); 202 advanceTokenPointer(); 203 } 204 return compositeNode; 205 } 206 207 Token getNextToken() { 208 if (pointer < tokenList.size()) { 209 return (Token) tokenList.get(pointer++); 210 } 211 return null; 212 } 213 214 Token getCurentToken() { 215 if (pointer < tokenList.size()) { 216 return (Token) tokenList.get(pointer); 217 } 218 return null; 219 } 220 221 void advanceTokenPointer() { 222 pointer++; 223 } 224 225 void expectNotNull(Token t, String expected) { 226 if (t == null) { 227 throw new IllegalStateException("All tokens consumed but was expecting " + expected); 228 } 229 } 230 231 // public void setStatusManager(StatusManager statusManager) { 232 // this.statusManager = statusManager; 233 // } 234}