All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.avaje.ebeaninternal.server.text.json.ReadJsonContext Maven / Gradle / Ivy

There is a newer version: 2.8.1
Show newest version
/**
 * Copyright (C) 2009 Authors
 * 
 * This file is part of Ebean.
 * 
 * Ebean is free software; you can redistribute it and/or modify it 
 * under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation; either version 2.1 of the License, or
 * (at your option) any later version.
 *  
 * Ebean is distributed in the hope that it will be useful, but 
 * WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU Lesser General Public License for more details.
 * 
 * You should have received a copy of the GNU Lesser General Public License
 * along with Ebean; if not, write to the Free Software Foundation, Inc.,
 * 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA  
 */
package com.avaje.ebeaninternal.server.text.json;

import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.util.HashSet;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.Set;

import com.avaje.ebean.bean.EntityBean;
import com.avaje.ebean.bean.EntityBeanIntercept;
import com.avaje.ebean.text.TextException;
import com.avaje.ebean.text.json.JsonElement;
import com.avaje.ebean.text.json.JsonReadBeanVisitor;
import com.avaje.ebean.text.json.JsonReadOptions;
import com.avaje.ebean.text.json.JsonValueAdapter;
import com.avaje.ebeaninternal.server.deploy.BeanDescriptor;
import com.avaje.ebeaninternal.server.util.ArrayStack;

public class ReadJsonContext {
    
    private final ReadJsonSource src;
    
    private final Map> visitorMap;

    private final JsonValueAdapter valueAdapter;

    private final PathStack pathStack;

    private final ArrayStack beanState;
    private ReadBeanState currentState;

    //private char prevChar;
    private char tokenStart;
    private String tokenKey;
    
    public ReadJsonContext(ReadJsonSource src, JsonValueAdapter dfltValueAdapter, JsonReadOptions options) {

        this.src = src;
        this.beanState = new ArrayStack();
        if (options == null){
            this.valueAdapter = dfltValueAdapter;
            this.visitorMap = null;
            this.pathStack = null;
        } else {
            this.valueAdapter = getValueAdapter(dfltValueAdapter, options.getValueAdapter());
            this.visitorMap = options.getVisitorMap();
            this.pathStack = (visitorMap == null || visitorMap.isEmpty()) ? null : new PathStack();
        }
    }
    
    private JsonValueAdapter getValueAdapter(JsonValueAdapter dfltValueAdapter, JsonValueAdapter valueAdapter) {
        return valueAdapter == null ? dfltValueAdapter : valueAdapter;
    }

    public JsonValueAdapter getValueAdapter() {
        return valueAdapter;
    }

    public char getToken() {
        return tokenStart;
    }

    public String getTokenKey() {
        return tokenKey;
    }
    
    public boolean isTokenKey() {
        return '\"' == tokenStart;
    }

    public boolean isTokenObjectEnd() {
        return '}' == tokenStart;
    }
        
    public boolean readObjectBegin() {
        readNextToken();
        if ('{' == tokenStart){
            return true;
        } else if ('n' == tokenStart) {
            return false;
        } else if (']' == tokenStart) {
            // an empty array
            return false;
        }
        throw new RuntimeException("Expected object begin at "+src.getErrorHelp());
    }
 
    public boolean readKeyNext() {
        readNextToken();
        if ('\"' == tokenStart){
            return true;
        } else if ('}' == tokenStart) {
            return false;
        }
        throw new RuntimeException("Expected '\"' or '}' at "+src.getErrorHelp());        
    }
    
    public boolean readValueNext() {
        readNextToken();
        if (',' == tokenStart){
            return true;
        } else if ('}' == tokenStart) {
            return false;
        }
        throw new RuntimeException("Expected ',' or '}' at "+src.getErrorHelp()+" but got "+tokenStart);        
    }
    
    public boolean readArrayBegin() {
        readNextToken();
        if ('[' == tokenStart){
            return true;
        } else if ('n' == tokenStart) {
            return false;
        }
        throw new RuntimeException("Expected array begin at "+src.getErrorHelp());
    }
    
    public boolean readArrayNext() {
        readNextToken();
        if (',' == tokenStart){
            return true;
        }
        if (']' == tokenStart){
            return false;
        }
        throw new RuntimeException("Expected ',' or ']' at "+src.getErrorHelp());
    }
    
    public String readScalarValue() {
        
        ignoreWhiteSpace();
        
        char prevChar = src.nextChar("EOF reading scalarValue?");
        if ('"' == prevChar){
            return readQuotedValue();
        } else {
            return readUnquotedValue(prevChar);
        }
    }
   
    
    
    public void readNextToken() {
        
        ignoreWhiteSpace();
        
        tokenStart = src.nextChar("EOF finding next token");
        switch (tokenStart) {
        case '"': 
            internalReadKey();
            break;
        case '{': break;
        case '}': break;
        case '[': break; // not expected
        case ']': break; // not expected
        case ',': break; // not expected
        case ':': break; // not expected
        case 'n': 
            internalReadNull();
            break; // not expected

        default:
            throw new RuntimeException("Unexpected tokenStart["+tokenStart+"] "+src.getErrorHelp());
        }
        
    }
    
    
    protected String readQuotedValue() {
        
        boolean escape = false;
        StringBuilder sb = new StringBuilder();

        do {
            char ch = src.nextChar("EOF reading quoted value");
            if (escape) {
                // in escape mode so just append the character
                sb.append(ch);
                
            } else {
                switch (ch) {
                case '\\':
                    // put into 'escape' mode for next character
                    escape = true;
                    break;
                case '"':
                    return sb.toString();

                default:
                    sb.append(ch);
                }
            }
        } while (true);
    }

    protected String readUnquotedValue(char c) {
        String v = readUnquotedValueRaw(c);
        if ("null".equals(v)){
            return null;
        } else {
            return v;
        }
    }
    
    private String readUnquotedValueRaw(char c) {

        StringBuilder sb = new StringBuilder();
        sb.append(c);
        
        do {
            tokenStart = src.nextChar("EOF reading unquoted value");
            switch (tokenStart) {
            case ',':
                src.back();
                return sb.toString();
                
            case '}':
                src.back();
                return sb.toString();
                
            case ' ':
                return sb.toString();

            case '\t':
                return sb.toString();

            case '\r':
                return sb.toString();
            
            case '\n':
                return sb.toString();

            default:
                sb.append(tokenStart);
            }
            
        } while (true);
        
    }

    private void internalReadNull() {
        
        StringBuilder sb = new StringBuilder(4);
        sb.append(tokenStart);
        for (int i = 0; i < 3; i++) {
            char c = src.nextChar("EOF reading null ");
            sb.append(c);
        }
        if (!"null".equals(sb.toString())){
            throw new TextException("Expected 'null' but got "+sb.toString()+" "+src.getErrorHelp());
        }
    }
    
    private void internalReadKey() {
        StringBuilder sb = new StringBuilder();
        do {
            char c = src.nextChar("EOF reading key");
            if ('\"' == c){
                tokenKey = sb.toString();
                break;
            } else {
                sb.append(c);
            }
        } while (true);
        
        ignoreWhiteSpace();
        
        char c = src.nextChar("EOF reading ':'");
        if (':' != c){
            throw new TextException("Expected to find colon after key at "+(src.pos()-1)+" but found ["+c+"]"+src.getErrorHelp());
        }
    }
    
    protected void ignoreWhiteSpace() {
        src.ignoreWhiteSpace();
    }
        
    public void pushBean(Object bean, String path, BeanDescriptor beanDescriptor){
        currentState = new ReadBeanState(bean, beanDescriptor);
        beanState.push(currentState);
        if (pathStack != null){
            pathStack.pushPathKey(path);
        }
    }
    
    public ReadBeanState popBeanState() {
        if (pathStack != null){
            String path = pathStack.peekWithNull();
            JsonReadBeanVisitor beanVisitor = visitorMap.get(path);
            if (beanVisitor != null){
                currentState.visit(beanVisitor);
            }
            pathStack.pop();
        }
        
        // return the current ReadBeanState as we can't call setLoadedState()
        // yet. We might bind master/detail beans together via mappedBy property
        // so wait until after that before calling ReadBeanStatesetLoadedState();
        ReadBeanState s = currentState;
        
        beanState.pop();
        currentState = beanState.peekWithNull();
        return s;
    }
    
    public void setProperty(String propertyName){
        currentState.setLoaded(propertyName);
    }
        
    /**
     * Got a key that doesn't map to a known property so read the json value
     * which could be json primitive, object or array.
     * 

* Provide these values to a JsonReadBeanVisitor if registered. *

*/ public JsonElement readUnmappedJson(String key) { ReadJsonRawReader rawReader = new ReadJsonRawReader(this); JsonElement rawJsonValue = rawReader.readUnknownValue(); if (visitorMap != null){ currentState.addUnmappedJson(key, rawJsonValue); } return rawJsonValue; } protected char nextChar() { tokenStart = src.nextChar("EOF getting nextChar for raw json"); return tokenStart; } public static class ReadBeanState implements PropertyChangeListener { private final Object bean; private final BeanDescriptor beanDescriptor; private final EntityBeanIntercept ebi; private final Set loadedProps; private Map unmapped; private ReadBeanState(Object bean, BeanDescriptor beanDescriptor) { this.bean = bean; this.beanDescriptor = beanDescriptor; if (bean instanceof EntityBean){ this.ebi = ((EntityBean)bean)._ebean_getIntercept(); this.loadedProps = new HashSet(); } else { this.ebi = null; this.loadedProps = null; } } public String toString(){ return bean.getClass().getSimpleName()+" loaded:"+loadedProps; } /** * Add a loaded/set property to the set of loadedProps. */ public void setLoaded(String propertyName){ if (ebi != null){ loadedProps.add(propertyName); } } private void addUnmappedJson(String key, JsonElement value){ if (unmapped == null){ unmapped = new LinkedHashMap(); } unmapped.put(key, value); } @SuppressWarnings("unchecked") private void visit(JsonReadBeanVisitor beanVisitor) { // listen for property change events so that // we can update the loadedProps if necessary if (ebi != null){ ebi.addPropertyChangeListener(this); } beanVisitor.visit((T)bean, unmapped); if (ebi != null){ ebi.removePropertyChangeListener(this); } } public void setLoadedState(){ if (ebi != null){ // takes into account reference beans beanDescriptor.setLoadedProps(ebi, loadedProps); } } public void propertyChange(PropertyChangeEvent evt) { String propName = evt.getPropertyName(); loadedProps.add(propName); } public Object getBean() { return bean; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy