Home | History | Annotate | Download | only in symupload
      1 // Copyright (c) 2006, Google Inc.
      2 // All rights reserved.
      3 //
      4 // Redistribution and use in source and binary forms, with or without
      5 // modification, are permitted provided that the following conditions are
      6 // met:
      7 //
      8 //     * Redistributions of source code must retain the above copyright
      9 // notice, this list of conditions and the following disclaimer.
     10 //     * Redistributions in binary form must reproduce the above
     11 // copyright notice, this list of conditions and the following disclaimer
     12 // in the documentation and/or other materials provided with the
     13 // distribution.
     14 //     * Neither the name of Google Inc. nor the names of its
     15 // contributors may be used to endorse or promote products derived from
     16 // this software without specific prior written permission.
     17 //
     18 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
     19 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
     20 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
     21 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
     22 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
     23 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
     24 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
     25 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
     26 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
     27 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
     28 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
     29 
     30 // symupload.m: Upload a symbol file to a HTTP server.  The upload is sent as
     31 // a multipart/form-data POST request with the following parameters:
     32 //  code_file: the basename of the module, e.g. "app"
     33 //  debug_file: the basename of the debugging file, e.g. "app"
     34 //  debug_identifier: the debug file's identifier, usually consisting of
     35 //                    the guid and age embedded in the pdb, e.g.
     36 //                    "11111111BBBB3333DDDD555555555555F"
     37 //  os: the operating system that the module was built for
     38 //  cpu: the CPU that the module was built for (x86 or ppc)
     39 //  symbol_file: the contents of the breakpad-format symbol file
     40 
     41 #include <unistd.h>
     42 
     43 #include <Foundation/Foundation.h>
     44 #include "HTTPMultipartUpload.h"
     45 
     46 typedef struct {
     47   NSString *symbolsPath;
     48   NSString *uploadURLStr;
     49   BOOL success;
     50 } Options;
     51 
     52 //=============================================================================
     53 static NSArray *ModuleDataForSymbolFile(NSString *file) {
     54   NSFileHandle *fh = [NSFileHandle fileHandleForReadingAtPath:file];
     55   NSData *data = [fh readDataOfLength:1024];
     56   NSString *str = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
     57   NSScanner *scanner = [NSScanner scannerWithString:str];
     58   NSString *line;
     59   NSMutableArray *parts = nil;
     60   const int MODULE_ID_INDEX = 3;
     61   
     62   if ([scanner scanUpToString:@"\n" intoString:&line]) {
     63     parts = [[NSMutableArray alloc] init];
     64     NSScanner *moduleInfoScanner = [NSScanner scannerWithString:line];
     65     NSString *moduleInfo;
     66     // Get everything BEFORE the module name.  None of these properties
     67     // can have spaces.
     68     for (int i = 0; i <= MODULE_ID_INDEX; i++) {
     69       [moduleInfoScanner scanUpToString:@" " intoString:&moduleInfo];
     70       [parts addObject:moduleInfo];
     71     }
     72 
     73     // Now get the module name. This can have a space so we scan to
     74     // the end of the line.
     75     [moduleInfoScanner scanUpToString:@"\n" intoString:&moduleInfo];
     76     [parts addObject:moduleInfo];
     77   }
     78 
     79   [str release];
     80 
     81   return parts;
     82 }
     83 
     84 //=============================================================================
     85 static NSString *CompactIdentifier(NSString *uuid) {
     86   NSMutableString *str = [NSMutableString stringWithString:uuid];
     87   [str replaceOccurrencesOfString:@"-" withString:@"" options:0
     88                             range:NSMakeRange(0, [str length])];
     89   
     90   return str;
     91 }
     92 
     93 //=============================================================================
     94 static void Start(Options *options) {
     95   NSURL *url = [NSURL URLWithString:options->uploadURLStr];
     96   HTTPMultipartUpload *ul = [[HTTPMultipartUpload alloc] initWithURL:url];
     97   NSMutableDictionary *parameters = [NSMutableDictionary dictionary];
     98   NSArray *moduleParts = ModuleDataForSymbolFile(options->symbolsPath);
     99   NSMutableString *compactedID =
    100     [NSMutableString stringWithString:[moduleParts objectAtIndex:3]];
    101   [compactedID replaceOccurrencesOfString:@"-" withString:@"" options:0
    102                                     range:NSMakeRange(0, [compactedID length])];
    103 
    104   // Add parameters
    105   [parameters setObject:compactedID forKey:@"debug_identifier"];
    106 
    107   // MODULE <os> <cpu> <uuid> <module-name>
    108   // 0      1    2     3      4
    109   [parameters setObject:[moduleParts objectAtIndex:1] forKey:@"os"];
    110   [parameters setObject:[moduleParts objectAtIndex:2] forKey:@"cpu"];
    111   [parameters setObject:[moduleParts objectAtIndex:4] forKey:@"debug_file"];
    112   [parameters setObject:[moduleParts objectAtIndex:4] forKey:@"code_file"];
    113   [ul setParameters:parameters];
    114   
    115   NSArray *keys = [parameters allKeys];
    116   int count = [keys count];
    117   for (int i = 0; i < count; ++i) {
    118     NSString *key = [keys objectAtIndex:i];
    119     NSString *value = [parameters objectForKey:key];
    120     fprintf(stdout, "'%s' = '%s'\n", [key UTF8String], 
    121             [value UTF8String]);
    122   }
    123 
    124   // Add file
    125   [ul addFileAtPath:options->symbolsPath name:@"symbol_file"];
    126 
    127   // Send it
    128   NSError *error = nil;
    129   NSData *data = [ul send:&error];
    130   NSString *result = [[NSString alloc] initWithData:data
    131                                            encoding:NSUTF8StringEncoding];
    132   int status = [[ul response] statusCode];
    133 
    134   fprintf(stdout, "Send: %s\n", error ? [[error description] UTF8String] :
    135           "No Error");
    136   fprintf(stdout, "Response: %d\n", status);
    137   fprintf(stdout, "Result: %lu bytes\n%s\n",
    138           (unsigned long)[data length], [result UTF8String]);
    139 
    140   [result release];
    141   [ul release];
    142   options->success = !error && status==200;
    143 }
    144 
    145 //=============================================================================
    146 static void
    147 Usage(int argc, const char *argv[]) {
    148   fprintf(stderr, "Submit symbol information.\n");
    149   fprintf(stderr, "Usage: %s <symbols> <upload-URL>\n", argv[0]);
    150   fprintf(stderr, "<symbols> should be created by using the dump_syms tool.\n");
    151   fprintf(stderr, "<upload-URL> is the destination for the upload\n");
    152   fprintf(stderr, "\t-h: Usage\n");
    153   fprintf(stderr, "\t-?: Usage\n");
    154 }
    155 
    156 //=============================================================================
    157 static void
    158 SetupOptions(int argc, const char *argv[], Options *options) {
    159   extern int optind;
    160   char ch;
    161 
    162   while ((ch = getopt(argc, (char * const *)argv, "h?")) != -1) {
    163     switch (ch) {
    164       default:
    165         Usage(argc, argv);
    166         exit(0);
    167         break;
    168     }
    169   }
    170 
    171   if ((argc - optind) != 2) {
    172     fprintf(stderr, "%s: Missing symbols file and/or upload-URL\n", argv[0]);
    173     Usage(argc, argv);
    174     exit(1);
    175   }
    176 
    177   options->symbolsPath = [NSString stringWithUTF8String:argv[optind]];
    178   options->uploadURLStr = [NSString stringWithUTF8String:argv[optind + 1]];
    179 }
    180 
    181 //=============================================================================
    182 int main (int argc, const char * argv[]) {
    183   NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
    184   Options options;
    185 
    186   bzero(&options, sizeof(Options));
    187   SetupOptions(argc, argv, &options);
    188   Start(&options);
    189 
    190   [pool release];
    191   return options.success ? 0 : 1;
    192 }
    193