/**
* 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 .
*
* 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 .
*
* Version: $Id$
*/
#include
#include
#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
#define USTATE_ERRORS 4
using namespace SourceMod;
UpdateReader::UpdateReader() : partFirst(NULL), partLast(NULL)
{
}
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;
}
else if (strcmp(name, "Errors") == 0)
{
ustate = USTATE_ERRORS;
}
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)
{
url.assign(update_url);
url.append(value);
}
break;
}
case USTATE_ERRORS:
{
if (strcmp(key, "error") == 0)
{
AddUpdateError("%s", value);
}
break;
}
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:
case USTATE_ERRORS:
{
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;
}
void UpdateReader::LinkPart(UpdatePart *part)
{
part->next = NULL;
if (partFirst == NULL)
{
partFirst = part;
partLast = part;
}
else
{
partLast->next = part;
partLast = part;
}
}
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);
if (mdl.GetSize() == 0)
{
AddUpdateError("Zero-length file returned for \"%s\"", curfile.c_str());
return;
}
if (strcasecmp(checksum, real_checksum) != 0)
{
AddUpdateError("Checksums for file \"%s\" do not match:", curfile.c_str());
AddUpdateError("Expected: %s Real: %s", checksum, real_checksum);
return;
}
UpdatePart *part = new UpdatePart;
part->data = (char*)malloc(mdl.GetSize());
memcpy(part->data, mdl.GetBuffer(), mdl.GetSize());
part->file = strdup(curfile.c_str());
part->length = mdl.GetSize();
LinkPart(part);
}
void UpdateReader::HandleFolder(const char *folder)
{
UpdatePart *part = new UpdatePart;
part->data = NULL;
part->length = 0;
part->file = strdup(folder);
LinkPart(part);
}
static bool md5_file(const char *file, char checksum[33])
{
MD5 md5;
FILE *fp;
long length;
void *fdata;
if ((fp = fopen(file, "rb")) == NULL)
{
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);
}
void UpdateReader::PerformUpdate(const char *url)
{
IWebForm *form;
MemoryDownloader master;
SMCStates states = {0, 0};
update_url = url;
form = webternet->CreateForm();
xfer = webternet->CreateSession();
xfer->SetFailOnHTTPError(true);
form->AddString("version", SM_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);
if (!xfer->PostAndDownload(url, form, &master, NULL))
{
AddUpdateError("Could not download \"%s\"", url);
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;
}
UpdatePart *UpdateReader::DetachParts()
{
UpdatePart *first;
first = partFirst;
partFirst = NULL;
partLast = NULL;
return first;
}