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

org.spf4j.avro.Configs Maven / Gradle / Ivy

/*
 * Copyright (c) 2001-2017, Zoltan Farkas All Rights Reserved.
 *
 * This library 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.
 *
 * This library 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 General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
 *
 * Additionally licensed with:
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.spf4j.avro;

import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.node.JsonNodeFactory;
import com.fasterxml.jackson.databind.node.ObjectNode;
import com.google.common.net.MediaType;
import edu.umd.cs.findbugs.annotations.CreatesObligation;
import edu.umd.cs.findbugs.annotations.SuppressFBWarnings;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.Reader;
import java.io.StringReader;
import java.util.Arrays;
import java.util.Iterator;
import java.util.List;
import java.util.Map.Entry;
import javax.annotation.Nullable;
import javax.annotation.WillNotClose;
import org.apache.avro.Schema;
import org.apache.avro.Schema.Field;
import org.apache.avro.io.DatumReader;
import org.apache.avro.io.Decoder;
import org.apache.avro.specific.SpecificData;
import org.apache.avro.specific.SpecificDatumReader;
import org.spf4j.base.CharSequences;
import org.spf4j.base.Json;
import org.spf4j.base.Pair;
import org.spf4j.io.PushbackReader;

/**
 * Utilities to read Objects from configuration files enforced via json/yaml files.
 *
 * @author Zoltan Farkas
 */
public final class Configs {

  private Configs() { }

  @Nullable
  private static MediaType getConfigMediaType(final CharSequence seq) {
    if (!CharSequences.startsWith(seq, "Content-Type:", 0)) {
      return null;
    }
    int length = seq.length();
    int indexOf = CharSequences.indexOf(seq, 0, length, ':');
    int next = indexOf + 1;
    int endMt = CharSequences.indexOf(seq, next, length, '\n');
    if (endMt < 0) {
      endMt = length;
    }
    return MediaType.parse(seq.subSequence(next, endMt).toString());
  }

  @Nullable
  public static  T read(final Class type, final SchemaResolver schemaResolver, final Reader... reader)
          throws IOException {
    if (reader.length == 0) {
      return null;
    }
    SpecificData sd = SpecificData.get();
    Schema rSchema = sd.getSchema(type);

    int l = reader.length - 1;
    ConfigHeader header = parseHeader(reader[l], rSchema, schemaResolver);
    JsonNode node = readTree(header.getReader(), header.getMediaType());
    Schema wSchema = header.getwSchema();
    for (int i = l - 1; i >= 0; i--) {
      header = parseHeader(reader[i], rSchema, schemaResolver);
      JsonNode newNode = readTree(header.getReader(), header.getMediaType());
      wSchema = header.getwSchema();
      node = override(newNode, node, wSchema);
    }
    DatumReader dr = new SpecificDatumReader<>(wSchema, rSchema);
    Adapter adapter = AvroCompatUtils.getAdapter();
    Decoder decoder = adapter.getJsonDecoder(wSchema, node.traverse());
    return dr.read(null, decoder);

  }

  private static JsonNode readTree(final Reader reader, final MediaType mt) throws IOException {
    if ("application".equals(mt.type()) && "json".equals(mt.subtype())) {
      return Json.MAPPER.readTree(reader);
    } else if ("text".equals(mt.type()) && "yaml".equals(mt.subtype())) {
      return Yaml.MAPPER.readTree(reader);
    } else {
      throw new IllegalArgumentException("Unsupported media type " + mt);
    }
  }

  @Nullable
  public static JsonNode override(@Nullable final JsonNode target,
          @Nullable final JsonNode defaults,
          final Schema targetSchema) {
    if (target == null) {
      return defaults;
    }
    Schema.Type type = targetSchema.getType();
    switch (type) {
      case RECORD:
        ObjectNode newNode = JsonNodeFactory.instance.objectNode();
        if (!(target instanceof ObjectNode)) {
          throw new IllegalArgumentException("Target must be of type Object/record, not: " + target);
        }
        newNode.setAll((ObjectNode) target);
        for (Field field : targetSchema.getFields()) {
           String fieldName  = field.name();
           JsonNode d = defaults.get(fieldName);
           if (d == null) {
             for (String alias : field.aliases()) {
               d = defaults.get(alias);
               if (d != null) {
                 break;
               }
             }
           }
           if (d == null) {
             continue;
           }
           JsonNode targetField = target.get(fieldName);
           if (targetField == null) {
             newNode.set(fieldName, d);
           } else {
             newNode.set(fieldName, override(targetField, d,
                     targetSchema.getField(fieldName).schema()));
           }
        }
        return newNode;
      case MAP:
        newNode = JsonNodeFactory.instance.objectNode();
        newNode.setAll((ObjectNode) target);
        for (Entry field : (Iterable>) defaults::fields) {
           String fieldName  = field.getKey();
           JsonNode targetField = target.get(fieldName);
           if (targetField == null) {
             newNode.set(fieldName, field.getValue());
           } else {
             newNode.set(fieldName, override(targetField, field.getValue(),
                     targetSchema.getValueType()));
           }
        }
        return newNode;
      case UNION:

      case ARRAY:
      case BOOLEAN:
      case BYTES:
      case DOUBLE:
      case ENUM:
      case FIXED:
      case FLOAT:
      case INT:
      case LONG:
      case NULL:
      case STRING:
        return target;
      default:
        throw new UnsupportedOperationException("Unsupported type: " + type);
    }
  }

  public static Pair getMatchingSchema(final JsonNode node, final List schemas) {
    if (node.isObject()) {
       Iterator> it = node.fields();
       if (it.hasNext()) {
         Entry entry = it.next();
         if (it.hasNext()) { // must be a record schema.
           Schema theSchema = getOneSchema(schemas, Schema.Type.RECORD, Schema.Type.MAP);
           return Pair.of(theSchema, node);
         } else {
           Schema s = getOneSchema(schemas, entry.getKey());
           if (s == null) {
              Schema theSchema = getOneSchema(schemas, Schema.Type.RECORD, Schema.Type.MAP);
              return Pair.of(theSchema, node);
           } else {
              return Pair.of(s, entry.getValue());
           }
         }
       } else {
           Schema theSchema = getOneSchema(schemas, Schema.Type.RECORD, Schema.Type.MAP);
           return Pair.of(theSchema, node);
       }
    } else if (node.isNull()) {
      return Pair.of(getOneSchema(schemas, Schema.Type.NULL), node);
    } else if (node.isArray()) {
      return Pair.of(getOneSchema(schemas, Schema.Type.ARRAY), node);
    } else if (node.isTextual()) {
      return Pair.of(getOneSchema(schemas, Schema.Type.STRING,  Schema.Type.BYTES, Schema.Type.FIXED), node);
    } else if (node.isBoolean()) {
      return Pair.of(getOneSchema(schemas, Schema.Type.BOOLEAN), node);
    } else if (node.isNumber()) {
      // direct encoded types not really handled here
      return Pair.of(getOneSchema(schemas,
              Schema.Type.DOUBLE,  Schema.Type.FLOAT, Schema.Type.INT, Schema.Type.LONG), node);
    } else {
      throw new IllegalStateException("IUnsupported node type " + node);
    }
  }

  @Nullable
  private static Schema getOneSchema(final List schemas, final String name) {
    for (Schema s : schemas) {
      if (s.getFullName().equals(name)) {
        return s;
      }
    }
   return null;
  }

  private static Schema getOneSchema(final List schemas, final Schema.Type type) {
    Schema result = null;
    for (Schema s : schemas) {
      if (s.getType() == type) {
        if (result == null) {
          result = s;
        } else {
          throw new IllegalArgumentException("Must have only one " + type + " schema in " + schemas);
        }
      }
    }
    if (result == null) {
      throw new IllegalArgumentException("No record " + type + "  in " + schemas);
    } else {
      return result;
    }
  }

  private static Schema getOneSchema(final List schemas, final Schema.Type... types) {
    Schema result = null;
    for (Schema s : schemas) {
      for (Schema.Type t : types) {
        if (s.getType() == t) {
          if (result == null) {
            result = s;
          } else {
            throw new IllegalArgumentException("Must have only one schema of types: "
                    + Arrays.toString(types) + " in " + schemas);
          }
        }
      }
    }
    if (result == null) {
      throw new IllegalArgumentException("No schema of type: " + Arrays.toString(types) + " in " + schemas);
    } else {
      return result;
    }
  }

  @Nullable
  public static  T read(final Class type, final Reader reader)
          throws IOException {
    return read(type, SchemaResolver.NONE, reader);
  }


  private static final class ConfigHeader {

    private final Schema wSchema;

    private final Reader reader;

    private final MediaType mediaType;

    ConfigHeader(final Schema wSchema, final Reader reader, final MediaType mediaType) {
      this.wSchema = wSchema;
      this.reader = reader;
      this.mediaType = mediaType;
    }

    @SuppressFBWarnings("EI_EXPOSE_REP")
    public Schema getwSchema() {
      return wSchema;
    }

    /**
     * The reader with the content of the config. (header component was consumed)
     * @return
     */
    @CreatesObligation
    public Reader getReader() {
      return reader;
    }

    public MediaType getMediaType() {
      return mediaType;
    }

    @Override
    public String toString() {
      return "ConfigHeader{" + "wSchema=" + wSchema + ", reader=" + reader + ", mediaType=" + mediaType + '}';
    }

  }

  private static ConfigHeader parseHeader(@WillNotClose final Reader reader, final Schema rSchema,
          final SchemaResolver schemaResolver) throws IOException {
    Schema wSchema;
    MediaType mt;
    PushbackReader pbr = new PushbackReader(reader);
    int firstChar = pbr.read();
    if (firstChar < 0) {
      mt = MediaType.JSON_UTF_8;
      Reader r = pbr;
      wSchema = rSchema;
      return new ConfigHeader(wSchema, r, mt);
    }
    BufferedReader content;
    Adapter adapter = AvroCompatUtils.getAdapter();
    if (firstChar == '#') {
      content = new BufferedReader(pbr);
        String header = content.readLine();
        if (header == null) {
            mt = MediaType.JSON_UTF_8;
            pbr.unread(firstChar);
            Reader r = pbr;
            wSchema = rSchema;
            return new ConfigHeader(wSchema, r, mt);
        }
        mt = getConfigMediaType(header);
        if (mt != null) {
          List schemaStrList = mt.parameters().get("avsc");
          if (schemaStrList.isEmpty()) {
            wSchema = rSchema;
          } else {
            wSchema = adapter.parseSchema(new StringReader(schemaStrList.get(0)), true, schemaResolver);
          }
        } else {
           mt = MediaType.create(header, header);
           wSchema = rSchema;
        }
    } else {
      mt = MediaType.JSON_UTF_8;
      pbr.unread(firstChar);
      content = new BufferedReader(pbr);
      wSchema = rSchema;
    }
    return new ConfigHeader(wSchema, content, mt);
  }


  @Nullable
  public static  T read(final Class type, final SchemaResolver schemaResolver, final Reader reader)
          throws IOException {
    SpecificData sd = SpecificData.get();
    Schema rSchema = sd.getSchema(type);
    ConfigHeader header = parseHeader(reader, rSchema, schemaResolver);
    Schema wSchema = header.getwSchema();
    Reader content = header.getReader();
    MediaType mt = header.getMediaType();
    DatumReader dr = new SpecificDatumReader<>(wSchema, rSchema);
    Decoder decoder;
    Adapter adapter = AvroCompatUtils.getAdapter();
    if ("application".equals(mt.type()) && "json".equals(mt.subtype())) {
      decoder = adapter.getJsonDecoder(wSchema, content);
    } else if ("text".equals(mt.type()) && "yaml".equals(mt.subtype())) {
      decoder = adapter.getYamlDecoder(wSchema, content);
    } else {
      throw new IllegalArgumentException("Unsupported media type " + mt);
    }
    return dr.read(null, decoder);
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy