2009-02-02 21:41:25 +01:00
|
|
|
/**
|
|
|
|
* vim: set ts=4 :
|
|
|
|
* =============================================================================
|
|
|
|
* SourceMod Updater Extension
|
|
|
|
* Copyright (C) 2004-2009 AlliedModders LLC. All rights reserved.
|
|
|
|
* =============================================================================
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it under
|
|
|
|
* the terms of the GNU General Public License, version 3.0, as published by the
|
|
|
|
* Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
|
|
|
* FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
|
|
|
|
* details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along with
|
|
|
|
* this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* As a special exception, AlliedModders LLC gives you permission to link the
|
|
|
|
* code of this program (as well as its derivative works) to "Half-Life 2," the
|
|
|
|
* "Source Engine," the "SourcePawn JIT," and any Game MODs that run on software
|
|
|
|
* by the Valve Corporation. You must obey the GNU General Public License in
|
|
|
|
* all respects for all other code used. Additionally, AlliedModders LLC grants
|
|
|
|
* this exception to all derivative works. AlliedModders LLC defines further
|
|
|
|
* exceptions, found in LICENSE.txt (as of this writing, version JULY-31-2007),
|
|
|
|
* or <http://www.sourcemod.net/license.php>.
|
|
|
|
*
|
|
|
|
* Version: $Id$
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include "extension.h"
|
|
|
|
#include "Updater.h"
|
|
|
|
#include "md5.h"
|
|
|
|
|
|
|
|
#define USTATE_NONE 0
|
|
|
|
#define USTATE_FOLDERS 1
|
|
|
|
#define USTATE_CHANGED 2
|
|
|
|
#define USTATE_CHANGE_FILE 3
|
2009-03-01 07:18:09 +01:00
|
|
|
#define USTATE_ERRORS 4
|
2009-02-02 21:41:25 +01:00
|
|
|
|
|
|
|
using namespace SourceMod;
|
|
|
|
|
2009-02-17 21:59:51 +01:00
|
|
|
UpdateReader::UpdateReader() : partFirst(NULL), partLast(NULL)
|
2009-02-02 21:41:25 +01:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
UpdateReader::~UpdateReader()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void UpdateReader::ReadSMC_ParseStart()
|
|
|
|
{
|
|
|
|
ignoreLevel = 0;
|
|
|
|
ustate = USTATE_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
SMCResult UpdateReader::ReadSMC_NewSection(const SMCStates *states, const char *name)
|
|
|
|
{
|
|
|
|
if (ignoreLevel)
|
|
|
|
{
|
|
|
|
ignoreLevel++;
|
|
|
|
return SMCResult_Continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (ustate)
|
|
|
|
{
|
|
|
|
case USTATE_NONE:
|
|
|
|
{
|
|
|
|
if (strcmp(name, "Folders") == 0)
|
|
|
|
{
|
|
|
|
ustate = USTATE_FOLDERS;
|
|
|
|
}
|
|
|
|
else if (strcmp(name, "Changed") == 0)
|
|
|
|
{
|
|
|
|
ustate = USTATE_CHANGED;
|
|
|
|
}
|
2009-03-01 07:18:09 +01:00
|
|
|
else if (strcmp(name, "Errors") == 0)
|
|
|
|
{
|
|
|
|
ustate = USTATE_ERRORS;
|
|
|
|
}
|
2009-02-02 21:41:25 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
ignoreLevel++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case USTATE_FOLDERS:
|
|
|
|
case USTATE_CHANGE_FILE:
|
|
|
|
{
|
|
|
|
ignoreLevel++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case USTATE_CHANGED:
|
|
|
|
{
|
|
|
|
curfile.assign(name);
|
|
|
|
url.clear();
|
|
|
|
checksum[0] = '\0';
|
|
|
|
ustate = USTATE_CHANGE_FILE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return SMCResult_Continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
SMCResult UpdateReader::ReadSMC_KeyValue(const SMCStates *states,
|
|
|
|
const char *key,
|
|
|
|
const char *value)
|
|
|
|
{
|
|
|
|
if (ignoreLevel)
|
|
|
|
{
|
|
|
|
return SMCResult_Continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (ustate)
|
|
|
|
{
|
|
|
|
case USTATE_CHANGE_FILE:
|
|
|
|
{
|
|
|
|
if (strcmp(key, "md5sum") == 0)
|
|
|
|
{
|
|
|
|
if (strlen(value) != 32)
|
|
|
|
{
|
|
|
|
return SMCResult_Continue;
|
|
|
|
}
|
|
|
|
strcpy(checksum, value);
|
|
|
|
}
|
|
|
|
else if (strcmp(key, "location") == 0)
|
|
|
|
{
|
2009-02-17 22:39:02 +01:00
|
|
|
url.assign(update_url);
|
2009-02-02 21:41:25 +01:00
|
|
|
url.append(value);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2009-03-01 07:18:09 +01:00
|
|
|
case USTATE_ERRORS:
|
|
|
|
{
|
|
|
|
if (strcmp(key, "error") == 0)
|
|
|
|
{
|
|
|
|
AddUpdateError("%s", value);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2009-02-02 21:41:25 +01:00
|
|
|
case USTATE_FOLDERS:
|
|
|
|
{
|
|
|
|
HandleFolder(value);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return SMCResult_Continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
SMCResult UpdateReader::ReadSMC_LeavingSection(const SMCStates *states)
|
|
|
|
{
|
|
|
|
if (ignoreLevel)
|
|
|
|
{
|
|
|
|
ignoreLevel--;
|
|
|
|
return SMCResult_Continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (ustate)
|
|
|
|
{
|
|
|
|
case USTATE_FOLDERS:
|
|
|
|
case USTATE_CHANGED:
|
2009-03-01 07:18:09 +01:00
|
|
|
case USTATE_ERRORS:
|
2009-02-02 21:41:25 +01:00
|
|
|
{
|
|
|
|
ustate = USTATE_NONE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case USTATE_CHANGE_FILE:
|
|
|
|
{
|
|
|
|
if (url.size() != 0 && checksum[0] != '\0')
|
|
|
|
{
|
|
|
|
HandleFile();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
AddUpdateError("Incomplete file definition in update manifest");
|
|
|
|
}
|
|
|
|
ustate = USTATE_CHANGED;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return SMCResult_Continue;
|
|
|
|
}
|
|
|
|
|
2009-02-17 21:59:51 +01:00
|
|
|
void UpdateReader::LinkPart(UpdatePart *part)
|
|
|
|
{
|
|
|
|
part->next = NULL;
|
|
|
|
if (partFirst == NULL)
|
|
|
|
{
|
|
|
|
partFirst = part;
|
|
|
|
partLast = part;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
partLast->next = part;
|
|
|
|
partLast = part;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-02-02 21:41:25 +01:00
|
|
|
void UpdateReader::HandleFile()
|
|
|
|
{
|
|
|
|
MD5 md5;
|
|
|
|
char real_checksum[33];
|
|
|
|
|
|
|
|
mdl.Reset();
|
|
|
|
|
|
|
|
if (!xfer->Download(url.c_str(), &mdl, NULL))
|
|
|
|
{
|
|
|
|
AddUpdateError("Could not download \"%s\"", url.c_str());
|
|
|
|
AddUpdateError("Error: %s", xfer->LastErrorMessage());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
md5.update((unsigned char *)mdl.GetBuffer(), mdl.GetSize());
|
|
|
|
md5.finalize();
|
|
|
|
md5.hex_digest(real_checksum);
|
|
|
|
|
2009-02-17 21:59:51 +01:00
|
|
|
if (mdl.GetSize() == 0)
|
2009-02-02 21:41:25 +01:00
|
|
|
{
|
2009-02-17 21:59:51 +01:00
|
|
|
AddUpdateError("Zero-length file returned for \"%s\"", curfile.c_str());
|
2009-02-02 21:41:25 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-02-17 21:59:51 +01:00
|
|
|
if (strcasecmp(checksum, real_checksum) != 0)
|
2009-02-02 21:41:25 +01:00
|
|
|
{
|
2009-02-17 21:59:51 +01:00
|
|
|
AddUpdateError("Checksums for file \"%s\" do not match:", curfile.c_str());
|
|
|
|
AddUpdateError("Expected: %s Real: %s", checksum, real_checksum);
|
2009-02-02 21:41:25 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-02-17 21:59:51 +01:00
|
|
|
UpdatePart *part = new UpdatePart;
|
|
|
|
part->data = (char*)malloc(mdl.GetSize());
|
2009-03-01 07:18:09 +01:00
|
|
|
memcpy(part->data, mdl.GetBuffer(), mdl.GetSize());
|
2009-02-17 21:59:51 +01:00
|
|
|
part->file = strdup(curfile.c_str());
|
|
|
|
part->length = mdl.GetSize();
|
|
|
|
LinkPart(part);
|
2009-02-02 21:41:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void UpdateReader::HandleFolder(const char *folder)
|
|
|
|
{
|
2009-02-17 21:59:51 +01:00
|
|
|
UpdatePart *part = new UpdatePart;
|
|
|
|
part->data = NULL;
|
|
|
|
part->length = 0;
|
|
|
|
part->file = strdup(folder);
|
|
|
|
LinkPart(part);
|
2009-02-02 21:41:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool md5_file(const char *file, char checksum[33])
|
|
|
|
{
|
|
|
|
MD5 md5;
|
|
|
|
FILE *fp;
|
|
|
|
long length;
|
|
|
|
void *fdata;
|
|
|
|
|
2009-03-01 07:18:09 +01:00
|
|
|
if ((fp = fopen(file, "rb")) == NULL)
|
2009-02-02 21:41:25 +01:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
fseek(fp, 0, SEEK_END);
|
|
|
|
length = ftell(fp);
|
|
|
|
fseek(fp, 0, SEEK_SET);
|
|
|
|
fdata = malloc(length);
|
|
|
|
if (fread(fdata, 1, length, fp) != size_t(length))
|
|
|
|
{
|
|
|
|
free(fdata);
|
|
|
|
fclose(fp);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
fclose(fp);
|
|
|
|
|
|
|
|
md5.update((unsigned char*)fdata, length);
|
|
|
|
md5.finalize();
|
|
|
|
md5.hex_digest(checksum);
|
|
|
|
|
|
|
|
free(fdata);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Path should be sourcemod relative, not gamedata relative */
|
|
|
|
static bool add_file(IWebForm *form, const char *file, unsigned int &num_files)
|
|
|
|
{
|
|
|
|
char path[PLATFORM_MAX_PATH];
|
|
|
|
|
|
|
|
smutils->BuildPath(Path_SM, path, sizeof(path), "%s", file);
|
|
|
|
|
|
|
|
char checksum[33];
|
|
|
|
if (!md5_file(path, checksum))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
char name[32];
|
|
|
|
smutils->Format(name, sizeof(name), "file_%d_name", num_files);
|
|
|
|
form->AddString(name, file);
|
|
|
|
smutils->Format(name, sizeof(name), "file_%d_md5", num_files);
|
|
|
|
form->AddString(name, checksum);
|
|
|
|
|
|
|
|
num_files++;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void add_folders(IWebForm *form, const char *root, unsigned int &num_files)
|
|
|
|
{
|
|
|
|
IDirectory *dir;
|
|
|
|
char path[PLATFORM_MAX_PATH];
|
|
|
|
char name[PLATFORM_MAX_PATH];
|
|
|
|
|
|
|
|
smutils->BuildPath(Path_SM, path, sizeof(path), "%s", root);
|
|
|
|
dir = libsys->OpenDirectory(path);
|
|
|
|
if (dir == NULL)
|
|
|
|
{
|
|
|
|
AddUpdateError("Could not open folder: %s", path);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (dir->MoreFiles())
|
|
|
|
{
|
|
|
|
if (strcmp(dir->GetEntryName(), ".") == 0 ||
|
|
|
|
strcmp(dir->GetEntryName(), "..") == 0)
|
|
|
|
{
|
|
|
|
dir->NextEntry();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
smutils->Format(name, sizeof(name), "%s/%s", root, dir->GetEntryName());
|
|
|
|
if (dir->IsEntryDirectory())
|
|
|
|
{
|
|
|
|
add_folders(form, name, num_files);
|
|
|
|
}
|
|
|
|
else if (dir->IsEntryFile())
|
|
|
|
{
|
|
|
|
add_file(form, name, num_files);
|
|
|
|
}
|
|
|
|
dir->NextEntry();
|
|
|
|
}
|
|
|
|
|
|
|
|
libsys->CloseDirectory(dir);
|
|
|
|
}
|
|
|
|
|
2009-02-17 22:39:02 +01:00
|
|
|
void UpdateReader::PerformUpdate(const char *url)
|
2009-02-02 21:41:25 +01:00
|
|
|
{
|
|
|
|
IWebForm *form;
|
|
|
|
MemoryDownloader master;
|
|
|
|
SMCStates states = {0, 0};
|
|
|
|
|
2009-02-17 22:39:02 +01:00
|
|
|
update_url = url;
|
|
|
|
|
2009-02-02 21:41:25 +01:00
|
|
|
form = webternet->CreateForm();
|
|
|
|
xfer = webternet->CreateSession();
|
|
|
|
xfer->SetFailOnHTTPError(true);
|
|
|
|
|
|
|
|
form->AddString("version", SVN_FULL_VERSION);
|
|
|
|
form->AddString("build", SM_BUILD_UNIQUEID);
|
|
|
|
|
|
|
|
unsigned int num_files = 0;
|
|
|
|
add_folders(form, "gamedata", num_files);
|
|
|
|
|
|
|
|
char temp[24];
|
|
|
|
smutils->Format(temp, sizeof(temp), "%d", num_files);
|
|
|
|
form->AddString("files", temp);
|
|
|
|
|
2009-02-17 22:39:02 +01:00
|
|
|
if (!xfer->PostAndDownload(url, form, &master, NULL))
|
2009-02-02 21:41:25 +01:00
|
|
|
{
|
2009-02-17 22:39:02 +01:00
|
|
|
AddUpdateError("Could not download \"%s\"", url);
|
2009-02-02 21:41:25 +01:00
|
|
|
AddUpdateError("Error: %s", xfer->LastErrorMessage());
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
SMCError error;
|
|
|
|
char errbuf[256];
|
|
|
|
error = textparsers->ParseSMCStream(master.GetBuffer(),
|
|
|
|
master.GetSize(),
|
|
|
|
this,
|
|
|
|
&states,
|
|
|
|
errbuf,
|
|
|
|
sizeof(errbuf));
|
|
|
|
if (error != SMCError_Okay)
|
|
|
|
{
|
|
|
|
AddUpdateError("Parse error in update manifest: %s", errbuf);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
delete xfer;
|
|
|
|
delete form;
|
|
|
|
}
|
2009-02-17 21:59:51 +01:00
|
|
|
|
|
|
|
UpdatePart *UpdateReader::DetachParts()
|
|
|
|
{
|
|
|
|
UpdatePart *first;
|
|
|
|
|
|
|
|
first = partFirst;
|
|
|
|
partFirst = NULL;
|
|
|
|
partLast = NULL;
|
|
|
|
|
|
|
|
return first;
|
|
|
|
}
|