Merge pull request #1503 from jclouds/closeable-api

expose Closeable, top-level apis via ContextBuilder.buildApi()
This commit is contained in:
Adrian Cole 2013-04-08 18:01:28 -07:00
commit 76066a2678
56 changed files with 281 additions and 64 deletions

View File

@ -20,6 +20,7 @@ package org.jclouds.atmos;
import static com.google.common.net.HttpHeaders.EXPECT;
import java.io.Closeable;
import java.net.URI;
import javax.inject.Named;
@ -77,7 +78,7 @@ import com.google.inject.Provides;
*/
@RequestFilters(SignRequest.class)
@Path("/rest/namespace")
public interface AtmosAsyncClient {
public interface AtmosAsyncClient extends Closeable {
/**
* Creates a default implementation of AtmosObject
*/

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.atmos;
import java.io.Closeable;
import java.net.URI;
import org.jclouds.atmos.domain.AtmosObject;
import org.jclouds.atmos.domain.BoundedSet;
@ -38,7 +39,7 @@ import com.google.inject.Provides;
* @see <a href="https://community.emc.com/community/labs/atmos_online" />
* @author Adrian Cole
*/
public interface AtmosClient {
public interface AtmosClient extends Closeable {
/**
* Creates a default implementation of AtmosObject
*/

View File

@ -22,6 +22,7 @@ import static com.google.common.base.Preconditions.checkNotNull;
import static com.google.common.util.concurrent.Futures.immediateFailedFuture;
import static com.google.common.util.concurrent.Futures.immediateFuture;
import java.io.IOException;
import java.net.URI;
import java.util.concurrent.ExecutionException;
@ -47,6 +48,7 @@ import org.jclouds.blobstore.domain.Blob;
import org.jclouds.blobstore.domain.BlobMetadata;
import org.jclouds.blobstore.functions.HttpGetOptionsListToGetOptions;
import org.jclouds.http.options.GetOptions;
import org.jclouds.lifecycle.Closer;
import com.google.common.base.Function;
import com.google.common.base.Throwables;
@ -69,13 +71,14 @@ public class StubAtmosAsyncClient implements AtmosAsyncClient {
private final ListOptionsToBlobStoreListOptions container2ContainerListOptions;
private final ResourceMetadataListToDirectoryEntryList resource2ObjectList;
private final ListeningExecutorService userExecutor;
private final Closer closer;
@Inject
private StubAtmosAsyncClient(LocalAsyncBlobStore blobStore, AtmosObject.Factory objectProvider,
HttpGetOptionsListToGetOptions httpGetOptionsConverter, ObjectToBlob object2Blob, BlobToObject blob2Object,
BlobMetadataToObject blob2ObjectInfo, ListOptionsToBlobStoreListOptions container2ContainerListOptions,
@Named(Constants.PROPERTY_USER_THREADS) ListeningExecutorService userExecutor,
ResourceMetadataListToDirectoryEntryList resource2ContainerList) {
ResourceMetadataListToDirectoryEntryList resource2ContainerList, Closer closer) {
this.blobStore = blobStore;
this.objectProvider = objectProvider;
this.httpGetOptionsConverter = httpGetOptionsConverter;
@ -86,6 +89,7 @@ public class StubAtmosAsyncClient implements AtmosAsyncClient {
"container2ContainerListOptions");
this.resource2ObjectList = checkNotNull(resource2ContainerList, "resource2ContainerList");
this.userExecutor = userExecutor;
this.closer = checkNotNull(closer, "closer");
}
@Override
@ -245,4 +249,8 @@ public class StubAtmosAsyncClient implements AtmosAsyncClient {
throw new UnsupportedOperationException();
}
@Override
public void close() throws IOException {
closer.close();
}
}

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.cloudservers;
import java.io.Closeable;
import java.util.Set;
import java.util.concurrent.ExecutionException;
@ -76,7 +77,7 @@ import com.google.common.util.concurrent.ListenableFuture;
*/
@RequestFilters({ AuthenticateRequest.class, AddTimestampQuery.class })
@Endpoint(Compute.class)
public interface CloudServersAsyncClient {
public interface CloudServersAsyncClient extends Closeable {
/**
* @see CloudServersClient#getLimits

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.cloudservers;
import java.io.Closeable;
import java.util.Set;
import javax.ws.rs.PathParam;
@ -44,7 +45,7 @@ import org.jclouds.cloudservers.options.RebuildServerOptions;
* @see <a href="http://docs.rackspacecloud.com/servers/api/cs-devguide-latest.pdf" />
* @author Adrian Cole
*/
public interface CloudServersClient {
public interface CloudServersClient extends Closeable {
/**
* All accounts, by default, have a preconfigured set of thresholds (or limits) to manage
* capacity and prevent abuse of the system. The system recognizes two kinds of limits: rate

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.cloudsigma;
import java.io.Closeable;
import java.util.Set;
import javax.ws.rs.Consumes;
@ -75,7 +76,7 @@ import com.google.common.util.concurrent.ListenableFuture;
*/
@RequestFilters(BasicAuthentication.class)
@Consumes(MediaType.TEXT_PLAIN)
public interface CloudSigmaAsyncClient {
public interface CloudSigmaAsyncClient extends Closeable {
/**
* @see CloudSigmaClient#listStandardDrives

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.cloudsigma;
import java.io.Closeable;
import java.util.Set;
import org.jclouds.cloudsigma.domain.Drive;
@ -38,7 +39,7 @@ import org.jclouds.cloudsigma.options.CloneDriveOptions;
* @see <a href="TODO: insert URL of cloudsigma documentation" />
* @author Adrian Cole
*/
public interface CloudSigmaClient {
public interface CloudSigmaClient extends Closeable {
/**
* Get profile info

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.cloudwatch;
import java.io.Closeable;
import java.util.Set;
import org.jclouds.cloudwatch.features.MetricApi;
@ -37,7 +38,7 @@ import com.google.inject.Provides;
* />
* @author Adrian Cole
*/
public interface CloudWatchApi {
public interface CloudWatchApi extends Closeable {
/**
*
* @return the Region codes configured

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.cloudwatch;
import java.io.Closeable;
import java.util.Set;
import org.jclouds.cloudwatch.features.MetricAsyncApi;
@ -38,7 +39,7 @@ import com.google.inject.Provides;
* />
* @author Adrian Cole
*/
public interface CloudWatchAsyncApi {
public interface CloudWatchAsyncApi extends Closeable {
/**
*
* @return the Region codes configured

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.ec2;
import java.io.Closeable;
import java.util.Set;
import org.jclouds.ec2.features.SubnetApi;
import org.jclouds.ec2.features.TagApi;
@ -50,7 +51,7 @@ import com.google.inject.Provides;
* @author Adrian Cole
* @see EC2AsyncApi
*/
public interface EC2Api {
public interface EC2Api extends Closeable {
/**
*
* @return the Region codes configured

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.ec2;
import java.io.Closeable;
import java.util.Set;
import org.jclouds.ec2.features.SubnetAsyncApi;
@ -38,7 +39,7 @@ import com.google.inject.Provides;
*
* @author Adrian Cole
*/
public interface EC2AsyncApi {
public interface EC2AsyncApi extends Closeable {
/**
*
* @return the Region codes configured

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.elasticstack;
import java.io.Closeable;
import java.util.Set;
import javax.ws.rs.Consumes;
@ -64,7 +65,7 @@ import com.google.common.util.concurrent.ListenableFuture;
*/
@RequestFilters(BasicAuthentication.class)
@Consumes(MediaType.TEXT_PLAIN)
public interface ElasticStackAsyncClient {
public interface ElasticStackAsyncClient extends Closeable {
/**
* @see ElasticStackClient#listServers()

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.elasticstack;
import java.io.Closeable;
import java.util.Set;
import org.jclouds.elasticstack.domain.Drive;
import org.jclouds.elasticstack.domain.DriveData;
@ -35,7 +36,7 @@ import org.jclouds.io.Payload;
* @see <a href="TODO: insert URL of elasticstack documentation" />
* @author Adrian Cole
*/
public interface ElasticStackClient {
public interface ElasticStackClient extends Closeable {
/**
* list of server uuids in your account
*

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.openstack.cinder.v1;
import java.io.Closeable;
import java.util.Set;
import org.jclouds.javax.annotation.Nullable;
import org.jclouds.location.Zone;
@ -38,7 +39,7 @@ import com.google.inject.Provides;
* @see <a href="http://api.openstack.org/">API Doc</a>
* @author Everett Toews
*/
public interface CinderApi {
public interface CinderApi extends Closeable {
/**
* @return the Zone codes configured
*/

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.openstack.cinder.v1;
import java.io.Closeable;
import java.util.Set;
import org.jclouds.javax.annotation.Nullable;
@ -39,7 +40,7 @@ import com.google.inject.Provides;
* @see <a href="http://api.openstack.org/">API Doc</a>
* @author Everett Toews
*/
public interface CinderAsyncApi {
public interface CinderAsyncApi extends Closeable {
/**
* @return the Zone codes configured
*/

View File

@ -18,6 +18,8 @@
*/
package org.jclouds.openstack.keystone.v2_0;
import java.io.Closeable;
import org.jclouds.openstack.keystone.v2_0.domain.ApiMetadata;
import org.jclouds.openstack.keystone.v2_0.features.ServiceApi;
import org.jclouds.openstack.keystone.v2_0.features.TenantApi;
@ -36,7 +38,7 @@ import com.google.common.base.Optional;
* @see <a href="http://keystone.openstack.org/" />
* @see KeystoneAsyncApi
*/
public interface KeystoneApi {
public interface KeystoneApi extends Closeable {
/**
* Discover API version information, links to documentation (PDF, HTML, WADL), and supported media types

View File

@ -18,6 +18,8 @@
*/
package org.jclouds.openstack.keystone.v2_0;
import java.io.Closeable;
import javax.ws.rs.Consumes;
import javax.ws.rs.GET;
import javax.ws.rs.core.MediaType;
@ -44,7 +46,7 @@ import com.google.common.util.concurrent.ListenableFuture;
* @see <a href="http://keystone.openstack.org/" />
* @see KeystoneApi
*/
public interface KeystoneAsyncApi {
public interface KeystoneAsyncApi extends Closeable {
/**
* @see KeystoneApi#getApiMetadata()

View File

@ -6,6 +6,7 @@ import static org.jclouds.openstack.keystone.v2_0.config.KeystoneProperties.CRED
import static org.jclouds.openstack.keystone.v2_0.config.KeystoneProperties.SERVICE_TYPE;
import static org.testng.Assert.assertTrue;
import java.io.Closeable;
import java.lang.annotation.Retention;
import java.lang.annotation.Target;
import java.net.URI;
@ -62,12 +63,12 @@ public class ProviderModuleExpectTest extends BaseRestApiExpectTest<ProviderModu
}
}
static interface DNSApi {
static interface DNSApi extends Closeable {
boolean zoneExists(@PathParam("zoneName") String zoneName);
}
@org.jclouds.rest.annotations.Endpoint(DNS.class)
static interface DNSAsyncApi {
static interface DNSAsyncApi extends Closeable {
@HEAD
@Path("/zones/{zoneName}")
@Fallback(FalseOnNotFoundOr404.class)

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.openstack.nova.v2_0;
import java.io.Closeable;
import java.util.Set;
import org.jclouds.javax.annotation.Nullable;
import org.jclouds.location.Zone;
@ -56,7 +57,7 @@ import com.google.inject.Provides;
* />
* @author Adrian Cole
*/
public interface NovaApi {
public interface NovaApi extends Closeable {
/**
*
* @return the Zone codes configured

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.openstack.nova.v2_0;
import java.io.Closeable;
import java.util.Set;
import org.jclouds.javax.annotation.Nullable;
@ -57,7 +58,7 @@ import com.google.inject.Provides;
* />
* @author Adrian Cole
*/
public interface NovaAsyncApi {
public interface NovaAsyncApi extends Closeable {
/**
*

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.rackspace.cloudloadbalancers.v1;
import java.io.Closeable;
import java.util.Set;
import javax.ws.rs.Path;
import javax.ws.rs.PathParam;
@ -48,7 +49,7 @@ import com.google.inject.Provides;
* @see CloudLoadBalancersAsyncApi
* @author Adrian Cole
*/
public interface CloudLoadBalancersApi {
public interface CloudLoadBalancersApi extends Closeable {
/**
* @return the Zone codes configured
*/

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.rackspace.cloudloadbalancers.v1;
import java.io.Closeable;
import java.util.Set;
import javax.ws.rs.Path;
@ -49,7 +50,7 @@ import com.google.inject.Provides;
* @see CloudLoadBalancersApi
* @author Adrian Cole
*/
public interface CloudLoadBalancersAsyncApi {
public interface CloudLoadBalancersAsyncApi extends Closeable {
/**
* @return the Zone codes configured
*/

View File

@ -18,6 +18,8 @@
*/
package org.jclouds.route53;
import java.io.Closeable;
import javax.ws.rs.PathParam;
import org.jclouds.rest.annotations.Delegate;
@ -34,7 +36,7 @@ import org.jclouds.route53.features.HostedZoneApi;
* />
* @author Adrian Cole
*/
public interface Route53Api {
public interface Route53Api extends Closeable {
/**
* returns the current status of a change batch request.

View File

@ -18,6 +18,8 @@
*/
package org.jclouds.route53;
import java.io.Closeable;
import javax.inject.Named;
import javax.ws.rs.GET;
import javax.ws.rs.Path;
@ -48,7 +50,7 @@ import com.google.common.util.concurrent.ListenableFuture;
@RequestFilters(RestAuthentication.class)
@VirtualHost
@Path("/{jclouds.api-version}")
public interface Route53AsyncApi {
public interface Route53AsyncApi extends Closeable {
/**
* @see Route53Api#getChange()

View File

@ -21,6 +21,7 @@ package org.jclouds.s3;
import static com.google.common.net.HttpHeaders.EXPECT;
import static org.jclouds.blobstore.attr.BlobScopes.CONTAINER;
import java.io.Closeable;
import java.util.Set;
import java.util.concurrent.ExecutionException;
@ -108,7 +109,7 @@ import com.google.inject.Provides;
*/
@RequestFilters(RequestAuthorizeSignature.class)
@BlobScope(CONTAINER)
public interface S3AsyncClient {
public interface S3AsyncClient extends Closeable {
public static final String VERSION = "2006-03-01";
/**

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.s3;
import java.io.Closeable;
import java.util.Set;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Future;
@ -49,7 +50,7 @@ import com.google.inject.Provides;
* @author James Murty
* @see <a href="http://docs.amazonwebservices.com/AmazonS3/2006-03-01/RESTAPI.html" />
*/
public interface S3Client {
public interface S3Client extends Closeable {
/**
* Creates a default implementation of S3Object

View File

@ -23,6 +23,7 @@ import static com.google.common.util.concurrent.Futures.immediateFailedFuture;
import static com.google.common.util.concurrent.Futures.immediateFuture;
import static com.google.common.util.concurrent.Futures.transform;
import java.io.IOException;
import java.util.Date;
import java.util.Map;
import java.util.Set;
@ -50,6 +51,7 @@ import org.jclouds.domain.LocationBuilder;
import org.jclouds.domain.LocationScope;
import org.jclouds.http.options.GetOptions;
import org.jclouds.javax.annotation.Nullable;
import org.jclouds.lifecycle.Closer;
import org.jclouds.s3.S3AsyncClient;
import org.jclouds.s3.blobstore.S3AsyncBlobStore;
import org.jclouds.s3.blobstore.functions.BlobToObject;
@ -100,6 +102,7 @@ public class StubS3AsyncClient implements S3AsyncClient {
private final ConcurrentMap<String, ConcurrentMap<String, Blob>> containerToBlobs;
private final ConcurrentMap<String, Location> containerToLocation;
private final ListeningExecutorService userExecutor;
private final Closer closer;
@Inject
private StubS3AsyncClient(@Named(Constants.PROPERTY_USER_THREADS) ListeningExecutorService userExecutor,
@ -108,7 +111,7 @@ public class StubS3AsyncClient implements S3AsyncClient {
S3Object.Factory objectProvider, Blob.Factory blobProvider,
HttpGetOptionsListToGetOptions httpGetOptionsConverter, ObjectToBlob object2Blob, BlobToObject blob2Object,
BlobToObjectMetadata blob2ObjectMetadata, BucketToContainerListOptions bucket2ContainerListOptions,
ResourceToBucketList resource2BucketList) {
ResourceToBucketList resource2BucketList, Closer closer) {
this.userExecutor = userExecutor;
this.containerToBlobs = containerToBlobs;
this.containerToLocation = containerToLocation;
@ -122,6 +125,7 @@ public class StubS3AsyncClient implements S3AsyncClient {
this.blob2ObjectMetadata = checkNotNull(blob2ObjectMetadata, "blob2ObjectMetadata");
this.bucket2ContainerListOptions = checkNotNull(bucket2ContainerListOptions, "bucket2ContainerListOptions");
this.resource2BucketList = checkNotNull(resource2BucketList, "resource2BucketList");
this.closer = checkNotNull(closer, "closer");
}
public static final String TEST_ACL_ID = "1a405254c932b52e5b5caaa88186bc431a1bacb9ece631f835daddaf0c47677c";
@ -336,4 +340,9 @@ public class StubS3AsyncClient implements S3AsyncClient {
return immediateFuture(containerToBlobs.get(bucketName).containsKey(key));
}
@Override
public void close() throws IOException {
closer.close();
}
}

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.sqs;
import java.io.Closeable;
import java.net.URI;
import java.util.Set;
import org.jclouds.javax.annotation.Nullable;
@ -40,7 +41,7 @@ import com.google.inject.Provides;
* @see SQSAsyncApi
*/
@Beta
public interface SQSApi {
public interface SQSApi extends Closeable {
/**
*

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.sqs;
import java.io.Closeable;
import java.net.URI;
import java.util.Set;
@ -48,7 +49,7 @@ import com.google.inject.Provides;
@Beta
@RequestFilters(FormSigner.class)
@VirtualHost
public interface SQSAsyncApi {
public interface SQSAsyncApi extends Closeable {
/**
*
* @return the Region codes configured

View File

@ -23,6 +23,7 @@ import static org.jclouds.providers.AnonymousProviderMetadata.forClientMappedToA
import static org.jclouds.sqs.reference.SQSParameters.ACTION;
import static org.testng.Assert.assertEquals;
import java.io.Closeable;
import java.net.URI;
import javax.ws.rs.POST;
@ -62,11 +63,11 @@ public class PermissionApiLiveTest extends BaseSQSApiLiveTest {
recreateQueueInRegion(prefix, null);
}
static interface AnonymousAttributesApi {
static interface AnonymousAttributesApi extends Closeable {
String getQueueArn();
}
static interface AnonymousAttributesAsyncApi {
static interface AnonymousAttributesAsyncApi extends Closeable {
@POST
@Path("/")
@FormParams(keys = { ACTION, "AttributeName.1" }, values = { "GetQueueAttributes", "QueueArn" })

View File

@ -18,6 +18,8 @@
*/
package org.jclouds.sts;
import java.io.Closeable;
import org.jclouds.aws.domain.SessionCredentials;
import org.jclouds.sts.domain.User;
import org.jclouds.sts.domain.UserAndSessionCredentials;
@ -33,7 +35,7 @@ import org.jclouds.sts.options.SessionCredentialsOptions;
* @see <a href="http://docs.amazonwebservices.com/STS/latest/APIReference" />
* @author Adrian Cole
*/
public interface STSApi {
public interface STSApi extends Closeable {
/**
* Returns a set of temporary credentials for an AWS account or IAM user,
* with a default timeout

View File

@ -18,6 +18,8 @@
*/
package org.jclouds.sts;
import java.io.Closeable;
import javax.inject.Named;
import javax.ws.rs.FormParam;
import javax.ws.rs.POST;
@ -47,7 +49,7 @@ import com.google.common.util.concurrent.ListenableFuture;
*/
@RequestFilters(FormSigner.class)
@VirtualHost
public interface STSAsyncApi {
public interface STSAsyncApi extends Closeable {
/**
* @see STSApi#createTemporaryCredentials()

View File

@ -20,6 +20,7 @@ package org.jclouds.openstack.swift;
import static com.google.common.net.HttpHeaders.EXPECT;
import java.io.Closeable;
import java.util.Map;
import java.util.Set;
@ -79,7 +80,7 @@ import com.google.inject.Provides;
* @see <a href="http://www.rackspacecloud.com/cf-devguide-20090812.pdf" />
* @author Adrian Cole
*/
public interface CommonSwiftAsyncClient {
public interface CommonSwiftAsyncClient extends Closeable {
@Provides
SwiftObject newSwiftObject();

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.openstack.swift;
import java.io.Closeable;
import java.util.Map;
import java.util.Set;
import org.jclouds.blobstore.domain.PageSet;
@ -38,7 +39,7 @@ import com.google.inject.Provides;
* @see <a href="http://www.rackspacecloud.com/cf-devguide-20090812.pdf" />
* @author Adrian Cole
*/
public interface CommonSwiftClient {
public interface CommonSwiftClient extends Closeable {
@Provides
SwiftObject newSwiftObject();

View File

@ -22,6 +22,7 @@ import static com.google.common.base.Preconditions.checkNotNull;
import static com.google.common.util.concurrent.Futures.immediateFuture;
import static com.google.common.util.concurrent.Futures.transform;
import java.io.IOException;
import java.net.URI;
import java.util.HashMap;
import java.util.Map;
@ -40,6 +41,7 @@ import org.jclouds.blobstore.domain.StorageMetadata;
import org.jclouds.blobstore.functions.HttpGetOptionsListToGetOptions;
import org.jclouds.blobstore.options.ListContainerOptions;
import org.jclouds.http.options.GetOptions;
import org.jclouds.lifecycle.Closer;
import org.jclouds.openstack.swift.CommonSwiftAsyncClient;
import org.jclouds.openstack.swift.SwiftAsyncClient;
import org.jclouds.openstack.swift.blobstore.functions.BlobToObject;
@ -77,6 +79,7 @@ public class StubSwiftAsyncClient implements CommonSwiftAsyncClient {
private final ListContainerOptionsToBlobStoreListContainerOptions container2ContainerListOptions;
private final ResourceToObjectList resource2ObjectList;
private final ListeningExecutorService userExecutor;
private final Closer closer;
@Inject
private StubSwiftAsyncClient(@Named(Constants.PROPERTY_USER_THREADS) ListeningExecutorService userExecutor,
@ -84,7 +87,7 @@ public class StubSwiftAsyncClient implements CommonSwiftAsyncClient {
SwiftObject.Factory objectProvider, HttpGetOptionsListToGetOptions httpGetOptionsConverter,
ObjectToBlob object2Blob, BlobToObject blob2Object, ResourceToObjectInfo blob2ObjectInfo,
ListContainerOptionsToBlobStoreListContainerOptions container2ContainerListOptions,
ResourceToObjectList resource2ContainerList) {
ResourceToObjectList resource2ContainerList, Closer closer) {
this.userExecutor = userExecutor;
this.blobStore = blobStore;
this.objectProvider = objectProvider;
@ -95,6 +98,7 @@ public class StubSwiftAsyncClient implements CommonSwiftAsyncClient {
this.container2ContainerListOptions = checkNotNull(container2ContainerListOptions,
"container2ContainerListOptions");
this.resource2ObjectList = checkNotNull(resource2ContainerList, "resource2ContainerList");
this.closer = checkNotNull(closer, "closer");
}
public ListenableFuture<Boolean> containerExists(final String container) {
@ -217,4 +221,9 @@ public class StubSwiftAsyncClient implements CommonSwiftAsyncClient {
public ListenableFuture<Boolean> objectExists(String bucketName, String key) {
return blobStore.blobExists(bucketName, key);
}
@Override
public void close() throws IOException {
closer.close();
}
}

View File

@ -18,6 +18,8 @@
*/
package org.jclouds.vcloud;
import java.io.Closeable;
import org.jclouds.rest.annotations.Delegate;
import org.jclouds.rest.annotations.RequestFilters;
import org.jclouds.vcloud.features.CatalogAsyncClient;
@ -39,7 +41,7 @@ import org.jclouds.vcloud.filters.AddVCloudAuthorizationAndCookieToRequest;
* @author Adrian Cole
*/
@RequestFilters(AddVCloudAuthorizationAndCookieToRequest.class)
public interface VCloudAsyncClient {
public interface VCloudAsyncClient extends Closeable {
/**
* Provides asynchronous access to VApp Template features.

View File

@ -18,6 +18,8 @@
*/
package org.jclouds.vcloud;
import java.io.Closeable;
import org.jclouds.rest.annotations.Delegate;
import org.jclouds.vcloud.features.CatalogClient;
import org.jclouds.vcloud.features.NetworkClient;
@ -35,7 +37,7 @@ import org.jclouds.vcloud.features.VmClient;
* @see <a href="http://communities.vmware.com/community/developer/forums/vcloudapi" />
* @author Adrian Cole
*/
public interface VCloudClient {
public interface VCloudClient extends Closeable {
/**
* Provides asynchronous access to VApp Template features.
*

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.vcloud;
import java.io.Closeable;
import java.net.URI;
import java.util.SortedMap;
@ -36,7 +37,7 @@ import com.google.common.util.concurrent.ListenableFuture;
* @see <a href="https://community.vcloudexpress.terremark.com/en-us/discussion_forums/f/60.aspx" />
* @author Adrian Cole
*/
public interface VCloudVersionsAsyncClient {
public interface VCloudVersionsAsyncClient extends Closeable {
/**
* Retrieve information for supported versions

View File

@ -18,9 +18,10 @@
*/
package org.jclouds.vcloud;
import java.io.Closeable;
import java.net.URI;
import java.util.SortedMap;
public interface VCloudVersionsClient {
public interface VCloudVersionsClient extends Closeable {
SortedMap<String, URI> getSupportedVersions();
}

View File

@ -18,6 +18,8 @@
*/
package org.jclouds.vcloud.internal;
import java.io.Closeable;
import javax.ws.rs.Consumes;
import javax.ws.rs.POST;
@ -40,7 +42,7 @@ import com.google.common.util.concurrent.ListenableFuture;
*/
@Endpoint(org.jclouds.vcloud.endpoints.VCloudLogin.class)
@RequestFilters(BasicAuthentication.class)
public interface VCloudLoginAsyncClient {
public interface VCloudLoginAsyncClient extends Closeable {
/**
* This request returns a token to use in subsequent requests. After 30 minutes of inactivity,

View File

@ -18,9 +18,11 @@
*/
package org.jclouds.vcloud.internal;
import java.io.Closeable;
import org.jclouds.vcloud.domain.VCloudSession;
public interface VCloudLoginClient {
public interface VCloudLoginClient extends Closeable {
VCloudSession login();
}

View File

@ -34,6 +34,7 @@ import static org.jclouds.trmk.vcloud_0_8.TerremarkVCloudMediaType.VAPPTEMPLATE_
import static org.jclouds.trmk.vcloud_0_8.TerremarkVCloudMediaType.VAPP_XML;
import static org.jclouds.trmk.vcloud_0_8.TerremarkVCloudMediaType.VDC_XML;
import java.io.Closeable;
import java.net.URI;
import java.util.Map;
import java.util.Set;
@ -124,7 +125,7 @@ import com.google.inject.Provides;
* @author Adrian Cole
*/
@RequestFilters(SetVCloudTokenCookie.class)
public interface TerremarkVCloudAsyncClient {
public interface TerremarkVCloudAsyncClient extends Closeable {
/**
* @see TerremarkVCloudClient#getCatalogItemInOrg

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.trmk.vcloud_0_8;
import java.io.Closeable;
import java.net.URI;
import java.util.Map;
import java.util.Set;
@ -55,7 +56,7 @@ import com.google.inject.Provides;
* />
* @author Adrian Cole
*/
public interface TerremarkVCloudClient {
public interface TerremarkVCloudClient extends Closeable {
Catalog getCatalog(URI catalogId);
/**

View File

@ -18,6 +18,8 @@
*/
package org.jclouds.trmk.vcloud_0_8.internal;
import java.io.Closeable;
import javax.ws.rs.Consumes;
import javax.ws.rs.POST;
@ -40,7 +42,7 @@ import com.google.common.util.concurrent.ListenableFuture;
*/
@Endpoint(org.jclouds.trmk.vcloud_0_8.endpoints.VCloudLogin.class)
@RequestFilters(BasicAuthentication.class)
public interface TerremarkVCloudLoginAsyncClient {
public interface TerremarkVCloudLoginAsyncClient extends Closeable {
/**
* This request returns a token to use in subsequent requests. After ten minutes of inactivity,

View File

@ -18,9 +18,11 @@
*/
package org.jclouds.trmk.vcloud_0_8.internal;
import java.io.Closeable;
import org.jclouds.trmk.vcloud_0_8.domain.VCloudSession;
public interface TerremarkVCloudLoginClient {
public interface TerremarkVCloudLoginClient extends Closeable {
VCloudSession login();
}

View File

@ -18,6 +18,7 @@
*/
package org.jclouds.trmk.vcloud_0_8.internal;
import java.io.Closeable;
import java.net.URI;
import java.util.SortedMap;
@ -36,7 +37,7 @@ import com.google.common.util.concurrent.ListenableFuture;
* @see <a href="https://community.vcloudexpress.terremark.com/en-us/discussion_forums/f/60.aspx" />
* @author Adrian Cole
*/
public interface TerremarkVCloudVersionsAsyncClient {
public interface TerremarkVCloudVersionsAsyncClient extends Closeable {
/**
* Retrieve information for supported versions

View File

@ -18,9 +18,10 @@
*/
package org.jclouds.trmk.vcloud_0_8.internal;
import java.io.Closeable;
import java.net.URI;
import java.util.SortedMap;
public interface TerremarkVCloudVersionsClient {
public interface TerremarkVCloudVersionsClient extends Closeable {
SortedMap<String, URI> getSupportedVersions();
}

View File

@ -45,6 +45,7 @@ import static org.jclouds.Constants.PROPERTY_PROVIDER;
import static org.jclouds.reflect.Reflection2.typeToken;
import static org.jclouds.util.Throwables2.propagateAuthorizationOrOriginalException;
import java.io.Closeable;
import java.util.List;
import java.util.Map;
import java.util.NoSuchElementException;
@ -611,6 +612,20 @@ public class ContextBuilder {
return (C) buildInjector().getInstance(Key.get(TypeLiteral.get(returnType.getType())));
}
/**
* This will return the top-level interface for the api or provider.
*
* Ex.
* <pre>
* api = ContextBuilder.newBuilder("openstack-nova")
* ...
* .buildApi(NovaApi.class);
*</pre>
*/
public <A extends Closeable> A buildApi(Class<A> api) {
return buildInjector().getInstance(api);
}
public ApiMetadata getApiMetadata() {
return apiMetadata;
}

View File

@ -27,6 +27,7 @@ import static org.jclouds.reflect.Reflection2.method;
import static org.jclouds.reflect.Reflection2.methods;
import static org.jclouds.rest.config.BinderUtils.bindHttpApi;
import java.io.Closeable;
import java.net.Proxy;
import java.net.URI;
import java.util.Map;
@ -120,12 +121,27 @@ public class RestModule extends AbstractModule {
public static void putInvokables(Class<?> sync, Class<?> async, Cache<Invokable<?, ?>, Invokable<?, ?>> cache) {
for (Invokable<?, ?> invoked : methods(sync)) {
Invokable<?, ?> delegatedMethod = method(async, invoked.getName(), getParameterTypes(invoked));
checkArgument(delegatedMethod.getExceptionTypes().equals(invoked.getExceptionTypes()),
checkArgument(
delegatedMethod.getExceptionTypes().equals(invoked.getExceptionTypes()) || isCloseable(delegatedMethod),
"invoked %s has different typed exceptions than target %s", invoked, delegatedMethod);
cache.put(invoked, delegatedMethod);
}
}
/**
* In JDK7 Closeable.close is declared in AutoCloseable, which throws
* Exception vs IOException, so we have to be more lenient about exception
* type declarations.
*
* <h4>note</h4>
*
* This will be refactored out when we delete Async code in jclouds 1.7.
*/
private static boolean isCloseable(Invokable<?, ?> delegatedMethod) {
return "close".equals(delegatedMethod.getName())
&& Closeable.class.isAssignableFrom(delegatedMethod.getDeclaringClass());
}
/**
* for portability with {@link Class#getMethod(String, Class...)}
*/

View File

@ -30,6 +30,7 @@ import static org.jclouds.util.Optionals2.unwrapIfOptional;
import static org.jclouds.util.Throwables2.getFirstThrowableOfType;
import static org.jclouds.util.Throwables2.propagateIfPossible;
import java.io.Closeable;
import java.lang.annotation.Annotation;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
@ -44,6 +45,7 @@ import javax.inject.Inject;
import javax.inject.Qualifier;
import org.jclouds.javax.annotation.Nullable;
import org.jclouds.lifecycle.Closer;
import org.jclouds.reflect.FunctionalReflection;
import org.jclouds.reflect.Invocation;
import org.jclouds.reflect.InvocationSuccess;
@ -129,12 +131,34 @@ public final class DelegatesToInvocationFunction<S, F extends Function<Invocatio
}
}
private static final Invokable<?, ?> CLOSE;
static {
try {
CLOSE = Invokable.from(Closeable.class.getMethod("close"));
} catch (SecurityException e) {
throw propagate(e);
} catch (NoSuchMethodException e) {
throw propagate(e);
}
}
protected Object handle(Invocation invocation) {
if (invocation.getInvokable().isAnnotationPresent(Provides.class))
return lookupValueFromGuice(invocation.getInvokable());
else if (invocation.getInvokable().isAnnotationPresent(Delegate.class))
Invokable<?, ?> invokable = invocation.getInvokable();
if (CLOSE.equals(invokable)) {
try {
injector.getInstance(Closer.class).close();
return null;
} catch (Throwable e) {
throw propagate(e);
}
} else if (invokable.isAnnotationPresent(Provides.class)) {
return lookupValueFromGuice(invokable);
} else if (invokable.isAnnotationPresent(Delegate.class)) {
return propagateContextToDelegate(invocation);
return methodInvoker.apply(invocation);
} else {
return methodInvoker.apply(invocation);
}
}
private final Injector injector;

View File

@ -20,6 +20,7 @@ package org.jclouds.http;
import static com.google.common.util.concurrent.Futures.immediateFuture;
import java.io.Closeable;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
@ -61,7 +62,7 @@ import com.google.inject.Provides;
* @see IntegrationTestClient
* @author Adrian Cole
*/
public interface IntegrationTestAsyncClient {
public interface IntegrationTestAsyncClient extends Closeable {
@Target({ ElementType.METHOD })
@Retention(RetentionPolicy.RUNTIME)
@HttpMethod("ROWDY")

View File

@ -18,6 +18,8 @@
*/
package org.jclouds.http;
import java.io.Closeable;
import org.jclouds.http.options.HttpRequestOptions;
import org.jclouds.io.Payload;
@ -29,7 +31,7 @@ import com.google.inject.Provides;
*
* @author Adrian Cole
*/
public interface IntegrationTestClient {
public interface IntegrationTestClient extends Closeable {
String rowdy(String path);
boolean exists(String path);

View File

@ -0,0 +1,73 @@
/**
* Licensed to jclouds, Inc. (jclouds) under one or more
* contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. jclouds 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
*
* 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.jclouds.rest.annotationparsing;
import static org.easymock.EasyMock.createMock;
import static org.easymock.EasyMock.expect;
import static org.easymock.EasyMock.replay;
import static org.easymock.EasyMock.verify;
import static org.jclouds.providers.AnonymousProviderMetadata.forClientMappedToAsyncClientOnEndpoint;
import java.io.Closeable;
import java.io.IOException;
import org.jclouds.ContextBuilder;
import org.jclouds.concurrent.config.ExecutorServiceModule;
import org.jclouds.providers.ProviderMetadata;
import org.testng.annotations.Test;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableSet;
import com.google.common.util.concurrent.ListeningExecutorService;
import com.google.inject.Module;
/**
* ensures that jclouds can be operated w/o reference to a context as the Api
* itself is closeable.
*
* @author Adrian Cole
*/
@Test(groups = "unit", testName = "ClosableApiTest")
public class ClosableApiTest {
static interface DelegatingApi extends Closeable {
}
static interface DelegatingAsyncApi extends Closeable {
}
ProviderMetadata provider = forClientMappedToAsyncClientOnEndpoint(DelegatingApi.class, DelegatingAsyncApi.class,
"http://mock");
public void testApiClosesExecutorServiceOnClose() throws IOException {
ListeningExecutorService executor = createMock(ListeningExecutorService.class);
expect(executor.shutdownNow()).andReturn(ImmutableList.<Runnable> of()).atLeastOnce();
replay(executor);
DelegatingApi api = ContextBuilder.newBuilder(provider)
.modules(ImmutableSet.<Module> builder()
.add(new ExecutorServiceModule(executor, executor))
.build())
.buildApi(DelegatingApi.class);
api.close();
verify(executor);
}
}

View File

@ -22,6 +22,8 @@ import static org.jclouds.providers.AnonymousProviderMetadata.forClientMappedToA
import static org.testng.Assert.assertEquals;
import static org.testng.Assert.assertTrue;
import java.io.Closeable;
import javax.ws.rs.Consumes;
import javax.ws.rs.GET;
import javax.ws.rs.Path;
@ -68,7 +70,7 @@ public class JAXBResponseParserAnnotationExpectTest extends
}
public interface TestJAXBApi {
public interface TestJAXBApi extends Closeable {
public TestJAXBDomain jaxbGetWithAnnotation();
public Object jaxbGetWithAnnotationAndCustomClass();
@ -78,7 +80,7 @@ public class JAXBResponseParserAnnotationExpectTest extends
public String jaxbGetWithTransformer();
}
public interface TestJAXBAsyncApi {
public interface TestJAXBAsyncApi extends Closeable {
@GET
@Path("/jaxb/annotation")
@JAXBResponseParser

View File

@ -21,6 +21,7 @@ package org.jclouds.rest.annotationparsing;
import static org.jclouds.providers.AnonymousProviderMetadata.forClientMappedToAsyncClientOnEndpoint;
import static org.testng.Assert.assertEquals;
import java.io.Closeable;
import java.util.NoSuchElementException;
import java.util.Set;
@ -50,7 +51,7 @@ import com.google.inject.name.Names;
@Test(groups = "unit", testName = "ProvidesAnnotationExpectTest")
public class ProvidesAnnotationExpectTest extends BaseRestClientExpectTest<ProvidesAnnotationExpectTest.ProvidingApi> {
static interface ProvidingApi {
static interface ProvidingApi extends Closeable {
@Provides
Set<String> set();
@ -67,7 +68,7 @@ public class ProvidesAnnotationExpectTest extends BaseRestClientExpectTest<Provi
Set<String> noSuchElementException();
}
static interface ProvidingAsyncApi {
static interface ProvidingAsyncApi extends Closeable {
@Provides
Set<String> set();

View File

@ -28,6 +28,7 @@ import static org.testng.Assert.assertEquals;
import static org.testng.Assert.assertNull;
import static org.testng.Assert.fail;
import java.io.Closeable;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
@ -175,7 +176,7 @@ public class RestAnnotationProcessorTest extends BaseRestApiTest {
}
@Path("/client/{jclouds.api-version}")
public static interface AsyncCallee {
public static interface AsyncCallee extends Closeable {
@GET
@Path("/{path}")
ListenableFuture<Void> onePath(@PathParam("path") String path);
@ -189,7 +190,7 @@ public class RestAnnotationProcessorTest extends BaseRestApiTest {
}
@Endpoint(Localhost2.class)
public static interface Caller {
public static interface Caller extends Closeable {
// tests that we can pull from suppliers
@Provides
@ -213,7 +214,7 @@ public class RestAnnotationProcessorTest extends BaseRestApiTest {
public Callee getCalleeWithPath(@EndpointParam URI endpoint, @PathParam("wibble") String wibble);
}
public static interface Callee {
public static interface Callee extends Closeable {
void onePath(String path);
}
@ -221,7 +222,7 @@ public class RestAnnotationProcessorTest extends BaseRestApiTest {
void onePath(String path);
}
public static interface AsyncCaller {
public static interface AsyncCaller extends Closeable {
@Provides
@Localhost2
URI getURI();