-
Notifications
You must be signed in to change notification settings - Fork 1.7k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
AVRO-3666: Separate parsing from Schema class
This allows using pluggable parser implementations, allowing multiple formats to be parsed with the same code.
- Loading branch information
Showing
13 changed files
with
1,081 additions
and
26 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
74 changes: 74 additions & 0 deletions
74
lang/java/avro/src/main/java/org/apache/avro/FormattedSchemaParser.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,74 @@ | ||
/* | ||
* Licensed to the Apache Software Foundation (ASF) under one | ||
* or more contributor license agreements. See the NOTICE file | ||
* distributed with this work for additional information | ||
* regarding copyright ownership. The ASF licenses this file | ||
* to you 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 | ||
* | ||
* https://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.apache.avro; | ||
|
||
import java.io.IOException; | ||
import java.net.URI; | ||
import java.util.Collection; | ||
|
||
/** | ||
* Schema parser for a specific schema format. | ||
* | ||
* <p> | ||
* The {@link SchemaParser} class uses this interface, supporting text based | ||
* schema sources. | ||
* </p> | ||
* | ||
* <h2>Note to implementers:</h2> | ||
* | ||
* <p> | ||
* Implementations are located using a {@link java.util.ServiceLoader}. See that | ||
* class for details. | ||
* </p> | ||
* | ||
* <p> | ||
* You can expect that schemas being read are invalid, so you are encouraged to | ||
* return {@code null} upon parsing failure where the input clearly doesn't make | ||
* sense (e.g., reading "/**" when expecting JSON). If the input is likely in | ||
* the correct format, but invalid, throw a {@link SchemaParseException} | ||
* instead. | ||
* </p> | ||
* | ||
* <p> | ||
* Note that throwing anything other than a {@code SchemaParseException} will | ||
* abort the parsing process, so reserve that for rethrowing exceptions. | ||
* </p> | ||
* | ||
* @see java.util.ServiceLoader | ||
*/ | ||
public interface FormattedSchemaParser { | ||
/** | ||
* Parse a schema from a text based source. Can use the base location of the | ||
* schema (e.g., the directory where the schema file lives) if available. | ||
* | ||
* <p> | ||
* Implementations should add all named schemas they parse to the collection. | ||
* </p> | ||
* | ||
* @param types a mutable collection of known types; parsed named | ||
* schemata will be added | ||
* @param baseUri the base location of the schema, or {@code null} if | ||
* not known | ||
* @param formattedSchema the schema as text | ||
* @return the parsed schema, or {@code null} if the format is not supported | ||
* @throws IOException when the schema cannot be read | ||
* @throws SchemaParseException when the schema cannot be parsed | ||
*/ | ||
Schema parse(Collection<Schema> types, URI baseUri, CharSequence formattedSchema) | ||
throws IOException, SchemaParseException; | ||
} |
90 changes: 90 additions & 0 deletions
90
lang/java/avro/src/main/java/org/apache/avro/JsonSchemaParser.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,90 @@ | ||
/* | ||
* Licensed to the Apache Software Foundation (ASF) under one | ||
* or more contributor license agreements. See the NOTICE file | ||
* distributed with this work for additional information | ||
* regarding copyright ownership. The ASF licenses this file | ||
* to you 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 | ||
* | ||
* https://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.apache.avro; | ||
|
||
import java.io.IOException; | ||
import java.net.URI; | ||
import java.util.ArrayList; | ||
import java.util.Collection; | ||
|
||
/** | ||
* Schema parser for JSON formatted schemata. This initial implementation simply | ||
* delegates to the {@link Schema.Parser} class, though it should be refactored | ||
* out of there. | ||
* | ||
* <p> | ||
* Note: this class is intentionally not available via the Java | ||
* {@link java.util.ServiceLoader}, as its use is hardcoded as fallback when no | ||
* service exists. This enables users to reliably override the standard JSON | ||
* parser as well. | ||
* </p> | ||
*/ | ||
public class JsonSchemaParser implements FormattedSchemaParser { | ||
/** | ||
* <p> | ||
* Parse a schema written in the internal (JSON) format without any validations. | ||
* </p> | ||
* | ||
* <p> | ||
* Using this method is only safe if used to parse a write schema (i.e., a | ||
* schema used to read Avro data). Other usages, for example by generated Avro | ||
* code, can cause interoperability problems. | ||
* </p> | ||
* | ||
* <p> | ||
* Use with care and sufficient testing! | ||
* </p> | ||
* | ||
* @param fragments one or more strings making up the schema (some schemata | ||
* exceed the compiler limits) | ||
* @return the parsed schema | ||
*/ | ||
public static Schema parseInternal(String... fragments) { | ||
StringBuilder buffer = new StringBuilder(); | ||
for (String fragment : fragments) { | ||
buffer.append(fragment); | ||
} | ||
return new JsonSchemaParser().parse(new ArrayList<>(), buffer, true); | ||
} | ||
|
||
@Override | ||
public Schema parse(Collection<Schema> schemas, URI baseUri, CharSequence formattedSchema) | ||
throws IOException, SchemaParseException { | ||
return parse(schemas, formattedSchema, false); | ||
} | ||
|
||
private Schema parse(Collection<Schema> schemas, CharSequence formattedSchema, boolean skipValidation) | ||
throws SchemaParseException { | ||
// TODO: refactor JSON parsing out of the Schema class | ||
Schema.Parser parser; | ||
if (skipValidation) { | ||
parser = new Schema.Parser(Schema.NameValidator.NO_VALIDATION); | ||
parser.setValidateDefaults(false); | ||
} else { | ||
parser = new Schema.Parser(); | ||
} | ||
if (schemas != null) { | ||
parser.addTypes(schemas); | ||
} | ||
Schema schema = parser.parse(formattedSchema.toString()); | ||
if (schemas != null) { | ||
schemas.addAll(parser.getTypes().values()); | ||
} | ||
return schema; | ||
} | ||
} |
Oops, something went wrong.