This is the next gen (v4) of AutoRest Java generator. It's built on AutoRest v3, written in Java, and supports OpenAPI3. It generates clients that work with com.azure:azure-core
.
You need to have the following installed on your machine:
- Node.JS v10.x - v13.x
- Java 8+
- Maven 3.x
You need to have autorest-beta installed through NPM:
npm i -g autorest
To use the latest released preview(https://github.com/Azure/autorest.java/releases), run
autorest --java
--use:@autorest/[email protected]
--input-file:path/to/specs.json
--output-folder:where/to/generate/java/files
--namespace:specified.java.package
The first time running it will take a little longer to download and install all the components.
To build from source code, clone this repo and checkout to v4 branch. Make sure all prerequisites are met, and run
mvn package -Dlocal
This will build a file javagen-jar-with-dependencies.jar
under javagen
module, a preprocess-jar-with-dependencies.jar
under preprocessor
module, a fluentgen-jar-with-dependencies.jar
under fluentgen
module, and a fluentnamer--jar-with-dependencies.jar
under fluentnamer
module.
And then run AutoRest
autorest --java
--use:where/this/repo/is/cloned/autorest.java
--input-file:path/to/specs.json
--output-folder:where/to/generate/java/files
--namespace:specified.java.package
Java files will be generated under where/to/generate/java/files/src/main/java/specified/java/package
.
To debug, add --java.debugger
to the argument list. The JVM will suspend at the beginning of the execution. Then attach a remote debugger in your IDE to localhost:5005
. Make sure you detach the debugger before killing the AutoRest process. Otherwise it will fail to shutdown the JVM and leave it orphaned. (which can be killed in the Task Manager)
Settings can be provided on the command line through --name:value
or in a README file through name: value
. The list of settings for AutoRest in general can be found at https://github.com/Azure/autorest/blob/master/docs/user/command-line-interface.md. The list of settings for AutoRest.Java specifically are listed below:
Option | Description |
---|---|
--enable-xml |
Generates models and clients that can be sent in XML over the wire. Default is false |
--client-side-validations |
Generate validations for required parameters and required model properties. Default is false. |
--generate-client-as-impl |
Append "Impl" to the names of service clients and method groups and place them in the implementation sub-package. Default is false. |
--generate-client-interfaces |
Implies --generate-client-as-impl and generates interfaces for all the "Impl"s. Default is false. |
--generate-sync-async-clients |
Implies --generate-client-as-impl and generates sync and async convenience layer clients for all the "Impl"s. Default is false. |
--implementation-subpackage=STRING |
The sub-package that the Service client and Method Group client implementation classes will be put into. Default is implementation . |
--models-subpackage=STRING |
The sub-package that Enums, Exceptions, and Model types will be put into. Default is models . |
--add-context-parameter |
Indicates whether the com.azure.core.util.Context parameter should be included in generated proxy methods. Default is false. |
--context-client-method-parameter |
Implies --add-context-parameter and indicates whether the com.azure.core.util.Context parameter should also be included in generated client methods. Default is false. |
--sync-methods=all|essential|none |
Specifies mode for generating sync wrappers. Supported value are essential - generates only one sync returning body or header (default) all - generates one sync method for each async methodnone - does not generate any sync methods |
--required-parameter-client-methods |
Indicates whether client method overloads with only required parameters should be generated. Default is false. |
--custom-types=COMMA,SEPARATED,STRINGS |
Specifies a list of files to put in the package specified in --custom-types-subpackage . |
--custom-types-subpackage=STRING |
The sub-package that the custom types should be generated in. The types that custom types reference, or inherit from will also be automatically moved to this sub-package. Recommended usage: You can set this value to models and set --models-subpackage=implementation.models to generate models to implementation.models by default and pick specific models to be public through --custom-types= . |
--client-type-prefix=STRING |
The prefix that will be added to each generated client type. |
--model-override-setter-from-superclass |
Indicates whether to override the superclass setter method in model. Default is false. |
fluent
option enables the generator extension for Azure Management Libraries for Java.
Following settings only works when fluent
option is specified.
Option | Description |
---|---|
--fluent |
Enum. LITE for Fluent Lite; PREMIUM for Fluent Premium. Case insensitive. Default is PREMIUM if provided as other values. |
--pom-file |
String. Name for Maven POM file. Default is pom.xml . |
--package-version |
String. Version number for Maven artifact. Default is 1.0.0-beta.1 . |
--service-name |
String. Service name used in Manager class and other documentations. If not provided, service name is deduced from title configure (from swagger or readme). |
--sdk-integration |
Boolean. Integrate to azure-sdk-for-java. Default is false . |
--track1-naming |
Boolean. Use track1 naming style (withFoo / foo as setter / getter). Default is true . |
--add-inner |
CSV. Treat as inner class (move to fluent.models namespace, append Inner to class name). |
--remove-inner |
CSV. Exclude from inner classes. |
--rename-model |
CSV. Rename classes. Each item is of pattern from:to . |
--name-for-ungrouped-operations |
String. Name for ungrouped operation group. Default to ResourceProviders for Lite. |
--resource-property-as-subresource |
Boolean, experimental. Automatically correct input-only resource type as SubResource . Default is false . |
Also fluent
option will change the default value for some vanilla options.
For example, generate-client-interfaces
, context-client-method-parameter
, required-parameter-client-methods
, model-override-setter-from-superclass
option is by default true
.
The code formatter would require Java 11+ runtime.
To set up customizations, create a Maven project with dependency:
<dependency>
<groupId>com.azure</groupId>
<artifactId>autorest-java-customizations</artifactId>
<version>1.0.0-beta.1</version>
</dependency>
Create a customization class extending from com.azure.autorest.customization.Customization
and override the void customize(LibraryCustomization)
method. You will have access to a LibraryCustomization
class where you will be able to customize the generated Java code before it's written to the disk. Currently, the following customizations are supported:
- Change class modifier
- Change method modifier
- Change method return type
- Add an annotation to a class
- Add an annotation to a method
- Remove an annotation from a class
- Refactor: Rename a class
- Refactor: Rename a method
- Refactor: Generate the getter and setter methods for a property
- Refactor: Rename a property and its corresponding getter and setter methods
- Refactor: Rename an enum member name
- Javadoc: Set the description for a class / method
- Javadoc: Set / remove a parameter's javadoc on a method
- Javadoc: Set the return javadoc on a method
- Javadoc: Set / remove an exception's javadoc on a method
There are 4 customization classes currently available, LibraryCustomization
, PackageCustomization
, ClassCustomization
and JavadocCustomization
. From a given LibraryCustomization
, you can navigate through the packages and classes intuitively with the following methods:
@Override
public void customize(LibraryCustomization customization) {
PackageCustomization models = customization.getPackage("com.azure.myservice.models");
/* code to customize on the package level */
ClassCustomization foo = models.getClass("Foo");
/* code to customize the Foo class */
JavadocCustomization getBarJavadoc = foo.methodJavadoc("getBar");
/* code to customize javadoc for getBar() method */
}
A class Foo
public class Foo {
}
with customization
@Override
public void customize(LibraryCustomization customization) {
PackageCustomization models = customization.getPackage("com.azure.myservice.models");
ClassCustomization foo = models.getClass("Foo");
foo.changeClassModifier(""); // change to package private
}
will generate
class Foo {
}
A method getBar
in the Foo
class
public class Foo {
private Bar bar;
public Bar getBar() {
return this.bar;
}
}
with customization
@Override
public void customize(LibraryCustomization customization) {
PackageCustomization models = customization.getPackage("com.azure.myservice.models");
ClassCustomization foo = models.getClass("Foo");
foo.changeMethodModifier("getBar", "private"); // change to private
}
will generate
public class Foo {
private Bar bar;
private Bar getBar() {
return this.bar;
}
}
You can change a method's return type, and pass a String formatter to transform the original return value statement. If the original return type is void
, simply pass the full return value String expression in place of the String formatter; if the new return type is void
, simply pass null
.
A method getId
in the Foo
class
public class Foo {
private String id;
public String getId() {
return this.id;
}
}
with customization
@Override
public void customize(LibraryCustomization customization) {
PackageCustomization models = customization.getPackage("com.azure.myservice.models");
ClassCustomization foo = models.getClass("Foo");
foo.changeMethodReturnType("getId", "UUID", "UUID.fromString(%s)"); // change to private
}
will generate
public class Foo {
private String id;
public UUID getId() {
String returnValue = this.id;
return UUID.fromString(returnValue);
}
}
The UUID
class will be automatically imported.
A class Foo
public class Foo {
}
with customization
@Override
public void customize(LibraryCustomization customization) {
PackageCustomization models = customization.getPackage("com.azure.myservice.models");
ClassCustomization foo = models.getClass("Foo");
foo.addClassAnnotation("JsonFlatten");
}
will generate
@JsonFlatten
public class Foo {
}
The JsonFlatten
class will be automatically imported.
A method getBar
in the Foo
class
public class Foo {
private Bar bar;
public Bar getBar() {
return this.bar;
}
}
with customization
@Override
public void customize(LibraryCustomization customization) {
PackageCustomization models = customization.getPackage("com.azure.myservice.models");
ClassCustomization foo = models.getClass("Foo");
foo.addMethodAnnotation("getBar", "Deprecated");
}
will generate
public class Foo {
private Bar bar;
@Deprecated
public Bar getBar() {
return this.bar;
}
}
The Deprecated
class will be automatically imported.
A class Foo
@Fluent
public class Foo {
}
with customization
@Override
public void customize(LibraryCustomization customization) {
PackageCustomization models = customization.getPackage("com.azure.myservice.models");
ClassCustomization foo = models.getClass("Foo");
foo.removeClassAnnotation("Fluent");
}
will generate
public class Foo {
}
A class Foo
public class Foo {
}
with customization
@Override
public void customize(LibraryCustomization customization) {
PackageCustomization models = customization.getPackage("com.azure.myservice.models");
models.renameClass("Foo", "FooInfo");
}
will generate
class FooInfo {
}
All references of Foo
will be modified to FooInfo
. When a valid value is provided, this customization is guaranteed to not break the build.
A method isSupportsUnicode
in the Foo
class
public class Foo {
private boolean supportsUnicode;
public boolean isSupportsUnicode() {
return this.supportsUnicode;
}
}
with customization
@Override
public void customize(LibraryCustomization customization) {
PackageCustomization models = customization.getPackage("com.azure.myservice.models");
ClassCustomization foo = models.getClass("Foo");
foo.renameMethod("isSupportsUnicode", "supportsUnicode");
}
will generate
public class Foo {
private boolean supportsUnicode;
public boolean supportsUnicode() {
return this.supportsUnicode;
}
}
All references of isSupportsUnicode()
will be modified to supportsUnicode()
. When a valid value is provided, this customization is guaranteed to not break the build.
A property active
in the Foo
class
public class Foo {
private boolean active;
}
with customization
@Override
public void customize(LibraryCustomization customization) {
PackageCustomization models = customization.getPackage("com.azure.myservice.models");
ClassCustomization foo = models.getClass("Foo");
foo.generateGetterAndSetter("active");
}
will generate
public class Foo {
private boolean active;
public boolean isActive() {
return this.active;
}
public Foo setActive(boolean active) {
this.active = active;
return this;
}
}
If the class already contains a getter or a setter method, the current method will be kept. This customization is guaranteed to not break the build.
A property whitelist
in the Foo
class
public class Foo {
private List<String> whiteList;
public List<String> getWhiteList() {
return this.whiteList;
}
public Foo setWhiteList(List<String> whiteList) {
this.whiteList = whiteList;
return this;
}
}
with customization
@Override
public void customize(LibraryCustomization customization) {
PackageCustomization models = customization.getPackage("com.azure.myservice.models");
ClassCustomization foo = models.getClass("Foo");
foo.renameProperty("whiteList", "allowList");
}
will generate
public class Foo {
private List<String> allowList;
public List<String> getAllowList() {
return this.allowList;
}
public Foo setAllowList(List<String> allowList) {
this.allowList = allowList;
return this;
}
}
This customization is guaranteed to not break the build.
An enum member JPG
in an enum class ImageFileType
:
public enum ImageFileType {
GIF("gif"),
JPG("jpg"),
TIFF("tiff"),
PNG("png");
}
with customization
@Override
public void customize(LibraryCustomization customization) {
PackageCustomization models = customization.getPackage("com.azure.myservice.models");
ClassCustomization imageFileType = models.getClass("ImageFileType");
foo.renameEnumMember("JPG", "JPEG");
}
will generate
public enum ImageFileType {
GIF("gif"),
JPEG("jpg"),
TIFF("tiff"),
PNG("png");
}
This customization is guaranteed to not break the build.
A class Foo
/** Class Foo. */
public class Foo {
private boolean active;
public boolean isActive() {
return this.active;
}
public Foo setActive(boolean active) {
this.active = active;
return this;
}
}
with customization
@Override
public void customize(LibraryCustomization customization) {
PackageCustomization models = customization.getPackage("com.azure.myservice.models");
ClassCustomization foo = models.getClass("Foo");
foo.classJavadoc().setDescription("A Foo object stored in Azure.")
foo.methodJavadoc("setActive").setDescription("Set the active value.");
}
will generate
/**
* A Foo object stored in Azure.
*/
public class Foo {
private boolean active;
public boolean isActive() {
return this.active;
}
/**
* Set the active value.
*/
public Foo setActive(boolean active) {
this.active = active;
return this;
}
}
A class Foo
public class Foo {
private boolean active;
public boolean isActive() {
return this.active;
}
/**
* Set the active value.
*/
public Foo setActive(boolean active) {
this.active = active;
return this;
}
}
with customization
@Override
public void customize(LibraryCustomization customization) {
PackageCustomization models = customization.getPackage("com.azure.myservice.models");
ClassCustomization foo = models.getClass("Foo");
foo.methodJavadoc("setActive").setParam("active", "if the foo object is in active state");
}
will generate
public class Foo {
private boolean active;
public boolean isActive() {
return this.active;
}
/**
* Set the active value.
*
* @param active if the foo object is in active state
*/
public Foo setActive(boolean active) {
this.active = active;
return this;
}
}
A Foo
class
public class Foo {
private boolean active;
public boolean isActive() {
return this.active;
}
/**
* Set the active value.
*
* @param active if the foo object is in active state
*/
public Foo setActive(boolean active) {
this.active = active;
return this;
}
}
with customization
@Override
public void customize(LibraryCustomization customization) {
PackageCustomization models = customization.getPackage("com.azure.myservice.models");
ClassCustomization foo = models.getClass("Foo");
foo.methodJavadoc("setActive").setReturn("the current foo object");
}
will generate
public class Foo {
private boolean active;
public boolean isActive() {
return this.active;
}
/**
* Set the active value.
*
* @param active if the foo object is in active state
* @return the current foo object
*/
public Foo setActive(boolean active) {
this.active = active;
return this;
}
}
A FooClient
class
public class FooClient {
/**
* Create a Foo object.
*
* @param foo the foo object to create in Azure
* @return the response for creating the foo object
*/
public CreateFooResponse createFoo(Foo foo) {
/* REST call to create foo */
}
}
with customization
@Override
public void customize(LibraryCustomization customization) {
PackageCustomization root = customization.getPackage("com.azure.myservice");
ClassCustomization fooClient = root.getClass("FooClient");
fooClient.methodJavadoc("createFoo").addThrows("HttpResponseException", "An unsuccessful response is received");
}
will generate
public class FooClient {
/**
* Create a Foo object.
*
* @param foo the foo object to create in Azure
* @return the response for creating the foo object
* @throws HttpResponseException An unsuccessful response is received
*/
public CreateFooResponse createFoo(Foo foo) {
/* REST call to create foo */
}
}
This contains the base classes and utilities for creating an AutoRest extension in Java. It handles the JSON RPC communications with AutoRest core, provides JSON and YAML parsers, and provides the POJO models for the code model output from modelerfour.
Extend from NewPlugin.java
class if you are writing a new extension in Java.
This contains the actual generator extension, including mappers that maps a code model to a Java client model, and templates that writes the Java client models into .java files.
This contains the generator extension for Azure Management Libraries.
This contains the generated classes from the test swaggers in src/main
. The code here should always be kept up-to-date with the output of the generator in javagen
.
This also contains test code for these generated code under src/test
. Running the tests will hit the test server running locally (see https://github.com/Azure/autorest.testserver for instructions) and verify the correctness of the generated code.
This project welcomes contributions and suggestions. Most contributions require you to agree to a Contributor License Agreement (CLA) declaring that you have the right to, and actually do, grant us the rights to use your contribution. For details, visit https://cla.microsoft.com.
When you submit a pull request, a CLA-bot will automatically determine whether you need to provide a CLA and decorate the PR appropriately (e.g., label, comment). Simply follow the instructions provided by the bot. You will only need to do this once across all repos using our CLA.
This project has adopted the Microsoft Open Source Code of Conduct. For more information see the Code of Conduct FAQ or contact [email protected] with any additional questions or comments.
- Please don't edit this section unless you're re-configuring how the Java extension plugs in to AutoRest AutoRest needs the below config to pick this up as a plug-in - see https://github.com/Azure/autorest/blob/master/docs/developer/architecture/AutoRest-extension.md
use: $(this-folder)/javagen
use: $(this-folder)/fluentgen