The goal of this project is to provide a radically simplified Java API for Azure. It follows a flavor of modern API design patterns (builder, fluent) optimized for readability, writeability and succinctness.
⚠️ NOTE: this was an experimental labs project, which developed the API design approach that the actual Azure Libraries for Java (a.k.a. Azure SDK for Java) have now started to follow, as of version 1.0 beta 2. Starting with that version, the Azure Libraries for Java have caught up with and surpassed the "shortcuts" prototype, so if you are looking for the latest and greatest implementation and evolution of the ideas originally prototyped in this "shortcuts" project, you can now go directly to the Azure Libraries for Java. The original creator of the "shortcuts" prototype is now the team lead for the Azure SDK for Java.
Here's an example for creating a virtual network, which is very representative of the approach followed by the shortcuts:
subscription.networks().define("mynetwork")
.withRegion(Region.US_WEST)
.withExistingResourceGroup("<resource-group-name>")
.withAddressSpace("10.0.0.0/28")
.withSubnet("Foo", "10.0.0.0/29")
.withSubnet("Bar", "10.0.0.8/29")
.create();
The shortcuts library supports Azure's "modern" ARM (Azure Resource Model) model. The previous "classic" ASM (Azure Service Model) API is no longer maintained nor documented.
A lot of short code samples are located in the packages com.microsoft.azure.shortcuts.resources.samples
(https://github.com/Microsoft/azure-shortcuts-for-java/tree/master/src/com/microsoft/azure/shortcuts/resources/samples).
This "shortcuts" project is now the experimental pilot for the new version of the Azure SDK for Java. Although its original goal has been to drastically simplify only the hardest of the most important scenarios that developers have been running into, the current revamping of the Azure SDK for Java (especially the management API) is in large parts being modeled after this "shortcuts" project.
To work on this project, it's easiest to use Eclipse and Maven (kudos to Ted Gao for the pointers):
- Create a directory for the workspace
- In that workspace directory, run
mvn -Declipse.workspace=. eclipse:configure-workspace
- In the project directory, after
git-clone
, runmvn eclipse:eclipse -DdownloadSources=true
- In Eclipse, use the workspace folder created earlier, and import the project into it (don't copy)
- Java 7+
⚠️ NOTE: Although the project is currently based on Java 7, switching to Java 8 is under consideration, as v8 offers some important programming features it'd make a lot of sense to take advantage of (especially lambda support).
- Azure SDK for Java v0.9.* (installed by the pom.xml file, so no need to install separately)
- An Azure subscription
Everything that is explicitly documented in this readme is being tested. The samples are excerpts from automation tests. Some typos are still occasionally possible - sorry! Someday this will be more automated for maximum reliability. But the general principles this project aspires to follow rigorously are "Documentation is code".
There are no JavaDocs. Someday there may be but note that an important goal of this design approach is to minimize the user's dependence on API documentation. The API should "just make sense". The basic philosophy here is: the more documentation a design requires, the less optimal it is. "If you have to explain it, then you're doing it wrong".
In that spirit, if you skip over this section and jump directly to the examples, chances are it will "just make sense". But if you'd like to learn more about the design approach in the abstract, read on:
The key design principles behind the shortcuts API are: to be intuitive, succint, consistent, and preventing you from winding up in an invalid state.
There are a small handful of general patterns to be aware of; once you remember these, everything else should be self-explanatory.
There are no constructors anywhere. The only class exposed to the user is Subscription
, but only so that the user can use the static authenticate()
method on it as the entry point to everything else. Everything else that is intended for the user's use is an interface.
To create a new instance of any type of a top level cloud entity (e.g. Network
), you use the top level "collection" of those objects hanging off of the Subscription
client object as the factory. And yes, there is only one single client object to instantiate and deal with.
In more detail:
- start with the "collection" of those objects hanging off as a member of the
Subscription
client instance object (e.g.subscription.networks()
), - then call
.define("name-of-the-new-entity")
on that collection. This starts the "definition". - from that point on, use command chaining (i.e. '.' dots) to specify the various required and optional parameters. They all look like this:
.with*()
(e.g..withExistingResourceGroup("myresourcegroup")
). Note that due to the special way the shortcuts APi is designed, after each such "with"-setter, AutoComplete will only suggest the set of setters that are valid/required at that stage of the definition. This way, it will force you to continue specifying the suggested "with" setters until you see.create()
among the possible choices. This is how the equivalent of required constructor parameters are exposed. - many resource types in Azure (e.g. virtual machines) require other associated resources (e.g. a resource group, a storage account) to be already present. The
.with*
setters often enable you to either select an existing related resource, i.e..withExisting*()
, or to request a new such resource to be created on the fly in a shortcut way, i.e..withNew*()
. When created on the fly, the associated resource is created in the same region and resource group and by default uses a derived name. - when
.create()
becomes available among the AutoComplete choices, it means you have reached a stage in the entity definition where all the other parameters ("with"-setters) are optional. Some of those setters are optional because Azure allows them to be so, and others are optional because the shortcuts assume some sort of a default, but Azure still requires them. Calling.create()
is what completes the definition and starts the actual creation process in the cloud.
Updates to existing entities are also done in a "builder pattern/fluent interface" kind of way:
- start with
.update()
on the collection as the "factory" of an update template - command-chain the needed
.with*()
settings (usually all optional) - and finish off with a call to
.apply()
In essence, the above is the shortcuts API's take on the "builder pattern + fluent interface + factory pattern + extra smarts" combo in action. It's just that instead of the more traditional .create()
or new
naming, the shortcuts use .define()
or .update()
for creating/updating objects. And instead of the more conventional .build()
, the shortcuts use .create()
or .apply()
.
⚠️ TODO The Update functionality is only beginning to be implemented in the shortcuts
In general, the shortcut naming tends to be consistent with the Azure SDK. However, it does not follow the SDK naming rigorously. Sometimes, simplicity or succinctness trumps consistency (e.g. Azure SDK has VirtualNetwork
, shortcuts have Network
.).
Some helpful pointers:
- In the cases when the same class name is used, make sure you reference the right package!
- As for class member naming, it is hard to avoid the impression that the Azure SDK has somewhat abused the "get/set" convention. The shortcuts don't. In fact, it is only on the very rare occasion that using the "get" prefix is justified, so you will practically never see it in the shortcuts.
- Since the shortcuts rely heavily on the fluent interface, you will not see
.set*(...)
anywhere, only.with*(...)
. The "with" naming convention in the context of fluent interface setters has been adopted because "set" functions are conventionally expected to returnvoid
, whereas "with" returns an object. Modern Java API implementations from other projects are increasingly adopting the same "with" setter naming convention. (e.g. AWS SDK)
And again, a quick look at any of the below code samples should make the above points rather obvious.
-
Many shortcut objects are wrappers of Azure SDK objects. Since the shortcuts might not expose all of the settings available on the underlying Azure SDK classes, for those shortcut objects, to get access to the underlying Azure SDK object, use the
.inner()
function. -
Some Azure SDK objects can also be used as input parameters in APIs where they make sense. For example, when a storage account is expected in some shortcut API, generally it can be provided as either:
- the shortcut
StorageAccount
object, - the Azure SDK's
StorageAccount
object, - or the resource id string.
- the shortcut
Inside the \*.samples
packages, you will find a number of runnable code samples. They are implemented as classes with main()
, so they can be run as console apps. So for each of the sample classes, you can just Debug As > Java Application.
Many of the samples rely on a credentials file in the root of the project:
For the "Resource" ARM-based APIs specifically, you can use the very experimental "my.authfile" format containing all the inputs needed by the Azure Active Directory authentication and relying on you setting up a service principal for your app.
Further simplification of the authentication process is a subject of active investigation, but for now you can create the file manually, as per the Authentication section.
Table of contents:
- Authentication
- Virtual Machines
- Virtual Networks
- Network Interfaces
- Public IP Addresses
- Network Security Groups
- Storage Accounts
- Regions
- Resource Groups
- Resources
- Resource Providers
- Availability Sets
This is the first step for all the other examples.:
ARM: import from the
com.microsoft.azure.shortcuts.resources.*
packages
String authFilePath = "<my-auth-file>"; // See explanation below
String subscriptionId = "<subscription-GUID>";
Subscription subscription = Subscription.authenticate(authFilePath, subscriptionId);
⚠️ NOTE: Active Directory auth for ARM currently requires a lot of inputs and token management logic. To simplify matters, the above constructor assumes you have set up a service principal for your application and can put the required inputs into this experimental PublishSettings-like files in the following formats.
<azureAuth>
<subscription
id="<subscription id>"
tenant="<tenant id>"
client="<client id>"
key="<client key>"
managementURI="https://management.core.windows.net/"
baseURL="https://management.azure.com/"
authURL="https://login.windows.net/"
/>
</azureAuth>
id=########-####-####-####-############
tenant=########-####-####-####-############
client=########-####-####-####-############
key=########
managementURI=https\://management.core.windows.net/
baseURL=https\://management.azure.com/
authURL=https\://login.windows.net/
You can just save a file with these contents, replacing the placeholders with the appropriate settings, and use it as your "auth-file" in the example above.
ARM: import from the
com.microsoft.azure.shortcuts.resources.*
packages
There are many variations possible of the following minimalistic approach:
VirtualMachine vmWin = subscription.virtualMachines().define("<vm-name>")
.withRegion(Region.US_WEST)
.withNewResourceGroup("<new-group-name>")
.withNewNetwork("10.0.0.0/28")
.withPrivateIpAddressDynamic()
.withNewPublicIpAddress("<new-domain-label>")
.withAdminUsername("<admin-user-name>")
.withAdminPassword("<password>")
.withLatestImage("MicrosoftWindowsServer", "WindowsServer", "2008-R2-SP1")
.withSize(Size.Type.BASIC_A1)
.withNewStorageAccount()
.create();
As a shortcut, this approach combines the creation or selection of the related required resources into one statement.
For example, a new resource group can be created for the virtual machine (.withNewResourceGroup(...)
) or an existing one can be selected (.withExistingResourceGroup(...)
).
Similarly, a new virtual network can be created (.withNewNetwork(...)
) or an existing one can be used (.withExistingNetwork(...)
).
The private IP within the virtual network can be either dynamically allocated (.withPrivateIpDynamic()
) or statically (.withPrivateIpStatic("<private-ip-address>")
).
Associating the VM with a public IP is optional. An existing public IP can be assigned (.withExistingPublicIpAddress(...)
), or a new one created (.withNewPublicIpAddress()
). If new, then it can be optionally associated with a leaf domain label which will form the DNS record for this VM (.withNewPublicIpAddress("<new-domain-label>")
).
If specifying the IP addresses and the network explicitly like in the above example, a new network interface is created implicitly behind the scenes and set as the primary interface for the virtual machine. The IP addresses go into the primary IP configuration for that network interface. If you want to use an already existing network interface in your subscription, then instead of .withNewNetwork()
, invoke .withExistingNetworkInterface()
. This will also skip over the selection of the rest of the networking information.
Creating a virtual machine requires a storage account to keep the VHD in. A new storage account can be requested (.withNewStorageAccount()
) or an existing one (.withExistingStorageAccount()
).
Any such related resource that is created in the process of creating a virtual machine will be created in the same resource group and region as the virtual machine.
A number of settings are optional so they can be specified at the creatable stage of the virtual machine definition, i.e. at the stage at which .create()
is available among the members. For example, the above example can rewritten to separate the creatable stage from the required stages:
VirtualMachine.DefinitionProvisionable vmProvisionable = subscription.virtualMachines().define("vm" + deploymentId)
.withRegion(Region.US_WEST)
.withNewResourceGroup(groupName)
.withNewNetwork("10.0.0.0/28")
.withPrivateIpAddressDynamic()
.withNewPublicIpAddress("vm" + deploymentId)
.withAdminUsername("shortcuts")
.withAdminPassword("Abcd.1234")
.withLatestImage("MicrosoftWindowsServer", "WindowsServer", "2008-R2-SP1");
At this stage, additional settings can be specified that are optional before create()
is invoked.
Based on the earlier creatable definition, the following code attaches 2 new empty data disks to the virtual machine definition. Their logical unit number (LUN) is set automatically based on the order of attachment:
vmProvisionable
.withNewDataDisk(100) // Attach a 100 GB disk as LUN 1
.withNewDataDisk(200); // Attach a 200 GB disk as LUN 2
Attaching an existing VHD file as a data disk:
vmProvisionable = vmProvisionable
.withExistingDataDisk("https://vm1455045717874store.blob.core.windows.net/vm1455045717874/disk0.vhd");
Based on the earlier creatable definition, the following code specifies a new availability set to be created for this virtual machine to be associated with:
vmProvisionable = vmProvisionable
.withNewAvailabilitySet("myAvailabilitySet");
All virtual machine names (or ids) in a subscription:
ARM: import from
com.microsoft.azure.shortcuts.resources.*
packages
Map<String, VirtualMachine> vms = subscription.virtualMachines().asMap();
System.out.println(String.format("Virtual machines: \n\t%s", String.join("\n\t", vms.keySet())));
Virtual machines in a specific resource group (resource model "ARM" only)
Map<String, VirtualMachine> vms = subscription.virtualMachines().asMap("<group-name>");
System.out.println(String.format("Virtual machines: \n\t%s", String.join("\n\t", vms.keySet())));
ARM: import from the
com.microsoft.azure.shortcuts.resources.*
packages
Using the resource id:
VirtualMachine vm = subscription.virtualMachines("<resource-id>");
Using the resource group name and virtual machine name:
VirtualMachine vm = subscription.virtualMachines("<resource-group-name>", "<vm-name>");
ARM: import from the
com.microsoft.azure.shortcuts.resources.*
packages
You need to specify the region to get the sizes. The returned sizes are indexed by their name:
Map<String, Size> sizes = subscription.sizes().asMap("westus");
Therefore, to get the names only:
Set<String> sizeNames = subscription.sizes().asMap("westus").keySet();
🚩 TODO:
Any of the following approaches:
subscription.virtualMachines().delete("<vm-resource-id>");
subscription.virtualMachines().delete("<resource-group-name>", "<vm-name>");
subscription.virtualMachines("<vm-resource-id>").delete();
subscription.virtualMachines("<resource-group-name>", "<vm-name>").delete();
By resource group and name:
subscription.virtualMachines().get("<resource-group-name>", "<vm-name>").stop();
By resource id:
subscription.virtualMachines().get("<vm-resource-id>").stop();
By resource group and name:
subscription.virtualMachines().get("<resource-group-name>", "<vm-name>").start();
By resource id:
subscription.virtualMachines().get("<vm-resource-id>").start();
By resource group and name:
subscription.virtualMachines().get("<resource-group-name>", "<vm-name>").restart();
By resource id:
subscription.virtualMachines().get("<vm-resource-id>").restart();
By resource group and name:
subscription.virtualMachines().get("<resource-group-name>", "<vm-name>").deallocate();
By resource id:
subscription.virtualMachines().get("<vm-resource-id>").deallocate();
ARM: import from the
com.microsoft.azure.shortcuts.resources.*
packages
With an explicitly defined address space, a default subnet containing the entirety of the IP address space, in a new auto-generated resource group:
Network network = subscription.networks().define("<new-network-name>")
.withRegion(Region.US_WEST)
.withNewResourceGroup()
.withAddressSpace("10.0.0.0/28")
.create();
With multiple, explicitly defined subnets and an existing resource group:
subscription.networks().define("<new-network-name>")
.withRegion(Region.US_WEST)
.withExistingResourceGroup("<existing-resource-group-name>")
.withAddressSpace("10.0.0.0/28")
.withSubnet("Foo", "10.0.0.0/29")
.withSubnet("Bar", "10.0.0.8/29")
.create();
With multiple, explicitly defined subnets and associating an existing network security group with one of them, using the granular child resource definition approach:
Network network = subscription.networks().define(newNetworkName)
.withRegion(Region.US_WEST)
.withExistingResourceGroup(existingGroupName)
.withAddressSpace("10.0.0.0/28")
.defineSubnet("subnetA")
.withAddressPrefix("10.0.0.0/29")
.withExistingNetworkSecurityGroup(existingNsgName)
.attach()
.defineSubnet("subnetB")
.withAddressPrefix("10.0.0.8/29")
.attach()
.create();
ARM: import from the
com.microsoft.azure.shortcuts.resources.*
packages
All networks in a subscription, as a map indexed by resource id:
Map<String, Network> networks = subscription.networks().asMap();
Resource ids only:
Set<String> networkIds = subscription.networks().asMap().keySet();
Networks in a specific resource group:
Map<String, Network> networks = subscription.networks().asMap("<resource-group-name">);
ARM: import from the
com.microsoft.azure.shortcuts.resources.*
packages
By providing a virtual network resource ID (returned as a key in networks().asMap()
):
Network network = subscription.networks("<network-resource-id>");
or by providing the resource group name and the virtual network name:
Network network = subscription.networks("<resource-group-name>", "<network-name>");
The subnets of the virtual network are available from network.subnets()
.
The IP addresses of DNS servers associated with the virtual network are available from network.dnsServerIPs()
.
The address spaces (in CIDR format) of the virtual network are available from network.addressSpaces()
.
ARM: import from the
com.microsoft.subscription.shortcuts.resources.*
packages
Any of the following methods:
subscription.networks().delete("<network-resource-id>");
subscription.networks().delete("<resource-group-name>", "<network-name>");
subscription.networks("<network-resource-id>").delete();
subscription.networks("<resource-group-name>", "<network-name>").delete();
ARM: import from the
com.microsoft.subscription.shortcuts.resources.*
packages
When using the minimum set of required inputs, a new resource group is created automatically, in the same region, with a name derived from the NIC's name. A virtual network providing the subnet the NIC is to be associated with is also created automatically, with one subnet covering the entirety of the address space:
NetworkInterface nicMinimal = subscription.networkInterfaces().define(newNetworkInterfaceName)
.withRegion(Region.US_WEST)
.withNewResourceGroup("<new-resource-group-name>")
.withNewNetwork("10.0.0.0/28")
.withPrivateIpAddressDynamic()
.withoutPublicIpAddress()
.create();
Creating a network interface with a new resource group, dynamic private IP and a new, dynamically allocated public IP with a leaf domain label automatically generated based on the name of the NIC, and associating with an existing network security group:
NetworkInterface nic = subscription.networkInterfaces().define("<new-nic-name>")
.withRegion(Region.US_WEST)
.withExistingResourceGroup("<new-group-name>")
.withExistingNetwork("<existing-network-id>")
.withSubnet("subnet1")
.withPrivateIpAddressStatic("10.0.0.5")
.withNewPublicIpAddress()
.withExistingNetworkSecurityGroup("<existing-nsg-id>")
.withTag("hello", "world")
.create();
In the subscription (all resource groups):
Map<String, NetworkInterface> nics = subscription.networkInterfaces().asMap();
In a specific resource group:
Map<String, NetworkInterface> nics = subscription.networkInterfaces().asMap("<resource-group-name>");
Using its resource id:
NetworkInterface nic = subscription.networkInterfaces().get("<resource-id>");
or:
NetworkInterface nic = subscription.networkInterfaces("<resource-id>");
Using its resource group and name:
NetworkInterface nic = subscription.networkInterfaces().get("<resource-group-name>", "<network-interface-name>");
or
NetworkInterface nic = subscription.networkInterfaces("<resource-group-name>", "<network-interface-name>");
Any of the following approaches:
subscription.networkInterfaces().delete("<resource-id>");
subscription.networkInterfaces().delete("<resource-group-name>", "<network-interface-name>");
subscription.networkInterfaces("<resource-id>").delete();
subscription.networkInterfaces("<resource-group-name>", "<network-interface-name>").delete();
ARM: import from the
com.microsoft.subscription.shortcuts.resources.*
packages
Providing minimal inputs will result in a public IP address for which a resource group will be automatically generated, dynamic IP allocation will be enabled and a leaf domain name will be specified, derived from the provided name:
PublicIpAddress pipMinimal = subscription.publicIpAddresses().define("<new-public-address-name>")
.withRegion(Region.US_WEST)
.withNewResourceGroup()
.create();
With static IP allocation, an explicitly defined leaf domain label and a tag:
PublicIpAddress pip = subscription.publicIpAddresses().define(newPublicIpAddressName + "2")
.withRegion(Region.US_WEST)
.withExistingResourceGroup(existingGroupName)
.withLeafDomainLabel("hellomarcins")
.withStaticIp()
.withTag("hello", "world")
.create();
From the entire subscription, as a Map
indexed by name:
Map<String, PublicIpAddress> pips = subscription.publicIpAddresses().asMap();
From a specific resource group, as a Map
indexed by name:
Map<String, PublicIpAddress> pips = subscription.publicIpAddresses().asMap("my-resoruce-group-name");
Using its resource id:
PublicIpAddress pip = subscription.publicIpAddresses().get("resource-id");
or:
PublicIpAddress pip = subscription.publicIpAddresses("resource-id");
Using its resource group and name:
PublicIpAddress pip = subscription.publicIpAddresses().get("<resource-group-name>", "<pip-name>");
or
PublicIpAddress pip = subscription.publicIpAddresses("<resource-group-name>", "<pip-name>");
Any of the following methods:
subscription.publicIpAddresses().delete("<pip-resource-id>");
subscription.publicIpAddresses().delete("<resource-group-name>", "<pip-name>");
subscription.publicIpAddresses("<pip-resource-id>").delete();
subscription.publicIpAddresses("<resource-group-name>", "<pip-name>").delete();
ARM: import from the
com.microsoft.subscription.shortcuts.resources.*
packages
Providing minimal inputs will result in a network security group for which a resource group will be automatically generated and a default set of rules applied:
NetworkSecurityGroup nsgMinimal = subscription.networkSecurityGroups().define("<network-security-group-name>")
.withRegion(Region.US_WEST)
.withNewResourceGroup()
.create();
With network security rules and tags:
NetworkSecurityGroup nsg = subscription.networkSecurityGroups().define("<nsg-name>")
.withRegion(Region.US_WEST)
.withExistingResourceGroup("<group-name>")
.defineRule("rule1")
.allowInbound()
.fromAnyAddress()
.fromPort(80)
.toAddress("10.0.0.0/29")
.toPort(80)
.withProtocol(Protocol.TCP)
.attach()
.defineRule("rule2")
.denyOutbound()
.fromAnyAddress()
.fromAnyPort()
.toAnyAddress()
.toAnyPort()
.withProtocol(Protocol.UDP)
.attach()
.create();
To associate or create an NSG while creating a network interface, see the Network Interfaces section - the .withExistingNetworkSecurityGroup()
and .withNewNetworkSecurityGroup()
methods.
To associate an NSG with a subnet while creating a virtual network, see the Virtual Networks section - the .withExistingNetworkSecurityGroup()
method.
From the entire subscription, as a Map
indexed by id:
Map<String, NetworkSecurityGroup> nsgs = subscription.networkSecurityGroups().asMap();
From a specific resource group, as a Map
indexed by id:
Map<String, NetworkSecurityGroup> nsgs = subscription.networkSecurityGroups().asMap("<resource-group-name>");
Using its resource id:
NetworkSecurityGroup nsg = subscription.networkSecurityGroups().get("<resource-id>");
or:
NetworkSecurityGroup nsg = subscription.networkSecurityGroups("<resource-id>");
Using its resource group and name:
NetworkSecurityGroup nsg = subscription.networkSecurityGroups().get("<resource-group-name>", "<nsg-name>");
or
NetworkSecurityGroup nsg = subscription.networkSecurityGroups("<resource-group-name>", "<nsg-name>");
Any of the following methods:
subscription.networkSecurityGroups().delete("<nsg-resource-id>");
subscription.networkSecurityGroups().delete("<resource-group-name>", "<nsg-name>");
subscription.networkSecurityGroups("<nsg-resource-id>").delete();
subscription.networkSecurityGroups("<resource-group-name>", "<nsg-name>").delete();
ARM: import from the
com.microsoft.subscription.shortcuts.resources.*
packages
With the required minimum set of input parameters:
StorageAccount storageAccount = subscription.storageAccounts().define("<new-storage-account-name>")
.withRegion(Region.US_WEST)
.withNewResourceGroup()
.create();
In an existing resource group:
subscription.storageAccounts().define("<new-storage-account-name>")
.withRegion(Region.US_WEST)
.withAccountType(AccountType.StandardLRS)
.withExistingResourceGroup("<existing-resource-group-name>")
.create();
ARM: import from
com.microsoft.azure.shortcuts.resources.*
packages
As a map, indexed by name:
Map<String, StorageAccount> storageAccounts = subscription.storageAccounts().asMap();
Names only:
List<String> storageAccountNames = subscription.storageAccounts().asMap().keySet();
Storage accounts in a selected resource group:
Map<String, StorageAccount> storageAccounts = subscription.storageAccounts().asMap("<resource-group-name>");
🚩 TODO:
ARM: import from the
com.microsoft.subscription.shortcuts.resources.*
packages
Getting a storage account using its unique resource id using any of the following methods:
StorageAccount storageAccount = subscription.storageAccounts().get("<storage-account-id>");
StorageAccount storageAccount = subscription.storageAccounts("<storage-account-id>");
StorageAccount storageAccount = subscription.storageAccounts("<resource-group-name>", "<storage-account-name>");
ARM: import from the
com.microsoft.subscription.shortcuts.resources.*
packages
Any of the following methods:
subscription.storageAccounts().delete("<storage-account-resource-id>");
subscription.storageAccounts().delete("<resource-group-name>", "<storage-account-name>");
subscription.storageAccounts("<storage-account-resource-id>").delete();
subscription.storageAccounts("<resource-group-name>", "<storage-account-name>").delete();
ARM: import from the
com.microsoft.subscription.shortcuts.resources.*
packages
The Region
enum provides the list (as constants) of all the possible Azure locations.
Region[] regions = Region.values();
🚩 TODO
ARM: import from the
com.microsoft.subscription.shortcuts.resources.*
packages
subscription.resourceGroups().define("myResourceGroup")
.withRegion(Region.US_WEST)
.withTag("hello", "world")
.create();
Indexed by name:
Map<String, ResourceGroup> resourceGroups = subscription.resourceGroups().asMap();
Names only:
Set<String> resourceGroupNames = subscription.resourceGroups().asMap().keySet();
Tags are key/value pairs.
subscription.resourceGroups().update("<resource-group-name>")
.withTag("foo", "bar")
.withoutTag("hello")
.apply();
You can also pass an instance of Map<String, String>
with all the tags in it:
subscription.resourceGroups().update("<resource-group-name>")
.withTags(myMap)
.apply();
Either of the following methods:
ResourceGroup resourceGroup = subscription.resourceGroups("<resource-group-name>");
ResourceGroup resourceGroup = subscription.resourceGroups().get("<resource-group-name>");
Either of the following methods:
subscription.resourceGroups().delete("<resource-group-name>");
subscription.resourceGroups("<resource-group-name>").delete();
This applies only to ARM, so import from the
com.microsoft.subscription.shortcuts.resources.*
packages
All resources in a subscription, indexed by id:
Map<String, Resource> resources = subscription.resources().asMap();
Resources in a specific resource group:
Map<String, Resource> resources = subscription.resources().asMap("<resource-group-name>");
If you know the full ID of the resource (e.g. you got it from the resources().asMap().keySet()
), then:
Resource resource = subscription.resources("<resource-id>");
Else, if you know the resource name, type, provider and resource group, then:
Resource resource = subscription.resources().get(
"<resource-name>",
"<resource-type>",
"<resource-provider-namespace>",
"<resource-group>");
Using its resource ID:
subscription.resources().delete("<resource-id">);
Or using its metadata:
subscription.resources().delete("<short-name>", "<resource-type>", "<provider-namespace>", "<resource-group-name>");
Or, if you've already gotten a reference to a Resource
object (represented by resource
below) from get()
, then:
resource.delete();
ARM: import from the
com.microsoft.azure.shortcuts.resources.*
packages
Providers as a Map
, indexed by namespace:
Map<String, Provider> providers = subscription.providers().asMap();
Namespaces only:
Set<String> providerNamespaces = subscription.providers().asMap().keySet();
Using the namespace of the provider you can get from providers().names()
:
Provider provider = subscription.providers("microsoft.classicstorage");
Provider provider = subscription.providers("<provider-namespace>");
for(ResourceType t : provider.resourceTypes().values()) {
System.out.println(String.format("%s: %s", t.name(), Arrays.toString(t.apiVersions())));
}
Either of the following methods;
String latestAPIVersion = subscription.providers("<provider-namespace>").resourceTypes().get("<resource-type>").latestApiVersion();
String latestAPIVersion = subscription.providers("<provider-namespace>").resourceTypes("<resource-type>").latestApiVersion();
ARM: import from the
com.microsoft.azure.shortcuts.resources.*
packages
With minimum inputs, in its own new default resource group:
subscription.availabilitySets().define("myavailabilityset")
.withRegion(Region.US_WEST)
.witgGroupNew()
.create();
Within an existing resource group, and setting a tag:
subscription.availabilitySets().define("myavailabilityset")
.withRegion(Region.US_WEST)
.withExistingResourceGroup("<existing-resource-group-name>")
.withTag("hello", "world")
.create();
Availability sets as a map, in a specific resource group, indexed by id:
Map<String, AvailabilitySet> availabilitySets = subscription.availabilitySets().asMap("<resource-group-name>");
Using its resource id:
AvailabilitySet availabilitySet = subscription.availabilitySets("<resource-id>");
or:
AvailabilitySet availabilitySet = subscription.availabilitySets().get("<resource-id>");
Using its resource group and name:
AvailabilitySet availabilitySet = subscription.availabilitySets("<resource-group-name>", "<availability-set-name>");
or:
AvailabilitySet availabilitySet = subscription.availabilitySets().get("<resource-group-name>", "<availability-set-name>");
Any of the following approaches:
subscription.availabilitySets().delete("<resource-id>");
subscription.availabilitySets().delete("<resource-group-name>", "<availability-set-name>");
subscription.availabilitySets("<resource-id>").delete();
subscription.availabilitySets("<resource-group-name>", "<availability-set-name>").delete();