Sample code for 30+ languages & platforms
Objective-C

Amazon S3 List More than 1000 Objects in Bucket

See more Amazon S3 Examples

S3 limits the size of the "List Objects" response to 1000 objects. To retrieve a listing of all of the objects in a bucket containing more than 1000 objects, we'll need to send several requests using continuation tokens.

Chilkat Objective-C Downloads

Objective-C
#import <CkoHttp.h>
#import <CkoStringBuilder.h>
#import <CkoXml.h>
#import <NSString.h>
#import <CkoDateTime.h>
#import <CkoDtObj.h>

BOOL success = NO;

// This example assumes the Chilkat HTTP API to have been previously unlocked.
// See Global Unlock Sample for sample code.

CkoHttp *http = [[CkoHttp alloc] init];

// Insert your access key here:
http.AwsAccessKey = @"AWS_ACCESS_KEY";

// Insert your secret key here:
http.AwsSecretKey = @"AWS_SECRET_KEY";

// For the example, we'll get a listing containing approx. 25 objects
// using continuation tokens with an artificially low max-keys set to 5.
// (You may omit the max-keys query parameter to get the default 1000 maximum
// number of keys per request.)

CkoStringBuilder *sbContinuationToken = [[CkoStringBuilder alloc] init];
BOOL bGetMore = YES;
CkoStringBuilder *sbUri = [[CkoStringBuilder alloc] init];
CkoXml *xml = [[CkoXml alloc] init];

NSString *itemKey = 0;
NSString *itemSizeDecimalStr = 0;
NSString *lastModTimestamp = 0;
CkoDateTime *dt = [[CkoDateTime alloc] init];
CkoDtObj *dtObj = [[CkoDtObj alloc] init];

while (bGetMore == YES) {

    // Make sure to set the "list-type" query param equal to "2".
    // This chooses the GET Bucket (List Objects) Version 2 of the method.

    // IMPORTANT: You must include the max-keys param to get a response with a continuation token.
    // S3 limits the number of objects in a single response to 1000. Therefore, your max-keys
    // can be up to 1000.  If, for example, you have 2000 objects and do not specify
    // max-keys, then the response will contain 1000 objects with no continuation token. 
    [sbUri SetString: @"chilkat100?list-type=2&max-keys=5"];
    if ([sbContinuationToken.Length intValue] > 0) {
        [sbUri Append: @"&continuation-token="];
        [sbUri Append: [sbContinuationToken GetEncoded: @"url" charset: @"utf-8"]];
    }

    NSString *strXml = [http S3_ListBucketObjects: [sbUri GetAsString]];
    if (http.LastMethodSuccess == NO) {
        NSLog(@"%@",http.LastErrorText);
        return;
    }

    NSLog(@"%@%d",@"Response status code = ",[http.LastStatus intValue]);

    success = [xml LoadXml: strXml];
    if (success == NO) {
        NSLog(@"%@",xml.LastErrorText);
        return;
    }

    // If the response status code was not 200, then the XML response is not a 
    // listing of objects, but instead contains error information.
    if ([http.LastStatus intValue] != 200) {
        NSLog(@"%@",[xml GetXml]);
        NSLog(@"%@",@"Failed.");
        return;
    }

    // If this is not the final response, then we'll get an XML response that begins
    // like this.  (The IsTruncated child will be "true", and the NextContinuationToken
    // will have a value to be used in the next request.)
    // 	<?xml version="1.0" encoding="UTF-8" ?>
    // 	<ListBucketResult xmlns="http://s3.amazonaws.com/doc/2006-03-01/">
    // 	    <Name>chilkat100</Name>
    // 	    <Prefix />
    // 	    <NextContinuationToken>1Mlcx+W9OKsr8cxp3DP6r71wgsTUMj0vqlntWoaJKNbYdrauLdf40LsUdBeSYGFhzbGIHdcf5DSLcEBWbqG+1fW1UcQkUW1V4qgQONAKOwb8y8vOLJAQ8iQ==</NextContinuationToken>
    // 	    <KeyCount>5</KeyCount>
    // 	    <MaxKeys>5</MaxKeys>
    // 	    <IsTruncated>true</IsTruncated>
    // 	

    // Iterate over the bucket items in this chunk get information for each..
    int numItems = [[xml NumChildrenHavingTag: @"Contents"] intValue];
    NSLog(@"%@%d",@"Number of bucket items = ",numItems);

    int i = 0;
    while (i < numItems) {
        xml.I = [NSNumber numberWithInt: i];

        itemKey = [xml GetChildContent: @"Contents[i]|Key"];
        itemSizeDecimalStr = [xml GetChildContent: @"Contents[i]|Size"];
        lastModTimestamp = [xml GetChildContent: @"Contents[i]|LastModified"];

        [dt SetFromRfc822: lastModTimestamp];
        // Get a local date/time.
        BOOL bLocal = YES;
        [dt ToDtObj: bLocal dtObj: dtObj];

        NSLog(@"%d%@%@%@%@%@%d%@%d%@%d%@%d%@%d",i,@": ",itemKey,@", ",itemSizeDecimalStr,@", ",[dtObj.Day intValue],@"-"
            ,[dtObj.Month intValue],@"-",[dtObj.Year intValue],@":",[dtObj.Hour intValue],@":",[dtObj.Minute intValue]);

        i = i + 1;
    }

    // Check IsTruncated.
    bGetMore = NO;
    if ([xml ChildContentMatches: @"IsTruncated" pattern: @"true" caseSensitive: YES] == YES) {
        [sbContinuationToken SetString: [xml GetChildContent: @"NextContinuationToken"]];
        if ([sbContinuationToken.Length intValue] > 0) {
            bGetMore = YES;
        }

    }

}

// The output of this example (when tested) was:

// Response status code = 200
// Number of bucket items = 5
// 0: Abc.ics, 1833, 25-5-2011:9:53
// 1: Corpse Bride film poster.jpg, 53481, 6-9-2016:13:32
// 2: chiliPepper.gif, 7718, 12-3-2017:12:18
// 3: chilkatdude.jpg, 35137, 20-5-2011:16:14
// 4: cloud.search/dfe/indexer/pscc/2016/3/28/id,x-2-15-0-25-87-0.json, 1238, 2-4-2016:12:0
// Response status code = 200
// Number of bucket items = 5
// 0: cloud.search/dfe/indexer/pscc/2016/3/28/idx-2-15-0-25-87-0.json, 1238, 2-4-2016:11:33
// 1: dude.gif, 6373, 25-5-2011:17:29
// 2: french.txt, 47, 12-3-2017:12:18
// 3: hamlet.xml, 279658, 2-5-2016:12:21
// 4: hamlet_play.xml, 279658, 20-3-2017:8:22
// Response status code = 200
// Number of bucket items = 5
// 0: images/sea_creatures/starfish123.jpg, 6229, 19-1-2017:10:45
// 1: images/sea_creatures/starfishåäö.jpg, 6229, 19-1-2017:12:7
// 2: new folder/, 0, 26-11-2014:12:36
// 3: new_starfish.jpg, 6229, 20-3-2017:8:22
// 4: pigs.xml, 2804, 20-3-2017:8:22
// Response status code = 200
// Number of bucket items = 5
// 0: somethingBig.zip, 13089458, 26-9-2016:9:29
// 1: starfish.jpg, 6229, 12-3-2017:12:18
// 2: starfish/, 0, 10-11-2014:10:7
// 3: starfish/starfish.jpg, 6229, 10-11-2014:10:8
// 4: starfish/starfish2.jpg, 6229, 19-11-2014:10:36
// Response status code = 200
// Number of bucket items = 5
// 0: starfish/starfish3.jpg, 6229, 24-11-2014:14:33
// 1: starfish2.jpg, 5987, 20-4-2012:12:6
// 2: starfish3.jpg, 5987, 11-4-2012:7:10
// 3: starfishA.jpg, 6229, 10-5-2016:8:44
// 4: starfishCust.jpg, 6229, 12-11-2014:18:25
// Response status code = 200
// Number of bucket items = 1
// 0: xyz.ics, 1833, 25-5-2011:8:52
// 
//