2009-06-28 13 views
5

Wyciągam włosy z tego ... Używam biblioteki otworkowej ASIHTTPRequest (http://allseeing-i.com/ASIHTTPRequest/) do uzyskiwania dostępu do pamięci Amazon S3. Jestem w stanie połączyć się dobrze i pobrać listę wiader bez żadnych problemów. Moja frustracja polega na próbowaniu UPLOAD (PUT i/lub POST) nowego obiektu (zdjęcia) do istniejącego zasobnika. Postępuję zgodnie z dokumentacją Amazon do tego listu (przynajmniej tak myślę.), Ale nic nie działa.Przesyłanie do usług Amazon S3 z aplikacji iPhone'a

Proszę, pomóż mi, zanim wyskoczę przez okno. Nie chcę umrzeć. :-(

Dzięki z góry za wszelką pomoc można uzyskać.

L.

+0

Hmmmm - bardzo trudno jest pomóc bez żadnych szczegółów poza "to nie działa". Co konkretnie? W jaki sposób? Kod? Co próbujesz? – Hunter

Odpowiedz

12

Oto prosty przykład przy użyciu PUT. Oczywiście należy się za pomocą kolejki zamiast synchronicznego zamówienie w rzeczywistym świat.

Jeżeli zmienisz nagłówki AMZ, nie zapomnij, aby zaktualizować „canonicalizedAmzHeaders” zgodnie z instrukcjami Amazon.

#import "ASIHTTPRequest.h" 
#import <CommonCrypto/CommonHMAC.h> 

... 

- (void)testS3 
{ 
    NSString *filePath = @"/path/to/file"; 
    NSString *contentType = @"text/plain"; 
    NSString *bucket = @"mybucket"; 
    NSString *path = @"test"; 
    NSString *secretAccessKey = @"my-secret-access-key"; 
    NSString *accessKey = @"my-access-key"; 

    NSDateFormatter *dateFormatter = [[[NSDateFormatter alloc] init] autorelease]; 
    [dateFormatter setDateFormat:@"EEE, d MMM yyyy HH:mm:ss zzzz"]; 
    NSString *date = [dateFormatter stringFromDate:[NSDate date]]; 

    ASIHTTPRequest *request = [[[ASIHTTPRequest alloc] initWithURL:[NSURL URLWithString:[NSString stringWithFormat:@"http://%@.s3.amazonaws.com/%@",bucket,path]]] autorelease]; 
    [request setPostBodyFilePath:filePath]; 
    [request setShouldStreamPostDataFromDisk:YES]; 
    [request setRequestMethod:@"PUT"]; 

    [request addRequestHeader:@"x-amz-acl" value:@"private"]; 
    [request addRequestHeader:@"Content-Type" value:contentType]; 
    [request addRequestHeader:@"Date" value:date]; 

    NSString *canonicalizedAmzHeaders = @"x-amz-acl:private"; 
    NSString *canonicalizedResource = [NSString stringWithFormat:@"/%@/%@",bucket,path]; 
    NSString *stringToSign = [NSString stringWithFormat:@"PUT\n\n%@\n%@\n%@\n%@",contentType,date,canonicalizedAmzHeaders,canonicalizedResource]; 

    NSString *signature = [self base64forData:[self HMACSHA1withKey:secretAccessKey forString:stringToSign]]; 
    NSString *auth = [NSString stringWithFormat:@"AWS %@:%@",accessKey,signature]; 
    [request addRequestHeader:@"Authorization" value:auth]; 


    [request start]; 
    NSLog(@"%@",[request responseString]); 

} 


// Source: http://stackoverflow.com/questions/476455/is-there-a-library-for-iphone-to-work-with-hmac-sha-1-encoding 

- (NSData *)HMACSHA1withKey:(NSString *)key forString:(NSString *)string 
{ 
    NSData *clearTextData = [string dataUsingEncoding:NSUTF8StringEncoding]; 
    NSData *keyData = [key dataUsingEncoding:NSUTF8StringEncoding]; 

    uint8_t digest[CC_SHA1_DIGEST_LENGTH] = {0}; 

    CCHmacContext hmacContext; 
    CCHmacInit(&hmacContext, kCCHmacAlgSHA1, keyData.bytes, keyData.length); 
    CCHmacUpdate(&hmacContext, clearTextData.bytes, clearTextData.length); 
    CCHmacFinal(&hmacContext, digest); 

    return [NSData dataWithBytes:digest length:CC_SHA1_DIGEST_LENGTH]; 
} 

//Source http://www.cocoadev.com/index.pl?BaseSixtyFour 

- (NSString *)base64forData:(NSData *)data 
{ 
    static const char encodingTable[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz/"; 

    if ([data length] == 0) 
     return @""; 

    char *characters = malloc((([data length] + 2)/3) * 4); 
    if (characters == NULL) 
     return nil; 
    NSUInteger length = 0; 

    NSUInteger i = 0; 
    while (i < [data length]) 
    { 
     char buffer[3] = {0,0,0}; 
     short bufferLength = 0; 
     while (bufferLength < 3 && i < [data length]) 
      buffer[bufferLength++] = ((char *)[data bytes])[i++]; 

     // Encode the bytes in the buffer to four characters, including padding "=" characters if necessary. 
     characters[length++] = encodingTable[(buffer[0] & 0xFC) >> 2]; 
     characters[length++] = encodingTable[((buffer[0] & 0x03) << 4) | ((buffer[1] & 0xF0) >> 4)]; 
     if (bufferLength > 1) 
      characters[length++] = encodingTable[((buffer[1] & 0x0F) << 2) | ((buffer[2] & 0xC0) >> 6)]; 
     else characters[length++] = '='; 
     if (bufferLength > 2) 
      characters[length++] = encodingTable[buffer[2] & 0x3F]; 
     else characters[length++] = '=';  
    } 

    return [[[NSString alloc] initWithBytesNoCopy:characters length:length encoding:NSASCIIStringEncoding freeWhenDone:YES] autorelease]; 
} 
+0

You Rock pokeb. Naprawdę doceniam pomoc. Mój kanonicznyAmzHeader był cały gruby. Mam jednak jeszcze jeden problem ... Próbowałem "dodać" żądanie do kolejki i zacząć od tego, ale bez skutku. Oto mój kod: [networkQueue cancelAllOperations]; [prośba ..... definicje tutaj]; [networkQueue addOperation: request]; [networkQueue go]; Na co powinienem zwrócić uwagę przy dodawaniu żądania do kolejki? jeszcze raz dziękuję za pomoc! L. –

+0

Czy najpierw utworzyłeś kolejkę? (np .: [self setNetworkQueue: [kolejka ASINetworkQueue]]) – pokeb

+0

Znów masz rację. Moja instancja networkQueue nie została nigdy zainicjalizowana. Jesteś meżczyzną. Naprawdę doceniam pomoc. –

1

Istnieje również Connection Kit Struktura kakao, która umożliwia przesyłanie danych do różnych usług, w tym do Amazon S3. Jestem prawie pewien, że ma kilka punktów wyjścia dla ciebie w swoim źródle.

11

W przypadku jest to pomocne dla każdego, znalezienie tego pytanie - wsparcie podstawowe S3 jest teraz wbudowany w ASIHTTPRequest:

http://allseeing-i.com/ASIHTTPRequest/S3

myślałam o dodanie wsparcia S3 od wieków, ale Twoje pytanie trącił go na początku kolejki :)

+0

ŁADNE! Uwielbiam opakowanie. Stałem się twoim największym fanem. Kontynuuj WIELKĄ pracę! –

+0

dziękuję za cudowną bibliotekę! :) –

Powiązane problemy