xref: /petsc/src/sys/webclient/box.c (revision d817401416fe7522c08df98fd1821bb4e0177fd0)
14a285bdaSBarry Smith 
24a285bdaSBarry Smith #include <petscwebclient.h>
3bb04b57dSBarry Smith #pragma clang diagnostic ignored "-Wdeprecated-declarations"
445e40e47SBarry Smith #pragma gcc diagnostic ignored "-Wdeprecated-declarations"
54a285bdaSBarry Smith 
64a285bdaSBarry Smith /*
74a285bdaSBarry Smith    These variables identify the code as a PETSc application to Box.
84a285bdaSBarry Smith 
9a8d69d7bSBarry Smith    See -   https://stackoverflow.com/questions/4616553/using-oauth-in-free-open-source-software
10a8d69d7bSBarry Smith    Users can get their own application IDs - goto https://developer.box.com
114a285bdaSBarry Smith 
124a285bdaSBarry Smith */
134a285bdaSBarry Smith #define PETSC_BOX_CLIENT_ID  "sse42nygt4zqgrdwi0luv79q1u1f0xza"
144a285bdaSBarry Smith #define PETSC_BOX_CLIENT_ST  "A0Dy4KgOYLB2JIYZqpbze4EzjeIiX5k4"
154a285bdaSBarry Smith 
16f044a08eSBarry Smith #if defined(PETSC_HAVE_SAWS)
174a285bdaSBarry Smith #include <mongoose.h>
184a285bdaSBarry Smith 
194a285bdaSBarry Smith static volatile char *result = NULL;
204a285bdaSBarry Smith 
214a285bdaSBarry Smith static int PetscBoxWebServer_Private(struct mg_connection *conn)
224a285bdaSBarry Smith {
234a285bdaSBarry Smith   const struct mg_request_info *request_info = mg_get_request_info(conn);
244a285bdaSBarry Smith   result = (char*) request_info->query_string;
25f044a08eSBarry Smith   return 1;  /* Mongoose will now not handle the request */
264a285bdaSBarry Smith }
274a285bdaSBarry Smith 
28f044a08eSBarry Smith /*
29f044a08eSBarry Smith     Box can only return an authorization code to a Webserver, hence we need to start one up and wait for
30f044a08eSBarry Smith     the authorization code to arrive from Box
31f044a08eSBarry Smith */
324a285bdaSBarry Smith static PetscErrorCode PetscBoxStartWebServer_Private(void)
334a285bdaSBarry Smith {
344a285bdaSBarry Smith   int                 optionsLen = 5;
354a285bdaSBarry Smith   const char          *options[optionsLen];
364a285bdaSBarry Smith   struct mg_callbacks callbacks;
374a285bdaSBarry Smith   struct mg_context   *ctx;
3868e69593SBarry Smith   char                keyfile[PETSC_MAX_PATH_LEN];
3968e69593SBarry Smith   PetscBool           exists;
404a285bdaSBarry Smith 
414a285bdaSBarry Smith   PetscFunctionBegin;
424a285bdaSBarry Smith   options[0] = "listening_ports";
434a285bdaSBarry Smith   options[1] = "8081s";
4468e69593SBarry Smith 
459566063dSJacob Faibussowitsch   PetscCall(PetscStrcpy(keyfile,"sslclient.pem"));
469566063dSJacob Faibussowitsch   PetscCall(PetscTestFile(keyfile,'r',&exists));
4768e69593SBarry Smith   if (!exists) {
489566063dSJacob Faibussowitsch     PetscCall(PetscGetHomeDirectory(keyfile,PETSC_MAX_PATH_LEN));
499566063dSJacob Faibussowitsch     PetscCall(PetscStrcat(keyfile,"/"));
509566063dSJacob Faibussowitsch     PetscCall(PetscStrcat(keyfile,"sslclient.pem"));
519566063dSJacob Faibussowitsch     PetscCall(PetscTestFile(keyfile,'r',&exists));
5228b400f6SJacob Faibussowitsch     PetscCheck(exists,PETSC_COMM_SELF,PETSC_ERR_FILE_OPEN,"Unable to locate sslclient.pem file in current directory or home directory");
5368e69593SBarry Smith   }
5468e69593SBarry Smith 
554a285bdaSBarry Smith   options[2] = "ssl_certificate";
5668e69593SBarry Smith   options[3] = keyfile;
574a285bdaSBarry Smith   options[4] = NULL;
584a285bdaSBarry Smith 
594a285bdaSBarry Smith   /* Prepare callbacks structure. We have only one callback, the rest are NULL. */
609566063dSJacob Faibussowitsch   PetscCall(PetscMemzero(&callbacks, sizeof(callbacks)));
614a285bdaSBarry Smith   callbacks.begin_request = PetscBoxWebServer_Private;
624a285bdaSBarry Smith   ctx = mg_start(&callbacks, NULL, options);
6328b400f6SJacob Faibussowitsch   PetscCheck(ctx,PETSC_COMM_SELF,PETSC_ERR_LIB,"Unable to start up webserver");
644a285bdaSBarry Smith   while (!result) {};
654a285bdaSBarry Smith   PetscFunctionReturn(0);
664a285bdaSBarry Smith }
674a285bdaSBarry Smith 
6868e69593SBarry Smith #if defined(PETSC_HAVE_UNISTD_H)
6968e69593SBarry Smith #include <unistd.h>
7068e69593SBarry Smith #endif
7168e69593SBarry Smith 
724a285bdaSBarry Smith /*@C
734a285bdaSBarry Smith      PetscBoxAuthorize - Get authorization and refresh token for accessing Box drive from PETSc
744a285bdaSBarry Smith 
754a285bdaSBarry Smith    Not collective, only the first process in MPI_Comm does anything
764a285bdaSBarry Smith 
774a285bdaSBarry Smith    Input Parameters:
784a285bdaSBarry Smith +  comm - the MPI communicator
794a285bdaSBarry Smith -  tokensize - size of the token arrays
804a285bdaSBarry Smith 
814a285bdaSBarry Smith    Output Parameters:
824a285bdaSBarry Smith +  access_token - can be used with PetscBoxUpload() for this one session
834a285bdaSBarry Smith -  refresh_token - can be used for ever to obtain new access_tokens with PetscBoxRefresh(), guard this like a password
844a285bdaSBarry Smith                    it gives access to your Box Drive
854a285bdaSBarry Smith 
8695452b02SPatrick Sanan    Notes:
8795452b02SPatrick Sanan     This call requires stdout and stdin access from process 0 on the MPI communicator
884a285bdaSBarry Smith 
89c4762a1bSJed Brown    You can run src/sys/webclient/tutorials/boxobtainrefreshtoken to get a refresh token and then in the future pass it to
904a285bdaSBarry Smith    PETSc programs with -box_refresh_token XXX
914a285bdaSBarry Smith 
92f044a08eSBarry Smith    This requires PETSc be installed using --with-saws or --download-saws
93f044a08eSBarry Smith 
9468e69593SBarry Smith    Requires the user have created a self-signed ssl certificate with
9568e69593SBarry Smith 
9668e69593SBarry Smith $    saws/CA.pl  -newcert  (using the passphrase of password)
9768e69593SBarry Smith $    cat newkey.pem newcert.pem > sslclient.pem
9868e69593SBarry Smith 
9968e69593SBarry Smith     and put the resulting file in either the current directory (with the application) or in the home directory. This seems kind of
10068e69593SBarry Smith     silly but it was all I could figure out.
10168e69593SBarry Smith 
1022b26979fSBarry Smith    Level: intermediate
1032b26979fSBarry Smith 
104db781477SPatrick Sanan .seealso: `PetscBoxRefresh()`, `PetscBoxUpload()`, `PetscURLShorten()`
1054a285bdaSBarry Smith 
1064a285bdaSBarry Smith @*/
1074a285bdaSBarry Smith PetscErrorCode PetscBoxAuthorize(MPI_Comm comm,char access_token[],char refresh_token[],size_t tokensize)
1084a285bdaSBarry Smith {
1094a285bdaSBarry Smith   SSL_CTX        *ctx;
1104a285bdaSBarry Smith   SSL            *ssl;
1114a285bdaSBarry Smith   int            sock;
1125dc0f0a4SBarry Smith   char           buff[8*1024],body[1024];
1134a285bdaSBarry Smith   PetscMPIInt    rank;
1145dc0f0a4SBarry Smith   PetscBool      flg,found;
1154a285bdaSBarry Smith 
1164a285bdaSBarry Smith   PetscFunctionBegin;
1179566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_rank(comm,&rank));
118dd400576SPatrick Sanan   if (rank == 0) {
119cc73adaaSBarry Smith     PetscCheck(isatty(fileno(PETSC_STDOUT)),PETSC_COMM_SELF,PETSC_ERR_USER,"Requires users input/output");
120d0609cedSBarry Smith     PetscCall(PetscPrintf(comm,"Cut and paste the following into your browser:\n\n"
1214a285bdaSBarry Smith                           "https://www.box.com/api/oauth2/authorize?"
1224a285bdaSBarry Smith                           "response_type=code&"
1234a285bdaSBarry Smith                           "client_id="
1244a285bdaSBarry Smith                           PETSC_BOX_CLIENT_ID
1254a285bdaSBarry Smith                           "&state=PETScState"
126d0609cedSBarry Smith                           "\n\n"));
1279566063dSJacob Faibussowitsch     PetscCall(PetscBoxStartWebServer_Private());
1289566063dSJacob Faibussowitsch     PetscCall(PetscStrbeginswith((const char*)result,"state=PETScState&code=",&flg));
12928b400f6SJacob Faibussowitsch     PetscCheck(flg,PETSC_COMM_SELF,PETSC_ERR_LIB,"Did not get expected string from Box got %s",result);
1309566063dSJacob Faibussowitsch     PetscCall(PetscStrncpy(buff,(const char*)result+22,sizeof(buff)));
1314a285bdaSBarry Smith 
1329566063dSJacob Faibussowitsch     PetscCall(PetscSSLInitializeContext(&ctx));
1339566063dSJacob Faibussowitsch     PetscCall(PetscHTTPSConnect("www.box.com",443,ctx,&sock,&ssl));
1349566063dSJacob Faibussowitsch     PetscCall(PetscStrcpy(body,"code="));
1359566063dSJacob Faibussowitsch     PetscCall(PetscStrcat(body,buff));
1369566063dSJacob Faibussowitsch     PetscCall(PetscStrcat(body,"&client_id="));
1379566063dSJacob Faibussowitsch     PetscCall(PetscStrcat(body,PETSC_BOX_CLIENT_ID));
1389566063dSJacob Faibussowitsch     PetscCall(PetscStrcat(body,"&client_secret="));
1399566063dSJacob Faibussowitsch     PetscCall(PetscStrcat(body,PETSC_BOX_CLIENT_ST));
1409566063dSJacob Faibussowitsch     PetscCall(PetscStrcat(body,"&grant_type=authorization_code"));
1414a285bdaSBarry Smith 
1429566063dSJacob Faibussowitsch     PetscCall(PetscHTTPSRequest("POST","www.box.com/api/oauth2/token",NULL,"application/x-www-form-urlencoded",body,ssl,buff,sizeof(buff)));
1439566063dSJacob Faibussowitsch     PetscCall(PetscSSLDestroyContext(ctx));
1444a285bdaSBarry Smith     close(sock);
1454a285bdaSBarry Smith 
1469566063dSJacob Faibussowitsch     PetscCall(PetscPullJSONValue(buff,"access_token",access_token,tokensize,&found));
14728b400f6SJacob Faibussowitsch     PetscCheck(found,PETSC_COMM_SELF,PETSC_ERR_LIB,"Box did not return access token");
1489566063dSJacob Faibussowitsch     PetscCall(PetscPullJSONValue(buff,"refresh_token",refresh_token,tokensize,&found));
14928b400f6SJacob Faibussowitsch     PetscCheck(found,PETSC_COMM_SELF,PETSC_ERR_LIB,"Box did not return refresh token");
1504a285bdaSBarry Smith 
1519566063dSJacob Faibussowitsch     PetscCall(PetscPrintf(comm,"Here is your Box refresh token, save it in a save place, in the future you can run PETSc\n"));
1529566063dSJacob Faibussowitsch     PetscCall(PetscPrintf(comm,"programs with the option -box_refresh_token %s\n",refresh_token));
1539566063dSJacob Faibussowitsch     PetscCall(PetscPrintf(comm,"to access Box Drive automatically\n"));
1544a285bdaSBarry Smith   }
1554a285bdaSBarry Smith   PetscFunctionReturn(0);
1564a285bdaSBarry Smith }
157f044a08eSBarry Smith #endif
1584a285bdaSBarry Smith 
1594a285bdaSBarry Smith /*@C
1604a285bdaSBarry Smith      PetscBoxRefresh - Get a new authorization token for accessing Box drive from PETSc from a refresh token
1614a285bdaSBarry Smith 
1624a285bdaSBarry Smith    Not collective, only the first process in the MPI_Comm does anything
1634a285bdaSBarry Smith 
1644a285bdaSBarry Smith    Input Parameters:
1654a285bdaSBarry Smith +   comm - MPI communicator
1664a285bdaSBarry Smith .   refresh token - obtained with PetscBoxAuthorize(), if NULL PETSc will first look for one in the options data
1674a285bdaSBarry Smith                     if not found it will call PetscBoxAuthorize()
1684a285bdaSBarry Smith -   tokensize - size of the output string access_token
1694a285bdaSBarry Smith 
170d8d19677SJose E. Roman    Output Parameters:
1714a285bdaSBarry Smith +   access_token - token that can be passed to PetscBoxUpload()
1724a285bdaSBarry Smith -   new_refresh_token - the old refresh token is no longer valid, not this is different than Google where the same refresh_token is used forever
1734a285bdaSBarry Smith 
1742b26979fSBarry Smith    Level: intermediate
1752b26979fSBarry Smith 
176db781477SPatrick Sanan .seealso: `PetscURLShorten()`, `PetscBoxAuthorize()`, `PetscBoxUpload()`
1774a285bdaSBarry Smith 
1784a285bdaSBarry Smith @*/
1794a285bdaSBarry Smith PetscErrorCode PetscBoxRefresh(MPI_Comm comm,const char refresh_token[],char access_token[],char new_refresh_token[],size_t tokensize)
1804a285bdaSBarry Smith {
1814a285bdaSBarry Smith   SSL_CTX        *ctx;
1824a285bdaSBarry Smith   SSL            *ssl;
1834a285bdaSBarry Smith   int            sock;
1845dc0f0a4SBarry Smith   char           buff[8*1024],body[1024];
1854a285bdaSBarry Smith   PetscMPIInt    rank;
1865dc0f0a4SBarry Smith   char           *refreshtoken = (char*)refresh_token;
1875dc0f0a4SBarry Smith   PetscBool      found;
1884a285bdaSBarry Smith 
1894a285bdaSBarry Smith   PetscFunctionBegin;
1909566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_rank(comm,&rank));
191dd400576SPatrick Sanan   if (rank == 0) {
1924a285bdaSBarry Smith     if (!refresh_token) {
1934a285bdaSBarry Smith       PetscBool set;
1949566063dSJacob Faibussowitsch       PetscCall(PetscMalloc1(512,&refreshtoken));
1959566063dSJacob Faibussowitsch       PetscCall(PetscOptionsGetString(NULL,NULL,"-box_refresh_token",refreshtoken,sizeof(refreshtoken),&set));
19699374591SBarry Smith #if defined(PETSC_HAVE_SAWS)
1974a285bdaSBarry Smith       if (!set) {
1989566063dSJacob Faibussowitsch         PetscCall(PetscBoxAuthorize(comm,access_token,new_refresh_token,512*sizeof(char)));
1999566063dSJacob Faibussowitsch         PetscCall(PetscFree(refreshtoken));
2004a285bdaSBarry Smith         PetscFunctionReturn(0);
2014a285bdaSBarry Smith       }
20299374591SBarry Smith #else
20328b400f6SJacob Faibussowitsch       PetscCheck(set,PETSC_COMM_SELF,PETSC_ERR_LIB,"Must provide refresh token with -box_refresh_token XXX");
20499374591SBarry Smith #endif
2054a285bdaSBarry Smith     }
2069566063dSJacob Faibussowitsch     PetscCall(PetscSSLInitializeContext(&ctx));
2079566063dSJacob Faibussowitsch     PetscCall(PetscHTTPSConnect("www.box.com",443,ctx,&sock,&ssl));
2089566063dSJacob Faibussowitsch     PetscCall(PetscStrcpy(body,"client_id="));
2099566063dSJacob Faibussowitsch     PetscCall(PetscStrcat(body,PETSC_BOX_CLIENT_ID));
2109566063dSJacob Faibussowitsch     PetscCall(PetscStrcat(body,"&client_secret="));
2119566063dSJacob Faibussowitsch     PetscCall(PetscStrcat(body,PETSC_BOX_CLIENT_ST));
2129566063dSJacob Faibussowitsch     PetscCall(PetscStrcat(body,"&refresh_token="));
2139566063dSJacob Faibussowitsch     PetscCall(PetscStrcat(body,refreshtoken));
2149566063dSJacob Faibussowitsch     if (!refresh_token) PetscCall(PetscFree(refreshtoken));
2159566063dSJacob Faibussowitsch     PetscCall(PetscStrcat(body,"&grant_type=refresh_token"));
2164a285bdaSBarry Smith 
2179566063dSJacob Faibussowitsch     PetscCall(PetscHTTPSRequest("POST","www.box.com/api/oauth2/token",NULL,"application/x-www-form-urlencoded",body,ssl,buff,sizeof(buff)));
2189566063dSJacob Faibussowitsch     PetscCall(PetscSSLDestroyContext(ctx));
2194a285bdaSBarry Smith     close(sock);
2204a285bdaSBarry Smith 
2219566063dSJacob Faibussowitsch     PetscCall(PetscPullJSONValue(buff,"access_token",access_token,tokensize,&found));
22228b400f6SJacob Faibussowitsch     PetscCheck(found,PETSC_COMM_SELF,PETSC_ERR_LIB,"Box did not return access token");
2239566063dSJacob Faibussowitsch     PetscCall(PetscPullJSONValue(buff,"refresh_token",new_refresh_token,tokensize,&found));
22428b400f6SJacob Faibussowitsch     PetscCheck(found,PETSC_COMM_SELF,PETSC_ERR_LIB,"Box did not return refresh token");
22593e1d32fSBarry Smith 
2269566063dSJacob Faibussowitsch     PetscCall(PetscPrintf(comm,"Here is your new Box refresh token, save it in a save place, in the future you can run PETSc\n"));
2279566063dSJacob Faibussowitsch     PetscCall(PetscPrintf(comm,"programs with the option -box_refresh_token %s\n",new_refresh_token));
2289566063dSJacob Faibussowitsch     PetscCall(PetscPrintf(comm,"to access Box Drive automatically\n"));
2294a285bdaSBarry Smith   }
2304a285bdaSBarry Smith   PetscFunctionReturn(0);
2314a285bdaSBarry Smith }
2324a285bdaSBarry Smith 
2334a285bdaSBarry Smith #include <sys/stat.h>
2344a285bdaSBarry Smith 
2354a285bdaSBarry Smith /*@C
2364a285bdaSBarry Smith      PetscBoxUpload - Loads a file to the Box Drive
2374a285bdaSBarry Smith 
23893e1d32fSBarry Smith      This routine has not yet been written; it is just copied from Google Drive
23993e1d32fSBarry Smith 
2404a285bdaSBarry Smith      Not collective, only the first process in the MPI_Comm uploads the file
2414a285bdaSBarry Smith 
2424a285bdaSBarry Smith   Input Parameters:
2434a285bdaSBarry Smith +   comm - MPI communicator
2444a285bdaSBarry Smith .   access_token - obtained with PetscBoxRefresh(), pass NULL to have PETSc generate one
2454a285bdaSBarry Smith -   filename - file to upload; if you upload multiple times it will have different names each time on Box Drive
2464a285bdaSBarry Smith 
2474a285bdaSBarry Smith   Options Database:
24810699b91SBarry Smith .  -box_refresh_token XXX - the token value
2494a285bdaSBarry Smith 
2504a285bdaSBarry Smith   Usage Patterns:
2514a285bdaSBarry Smith     With PETSc option -box_refresh_token XXX given
2524a285bdaSBarry Smith     PetscBoxUpload(comm,NULL,filename);        will upload file with no user interaction
2534a285bdaSBarry Smith 
2544a285bdaSBarry Smith     Without PETSc option -box_refresh_token XXX given
2554a285bdaSBarry Smith     PetscBoxUpload(comm,NULL,filename);        for first use will prompt user to authorize access to Box Drive with their processor
2564a285bdaSBarry Smith 
2574a285bdaSBarry Smith     With PETSc option -box_refresh_token  XXX given
2584a285bdaSBarry Smith     PetscBoxRefresh(comm,NULL,access_token,sizeof(access_token));
2594a285bdaSBarry Smith     PetscBoxUpload(comm,access_token,filename);
2604a285bdaSBarry Smith 
2614a285bdaSBarry Smith     With refresh token entered in some way by the user
2624a285bdaSBarry Smith     PetscBoxRefresh(comm,refresh_token,access_token,sizeof(access_token));
2634a285bdaSBarry Smith     PetscBoxUpload(comm,access_token,filename);
2644a285bdaSBarry Smith 
2654a285bdaSBarry Smith     PetscBoxAuthorize(comm,access_token,refresh_token,sizeof(access_token));
2664a285bdaSBarry Smith     PetscBoxUpload(comm,access_token,filename);
2674a285bdaSBarry Smith 
2682b26979fSBarry Smith    Level: intermediate
2692b26979fSBarry Smith 
270db781477SPatrick Sanan .seealso: `PetscURLShorten()`, `PetscBoxAuthorize()`, `PetscBoxRefresh()`
2714a285bdaSBarry Smith 
2724a285bdaSBarry Smith @*/
2734a285bdaSBarry Smith PetscErrorCode PetscBoxUpload(MPI_Comm comm,const char access_token[],const char filename[])
2744a285bdaSBarry Smith {
2754a285bdaSBarry Smith   SSL_CTX        *ctx;
2764a285bdaSBarry Smith   SSL            *ssl;
2774a285bdaSBarry Smith   int            sock;
2784a285bdaSBarry Smith   char           head[1024],buff[8*1024],*body,*title;
2794a285bdaSBarry Smith   PetscMPIInt    rank;
2804a285bdaSBarry Smith   struct stat    sb;
2814a285bdaSBarry Smith   size_t         len,blen,rd;
2824a285bdaSBarry Smith   FILE           *fd;
2832da392ccSBarry Smith   int            err;
2844a285bdaSBarry Smith 
2854a285bdaSBarry Smith   PetscFunctionBegin;
2869566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_rank(comm,&rank));
287dd400576SPatrick Sanan   if (rank == 0) {
2889566063dSJacob Faibussowitsch     PetscCall(PetscStrcpy(head,"Authorization: Bearer "));
2899566063dSJacob Faibussowitsch     PetscCall(PetscStrcat(head,access_token));
2909566063dSJacob Faibussowitsch     PetscCall(PetscStrcat(head,"\r\n"));
2919566063dSJacob Faibussowitsch     PetscCall(PetscStrcat(head,"uploadType: multipart\r\n"));
2924a285bdaSBarry Smith 
2932da392ccSBarry Smith     err = stat(filename,&sb);
29428b400f6SJacob Faibussowitsch     PetscCheck(!err,PETSC_COMM_SELF,PETSC_ERR_FILE_OPEN,"Unable to stat file: %s",filename);
2954a285bdaSBarry Smith     len = 1024 + sb.st_size;
2969566063dSJacob Faibussowitsch     PetscCall(PetscMalloc1(len,&body));
297d0609cedSBarry Smith     PetscCall(PetscStrcpy(body,"--foo_bar_baz\r\n"
2984a285bdaSBarry Smith                                "Content-Type: application/json\r\n\r\n"
299d0609cedSBarry Smith                                "{"));
3009566063dSJacob Faibussowitsch     PetscCall(PetscPushJSONValue(body,"title",filename,len));
3019566063dSJacob Faibussowitsch     PetscCall(PetscStrcat(body,","));
3029566063dSJacob Faibussowitsch     PetscCall(PetscPushJSONValue(body,"mimeType","text.html",len));
3039566063dSJacob Faibussowitsch     PetscCall(PetscStrcat(body,","));
3049566063dSJacob Faibussowitsch     PetscCall(PetscPushJSONValue(body,"description","a file",len));
305*d8174014SToby Isaac     PetscCall(PetscStrcat(body, "}\r\n\r\n"
3064a285bdaSBarry Smith                                 "--foo_bar_baz\r\n"
307d0609cedSBarry Smith                                 "Content-Type: text/html\r\n\r\n"));
3089566063dSJacob Faibussowitsch     PetscCall(PetscStrlen(body,&blen));
3094a285bdaSBarry Smith     fd = fopen (filename, "r");
31028b400f6SJacob Faibussowitsch     PetscCheck(fd,PETSC_COMM_SELF,PETSC_ERR_FILE_OPEN,"Unable to open file: %s",filename);
3114a285bdaSBarry Smith     rd = fread (body+blen, sizeof (unsigned char), sb.st_size, fd);
312cc73adaaSBarry Smith     PetscCheck(rd == (size_t)sb.st_size,PETSC_COMM_SELF,PETSC_ERR_FILE_OPEN,"Unable to read entire file: %s %d %d",filename,(int)rd,(int)sb.st_size);
3134a285bdaSBarry Smith     fclose(fd);
3144a285bdaSBarry Smith     body[blen + rd] = 0;
315d0609cedSBarry Smith     PetscCall(PetscStrcat(body,"\r\n\r\n"
316d0609cedSBarry Smith                                "--foo_bar_baz\r\n"));
3179566063dSJacob Faibussowitsch     PetscCall(PetscSSLInitializeContext(&ctx));
3189566063dSJacob Faibussowitsch     PetscCall(PetscHTTPSConnect("www.boxapis.com",443,ctx,&sock,&ssl));
3199566063dSJacob Faibussowitsch     PetscCall(PetscHTTPSRequest("POST","www.boxapis.com/upload/drive/v2/files/",head,"multipart/related; boundary=\"foo_bar_baz\"",body,ssl,buff,sizeof(buff)));
3209566063dSJacob Faibussowitsch     PetscCall(PetscFree(body));
3219566063dSJacob Faibussowitsch     PetscCall(PetscSSLDestroyContext(ctx));
3224a285bdaSBarry Smith     close(sock);
3239566063dSJacob Faibussowitsch     PetscCall(PetscStrstr(buff,"\"title\"",&title));
32428b400f6SJacob Faibussowitsch     PetscCheck(title,PETSC_COMM_SELF,PETSC_ERR_LIB,"Upload of file %s failed",filename);
3254a285bdaSBarry Smith   }
3264a285bdaSBarry Smith   PetscFunctionReturn(0);
3274a285bdaSBarry Smith }
328