mirror of
https://github.com/sheumann/DiskBrowser.git
synced 2025-01-02 22:31:21 +00:00
Initial working support for mounting disks.
This commit is contained in:
parent
15931f20a2
commit
d9186312ce
2
Makefile
2
Makefile
@ -7,7 +7,7 @@ JSONTEST_PROG = jsontest
|
|||||||
HTTPTEST_OBJS = httptest.a hostname.a http.a readtcp.a seturl.a strcasecmp.a tcpconnection.a urlparser.a
|
HTTPTEST_OBJS = httptest.a hostname.a http.a readtcp.a seturl.a strcasecmp.a tcpconnection.a urlparser.a
|
||||||
HTTPTEST_PROG = httptest
|
HTTPTEST_PROG = httptest
|
||||||
|
|
||||||
DISKBROWSER_OBJS = diskbrowser.a hostname.a http.a json.a jsonutil.a readtcp.a seturl.a strcasecmp.a tcpconnection.a urlparser.a
|
DISKBROWSER_OBJS = diskbrowser.a hostname.a http.a json.a jsonutil.a readtcp.a seturl.a strcasecmp.a tcpconnection.a urlparser.a asprintf.a
|
||||||
DISKBROWSER_RSRC = diskbrowser.rez
|
DISKBROWSER_RSRC = diskbrowser.rez
|
||||||
DISKBROWSER_PROG = DiskBrowser
|
DISKBROWSER_PROG = DiskBrowser
|
||||||
|
|
||||||
|
55
asprintf.c
Normal file
55
asprintf.c
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
#ifdef __ORCAC__
|
||||||
|
#pragma optimize -1 /* must be at least 8 */
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdarg.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <limits.h>
|
||||||
|
|
||||||
|
int asprintf(char **ret, const char *format, ...) {
|
||||||
|
va_list va;
|
||||||
|
int i;
|
||||||
|
int length = -1;
|
||||||
|
|
||||||
|
*ret = NULL;
|
||||||
|
|
||||||
|
for (i = 0; i < 2; i++) {
|
||||||
|
va_start(va, format);
|
||||||
|
length = vsnprintf(*ret, length+1, format, va);
|
||||||
|
if (length <= 0 || length == INT_MAX) {
|
||||||
|
length = -1;
|
||||||
|
if (i > 0)
|
||||||
|
free(*ret);
|
||||||
|
*ret = NULL;
|
||||||
|
goto done;
|
||||||
|
}
|
||||||
|
if (i == 0) {
|
||||||
|
*ret = malloc(length+1);
|
||||||
|
if (*ret == NULL) {
|
||||||
|
length = -1;
|
||||||
|
goto done;
|
||||||
|
}
|
||||||
|
#ifndef __ORCAC__
|
||||||
|
/* Should be here per standard, but ORCA/C allows only one va_end */
|
||||||
|
va_end(va);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
done:
|
||||||
|
va_end(va);
|
||||||
|
return length;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef TEST_ASPRINTF
|
||||||
|
|
||||||
|
int main(void) {
|
||||||
|
char *s;
|
||||||
|
|
||||||
|
asprintf(&s, "testing %i %i %x", 1, 2, 3);
|
||||||
|
printf("%s\n", s);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
6
asprintf.h
Normal file
6
asprintf.h
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
#ifndef ASPRINTF_H
|
||||||
|
#define ASPRINTF_H
|
||||||
|
|
||||||
|
int asprintf(char **ret, const char *format, ...);
|
||||||
|
|
||||||
|
#endif
|
207
diskbrowser.c
207
diskbrowser.c
@ -25,6 +25,7 @@
|
|||||||
#include "http.h"
|
#include "http.h"
|
||||||
#include "readtcp.h"
|
#include "readtcp.h"
|
||||||
#include "tcpconnection.h"
|
#include "tcpconnection.h"
|
||||||
|
#include "asprintf.h"
|
||||||
#include "json.h"
|
#include "json.h"
|
||||||
#include "jsonutil.h"
|
#include "jsonutil.h"
|
||||||
|
|
||||||
@ -45,6 +46,7 @@ char finderRequestName[] = "\pApple~Finder~";
|
|||||||
#define mountDiskButton 1008
|
#define mountDiskButton 1008
|
||||||
|
|
||||||
#define searchErrorAlert 3000
|
#define searchErrorAlert 3000
|
||||||
|
#define mountErrorAlert 3001
|
||||||
|
|
||||||
Word resourceFileID;
|
Word resourceFileID;
|
||||||
|
|
||||||
@ -82,12 +84,16 @@ struct diskListEntry {
|
|||||||
char *memPtr;
|
char *memPtr;
|
||||||
Byte memFlag;
|
Byte memFlag;
|
||||||
char *idPtr;
|
char *idPtr;
|
||||||
|
char *extPtr;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct diskListEntry diskList[DISK_LIST_LENGTH];
|
struct diskListEntry diskList[DISK_LIST_LENGTH];
|
||||||
|
|
||||||
int diskListPos = 0;
|
int diskListPos = 0;
|
||||||
|
|
||||||
|
char *wantedExt;
|
||||||
|
char *wantedItemID;
|
||||||
|
|
||||||
static struct MountURLRec mountURLRec = {sizeof(struct MountURLRec)};
|
static struct MountURLRec mountURLRec = {sizeof(struct MountURLRec)};
|
||||||
|
|
||||||
Session sess = {0};
|
Session sess = {0};
|
||||||
@ -224,11 +230,13 @@ boolean processDoc(json_value *docObj) {
|
|||||||
return false;
|
return false;
|
||||||
json_value *id = findEntryInObject(docObj, "identifier", json_string);
|
json_value *id = findEntryInObject(docObj, "identifier", json_string);
|
||||||
json_value *title = findEntryInObject(docObj, "title", json_string);
|
json_value *title = findEntryInObject(docObj, "title", json_string);
|
||||||
if (id == NULL || title == NULL)
|
json_value *ext = findEntryInObject(docObj, "emulator_ext", json_string);
|
||||||
|
if (id == NULL || title == NULL || ext == NULL)
|
||||||
return true;
|
return true;
|
||||||
diskList[diskListPos].idPtr = id->u.string.ptr;
|
diskList[diskListPos].idPtr = id->u.string.ptr;
|
||||||
// TODO character set translation
|
// TODO character set translation
|
||||||
diskList[diskListPos++].memPtr = title->u.string.ptr;
|
diskList[diskListPos].memPtr = title->u.string.ptr;
|
||||||
|
diskList[diskListPos++].extPtr = ext->u.string.ptr;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -261,60 +269,23 @@ char *EncodeQueryString(char *queryString) {
|
|||||||
return encodedStr;
|
return encodedStr;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ShowErrorAlert(enum NetDiskError error) {
|
void ShowErrorAlert(enum NetDiskError error, int alertNumber) {
|
||||||
char numStr[6] = "";
|
char numStr[6] = "";
|
||||||
char *subs[1] = {numStr};
|
char *subs[1] = {numStr};
|
||||||
|
|
||||||
snprintf(numStr, sizeof(numStr), "%u", error);
|
snprintf(numStr, sizeof(numStr), "%u", error);
|
||||||
|
|
||||||
AlertWindow(awResource+awCString+awButtonLayout,
|
AlertWindow(awResource+awCString+awButtonLayout,
|
||||||
(Pointer)subs, searchErrorAlert);
|
(Pointer)subs, alertNumber);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Do a search */
|
|
||||||
void DoSearch(void) {
|
|
||||||
char *searchURL = NULL;
|
|
||||||
int urlLength = 0;
|
|
||||||
enum NetDiskError result = 0;
|
|
||||||
|
|
||||||
WaitCursor();
|
enum NetDiskError ReadJSONFromURL(char *url, json_value** jsonResult) {
|
||||||
|
enum NetDiskError result = OPERATION_SUCCESSFUL;
|
||||||
|
|
||||||
GetLETextByID(window, searchLine, (StringPtr)&queryBuf);
|
*jsonResult = NULL;
|
||||||
|
|
||||||
char *queryString = EncodeQueryString(queryBuf+1);
|
result = SetURL(&sess, url, FALSE, FALSE);
|
||||||
if (queryString == NULL) {
|
|
||||||
result = OUT_OF_MEMORY;
|
|
||||||
goto errorReturn;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (int i = 0; i < 2; i++) {
|
|
||||||
urlLength = snprintf(searchURL, urlLength,
|
|
||||||
"http://archive.org/advancedsearch.php?"
|
|
||||||
"q=emulator%%3A%s%%20%s"
|
|
||||||
"&fl%%5B%%5D=identifier&fl%%5B%%5D=title"
|
|
||||||
"&fl%%5B%%5D=emulator_ext"
|
|
||||||
"&sort%%5B%%5D=titleSorterRaw+asc"
|
|
||||||
"&rows=%i&page=%i&output=json",
|
|
||||||
gsDisksOnly ? "apple2gs" : "apple2%2A",
|
|
||||||
queryString,
|
|
||||||
DISK_LIST_LENGTH,
|
|
||||||
pageNum);
|
|
||||||
if (urlLength <= 0) {
|
|
||||||
result = URL_TOO_LONG;
|
|
||||||
goto errorReturn;
|
|
||||||
}
|
|
||||||
if (i == 0) {
|
|
||||||
searchURL = malloc(urlLength);
|
|
||||||
if (searchURL == NULL) {
|
|
||||||
result = OUT_OF_MEMORY;
|
|
||||||
goto errorReturn;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
free(queryString);
|
|
||||||
queryString = NULL;
|
|
||||||
|
|
||||||
result = SetURL(&sess, searchURL, FALSE, FALSE);
|
|
||||||
if (result != OPERATION_SUCCESSFUL)
|
if (result != OPERATION_SUCCESSFUL)
|
||||||
goto errorReturn;
|
goto errorReturn;
|
||||||
|
|
||||||
@ -343,15 +314,58 @@ void DoSearch(void) {
|
|||||||
goto errorReturn;
|
goto errorReturn;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
*jsonResult = json_parse(netBuf, sess.contentLength);
|
||||||
if (json)
|
if (*jsonResult == NULL) {
|
||||||
json_value_free(json);
|
|
||||||
json = json_parse(netBuf, sess.contentLength);
|
|
||||||
if (json == NULL) {
|
|
||||||
result = JSON_PARSING_ERROR;
|
result = JSON_PARSING_ERROR;
|
||||||
goto errorReturn;
|
goto errorReturn;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
errorReturn:
|
||||||
|
free(netBuf);
|
||||||
|
netBuf = NULL;
|
||||||
|
EndTCPConnection(&sess);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Do a search */
|
||||||
|
void DoSearch(void) {
|
||||||
|
char *searchURL = NULL;
|
||||||
|
enum NetDiskError result = 0;
|
||||||
|
|
||||||
|
WaitCursor();
|
||||||
|
|
||||||
|
GetLETextByID(window, searchLine, (StringPtr)&queryBuf);
|
||||||
|
|
||||||
|
char *queryString = EncodeQueryString(queryBuf+1);
|
||||||
|
if (queryString == NULL) {
|
||||||
|
result = OUT_OF_MEMORY;
|
||||||
|
goto errorReturn;
|
||||||
|
}
|
||||||
|
|
||||||
|
asprintf(&searchURL,
|
||||||
|
"http://archive.org/advancedsearch.php?"
|
||||||
|
"q=emulator%%3A%s%%20%s"
|
||||||
|
"&fl%%5B%%5D=identifier&fl%%5B%%5D=title"
|
||||||
|
"&fl%%5B%%5D=emulator_ext"
|
||||||
|
"&sort%%5B%%5D=titleSorterRaw+asc"
|
||||||
|
"&rows=%i&page=%i&output=json",
|
||||||
|
gsDisksOnly ? "apple2gs" : "apple2%2A",
|
||||||
|
queryString,
|
||||||
|
DISK_LIST_LENGTH,
|
||||||
|
pageNum);
|
||||||
|
if (searchURL == NULL) {
|
||||||
|
result = URL_TOO_LONG;
|
||||||
|
goto errorReturn;
|
||||||
|
}
|
||||||
|
free(queryString);
|
||||||
|
queryString = NULL;
|
||||||
|
|
||||||
|
if (json)
|
||||||
|
json_value_free(json);
|
||||||
|
result = ReadJSONFromURL(searchURL, &json);
|
||||||
|
if (result != OPERATION_SUCCESSFUL)
|
||||||
|
goto errorReturn;
|
||||||
|
|
||||||
json_value *response = findEntryInObject(json, "response", json_object);
|
json_value *response = findEntryInObject(json, "response", json_object);
|
||||||
if (response == NULL) {
|
if (response == NULL) {
|
||||||
result = NOT_EXPECTED_CONTENTS;
|
result = NOT_EXPECTED_CONTENTS;
|
||||||
@ -388,7 +402,6 @@ void DoSearch(void) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
free(searchURL);
|
free(searchURL);
|
||||||
EndTCPConnection(&sess);
|
|
||||||
InitCursor();
|
InitCursor();
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -396,11 +409,97 @@ errorReturn:
|
|||||||
NewList2(NULL, 1, (Ref) diskList, refIsPointer, 0, (Handle)disksListHandle);
|
NewList2(NULL, 1, (Ref) diskList, refIsPointer, 0, (Handle)disksListHandle);
|
||||||
free(queryString);
|
free(queryString);
|
||||||
free(searchURL);
|
free(searchURL);
|
||||||
EndTCPConnection(&sess);
|
|
||||||
InitCursor();
|
InitCursor();
|
||||||
ShowErrorAlert(result);
|
ShowErrorAlert(result, searchErrorAlert);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void MountFile(char *itemID, char *fileName) {
|
||||||
|
char *fileURL = NULL;
|
||||||
|
asprintf(&fileURL, "http://archive.org/download/%s/%s", itemID, fileName);
|
||||||
|
if (fileURL == NULL) {
|
||||||
|
SysBeep();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
mountURLRec.result = NETDISK_NOT_PRESENT;
|
||||||
|
mountURLRec.url = fileURL;
|
||||||
|
mountURLRec.flags = flgUseCache;
|
||||||
|
|
||||||
|
SendRequest(MountURL, sendToName|stopAfterOne, (Long)NETDISK_REQUEST_NAME,
|
||||||
|
(Long)&mountURLRec, NULL);
|
||||||
|
|
||||||
|
free(fileURL);
|
||||||
|
}
|
||||||
|
|
||||||
|
boolean processFile(json_value *fileObj) {
|
||||||
|
if (fileObj == NULL || fileObj->type != json_object)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
json_value *name = findEntryInObject(fileObj, "name", json_string);
|
||||||
|
if (name == NULL)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
char *nameExt = strrchr(name->u.string.ptr, '.');
|
||||||
|
if (nameExt == NULL)
|
||||||
|
return true;
|
||||||
|
if (strcmp(nameExt + 1, wantedExt) == 0) {
|
||||||
|
MountFile(wantedItemID, name->u.string.ptr);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void DoMount(void) {
|
||||||
|
unsigned int itemNumber = NextMember2(0, (Handle)disksListHandle);
|
||||||
|
char *filesURL = NULL;
|
||||||
|
enum NetDiskError result = 0;
|
||||||
|
json_value *filesJSON = NULL;
|
||||||
|
|
||||||
|
if (itemNumber == 0) {
|
||||||
|
// shouldn't happen
|
||||||
|
result = 1;
|
||||||
|
goto errorReturn;
|
||||||
|
}
|
||||||
|
itemNumber--;
|
||||||
|
|
||||||
|
if (diskList[itemNumber].idPtr == NULL) {
|
||||||
|
result = 2;
|
||||||
|
goto errorReturn;
|
||||||
|
}
|
||||||
|
|
||||||
|
asprintf(&filesURL,
|
||||||
|
"http://archive.org/metadata/%s/files",
|
||||||
|
diskList[itemNumber].idPtr);
|
||||||
|
if (filesURL == NULL) {
|
||||||
|
result = 3;
|
||||||
|
goto errorReturn;
|
||||||
|
}
|
||||||
|
|
||||||
|
result = ReadJSONFromURL(filesURL, &filesJSON);
|
||||||
|
if (result != OPERATION_SUCCESSFUL)
|
||||||
|
goto errorReturn;
|
||||||
|
|
||||||
|
json_value *resultJSON = findEntryInObject(filesJSON, "result", json_array);
|
||||||
|
if (resultJSON == NULL) {
|
||||||
|
result = NOT_EXPECTED_CONTENTS;
|
||||||
|
goto errorReturn;
|
||||||
|
}
|
||||||
|
|
||||||
|
wantedExt = diskList[itemNumber].extPtr;
|
||||||
|
wantedItemID = diskList[itemNumber].idPtr;
|
||||||
|
processArray(resultJSON, json_object, processFile);
|
||||||
|
|
||||||
|
errorReturn:
|
||||||
|
if (result != 0)
|
||||||
|
ShowErrorAlert(result, mountErrorAlert);
|
||||||
|
if (filesJSON != NULL)
|
||||||
|
json_value_free(filesJSON);
|
||||||
|
free(filesURL);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/* Update state of controls based on user input */
|
/* Update state of controls based on user input */
|
||||||
void UpdateControlState(void) {
|
void UpdateControlState(void) {
|
||||||
unsigned long targetCtlID = GetCtlID(FindTargetCtl());
|
unsigned long targetCtlID = GetCtlID(FindTargetCtl());
|
||||||
@ -448,7 +547,7 @@ void HandleEvent(int eventCode, WmTaskRec *taskRec) {
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case mountDiskButton:
|
case mountDiskButton:
|
||||||
// TODO
|
DoMount();
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -123,7 +123,7 @@ resource rControlTemplate (disksList) {
|
|||||||
$0000, /* List Type */
|
$0000, /* List Type */
|
||||||
0, /* List Start */
|
0, /* List Start */
|
||||||
10, /* ListMemHeight */
|
10, /* ListMemHeight */
|
||||||
9, /* List Mem Size */
|
13, /* List Mem Size */
|
||||||
0, /* List Ref */
|
0, /* List Ref */
|
||||||
0 /* Color Ref */
|
0 /* Color Ref */
|
||||||
}};
|
}};
|
||||||
@ -150,9 +150,17 @@ resource rPString(mountDiskButton) { "Mount Disk" };
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#define searchErrorAlert 3000
|
#define searchErrorAlert 3000
|
||||||
|
#define mountErrorAlert 3001
|
||||||
|
|
||||||
resource rAlertString (searchErrorAlert) {
|
resource rAlertString (searchErrorAlert) {
|
||||||
"52/"
|
"52/"
|
||||||
"There was an error when searching for disks. (error code *0)"
|
"There was an error when searching for disks. (error code *0)"
|
||||||
"/^#0\$00"
|
"/^#0\$00"
|
||||||
};
|
};
|
||||||
|
|
||||||
|
resource rAlertString (mountErrorAlert) {
|
||||||
|
"52/"
|
||||||
|
"There was an error when trying to mount a disk. (error code *0)"
|
||||||
|
"/^#0\$00"
|
||||||
|
};
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user