/*
* The Mana World
* Copyright 2004 The Mana World Development Team
*
* This file is part of The Mana World.
*
* The Mana World is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* any later version.
*
* The Mana World 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 The Mana World; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
* $Id$
*/
#include "updatewindow.h"
#include <iostream>
#include <sstream>
#include <SDL.h>
#include <SDL_thread.h>
#include <curl/curl.h>
#include <guichan/widgets/label.hpp>
#include "browserbox.h"
#include "button.h"
#include "progressbar.h"
#include "scrollarea.h"
#include "../configuration.h"
#include "../log.h"
#include "../main.h"
UpdaterWindow::UpdaterWindow():
Window("Updating..."),
mThread(NULL), mMutex(NULL), mDownloadStatus(UPDATE_NEWS),
mUpdateHost(""), mCurrentFile("news.txt"), mBasePath(""),
mStoreInMemory(true), mDownloadComplete(true), mDownloadedBytes(0),
mMemoryBuffer(NULL), mCurlError(new char[CURL_ERROR_SIZE]),
mFileIndex(0)
{
mCurlError[0] = 0;
int h = 240;
int w = 320;
setContentSize(w, h);
mBrowserBox = new BrowserBox();
mBrowserBox->setOpaque(false);
mCancelButton = new Button("Cancel");
mCancelButton->setPosition(5, h - 5 - mCancelButton->getHeight());
mCancelButton->setEventId("cancel");
mCancelButton->addActionListener(this);
mPlayButton = new Button("Play");
mPlayButton->setPosition(mCancelButton->getX() +
mCancelButton->getWidth() + 5,
h - 5 - mPlayButton->getHeight());
mPlayButton->setEventId("play");
mPlayButton->setEnabled(false);
mPlayButton->addActionListener(this);
mProgressBar = new ProgressBar(0.0, 5, mCancelButton->getY() - 20 - 5,
w - 10, 20, 37, 70, 200);
mLabel = new gcn::Label("Connecting...");
mLabel->setPosition(5, mProgressBar->getY() - mLabel->getHeight() - 5);
mScrollArea = new ScrollArea(mBrowserBox);
mScrollArea->setDimension(gcn::Rectangle(5, 5, 310,
mLabel->getY() - 12));
add(mScrollArea);
add(mLabel);
add(mProgressBar);
add(mCancelButton);
add(mPlayButton);
mCancelButton->requestFocus();
setLocationRelativeTo(getParent());
// That commented line causes problems if the client had a false value for an update host
// in its config file.
// The update host, unlike the home dir, isn't dynamically set at the start of the client.
//mUpdateHost = config.getValue("updatehost", "themanaworld.org/files");
mUpdateHost = "themanaworld.org/files";
mBasePath = config.getValue("homeDir", ".");
// Try to download the updates list
download();
}
UpdaterWindow::~UpdaterWindow()
{
if (mThread)
{
SDL_WaitThread(mThread, NULL);
mThread = NULL;
}
free(mMemoryBuffer);
// Remove downloaded files
remove((mBasePath + "/updates/news.txt").c_str());
remove((mBasePath + "/updates/resources.txt").c_str());
remove((mBasePath + "/updates/download.temp").c_str());
delete [] mCurlError;
delete mScrollArea;
delete mLabel;
delete mProgressBar;
delete mCancelButton;
delete mPlayButton;
}
void UpdaterWindow::setProgress(float p)
{
mProgressBar->setProgress(p);
}
void UpdaterWindow::setLabel(const std::string &str)
{
mLabel->setCaption(str);
mLabel->adjustSize();
}
void UpdaterWindow::enable()
{
mPlayButton->setEnabled(true);
mPlayButton->requestFocus();
}
void UpdaterWindow::action(const std::string& eventId)
{
if (eventId == "cancel") {
// Skip the updating process
if (mDownloadStatus == UPDATE_COMPLETE)
{
state = EXIT;
}
else {
mDownloadStatus = UPDATE_ERROR;
}
}
else if (eventId == "play") {
state = LOGIN;
}
}
void UpdaterWindow::loadNews()
{
int contentsLength = mDownloadedBytes;
char *fileContents = mMemoryBuffer;
if (!fileContents)
{
logger->log("Couldn't load news");
return;
}
// Reallocate and include terminating 0 character
fileContents = (char*)realloc(fileContents, contentsLength + 1);
fileContents[contentsLength] = '\0';
mBrowserBox->clearRows();
// Tokenize and add each line separately
char *line = strtok(fileContents, "\n");
while (line != NULL)
{
mBrowserBox->addRow(line);
line = strtok(NULL, "\n");
}
//free(fileContents);
mScrollArea->setVerticalScrollAmount(0);
setVisible(true);
}
void UpdaterWindow::addRow(const std::string &row)
{
mBrowserBox->addRow(row);
mScrollArea->setVerticalScrollAmount(mScrollArea->getVerticalMaxScroll());
}
int UpdaterWindow::updateProgress(void *ptr,
double dt, double dn, double ut, double un)
{
float progress = dn / dt;
UpdaterWindow *uw = reinterpret_cast<UpdaterWindow *>(ptr);
if (progress < 0) progress = 0.0f;
if (progress > 1) progress = 1.0f;
std::stringstream progressString;
progressString << uw->mCurrentFile
<< " (" << ((int)(progress * 100)) << "%)";
uw->setLabel(progressString.str().c_str());
uw->setProgress(progress);
if (state != UPDATE || uw->mDownloadStatus == UPDATE_ERROR) {
// If the action was canceled return an error code to stop the mThread
return -1;
}
return 0;
}
size_t UpdaterWindow::memoryWrite(void *ptr,
size_t size, size_t nmemb, FILE *stream)
{
UpdaterWindow *uw = reinterpret_cast<UpdaterWindow *>(stream);
size_t totalMem = size * nmemb;
uw->mMemoryBuffer = (char*)realloc(uw->mMemoryBuffer,
uw->mDownloadedBytes + totalMem + 1);
if (uw->mMemoryBuffer)
{
memcpy(&(uw->mMemoryBuffer[uw->mDownloadedBytes]), ptr, totalMem);
uw->mDownloadedBytes += totalMem;
// Make sure the memory buffer is NULL terminated, because this
// function is used to download text files that are later parsed as a
// string.
uw->mMemoryBuffer[uw->mDownloadedBytes] = 0;
}
return totalMem;
}
int UpdaterWindow::downloadThread(void *ptr)
{
CURL *curl;
CURLcode res;
FILE *outfile = NULL;
UpdaterWindow *uw = reinterpret_cast<UpdaterWindow *>(ptr);
std::string outFilename;
std::string url(uw->mUpdateHost + "/" + uw->mCurrentFile);
uw->setLabel(uw->mCurrentFile + " (0%)");
curl = curl_easy_init();
if (curl)
{
logger->log("Downloading: %s", url.c_str());
if (uw->mStoreInMemory)
{
uw->mDownloadedBytes = 0;
curl_easy_setopt(curl, CURLOPT_FAILONERROR, 1);
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION,
UpdaterWindow::memoryWrite);
curl_easy_setopt(curl, CURLOPT_WRITEDATA, ptr);
}
else
{
// Download in the proper folder : ./updates under win,
// /home/user/.tmw/updates for unices
outFilename = uw->mBasePath + "/updates/download.temp";
outfile = fopen(outFilename.c_str(), "wb");
curl_easy_setopt(curl, CURLOPT_WRITEDATA, outfile);
}
curl_easy_setopt(curl, CURLOPT_ERRORBUFFER, uw->mCurlError);
curl_easy_setopt(curl, CURLOPT_URL, url.c_str());
curl_easy_setopt(curl, CURLOPT_NOPROGRESS, 0);
curl_easy_setopt(curl, CURLOPT_PROGRESSFUNCTION,
UpdaterWindow::updateProgress);
curl_easy_setopt(curl, CURLOPT_PROGRESSDATA, ptr);
curl_easy_setopt(curl, CURLOPT_NOSIGNAL, 1);
curl_easy_setopt(curl, CURLOPT_CONNECTTIMEOUT, 15);
if ((res = curl_easy_perform(curl)) != 0)
{
uw->mDownloadStatus = UPDATE_ERROR;
std::cerr << "curl error " << res << " : " << uw->mCurlError
<< std::endl;
}
curl_easy_cleanup(curl);
uw->mDownloadComplete = true;
if (!uw->mStoreInMemory)
{
fclose(outfile);
// If the download was successful give the file the proper name
// else it will be deleted later
std::string newName(uw->mBasePath + "/updates/" +
uw->mCurrentFile.c_str());
rename(outFilename.c_str(), newName.c_str());
}
}
return 0;
}
void UpdaterWindow::download()
{
mDownloadComplete = false;
mThread = SDL_CreateThread(UpdaterWindow::downloadThread, this);
if (mThread == NULL) {
logger->log("Unable to create mThread");
mDownloadStatus = UPDATE_ERROR;
}
}
void updateInputHandler(SDL_KeyboardEvent *keyEvent)
{
if (keyEvent->keysym.sym == SDLK_ESCAPE)
{
state = EXIT;
}
}
void UpdaterWindow::logic()
{
switch (mDownloadStatus) {
case UPDATE_ERROR:
if (mThread)
{
SDL_WaitThread(mThread, NULL);
mThread = NULL;
}
addRow("");
addRow("##1 The update process is incomplete.");
addRow("##1 It is strongly recommended that");
addRow("##1 you try again later");
addRow(mCurlError);
mDownloadStatus = UPDATE_COMPLETE;
break;
case UPDATE_NEWS:
if (mDownloadComplete) {
// Try to open news.txt
loadNews();
// Doesn't matter if it couldn't find news.txt,
// go to the next step
mCurrentFile = "resources.txt";
if (mMemoryBuffer != NULL)
{
free(mMemoryBuffer);
mMemoryBuffer = NULL;
}
download();
mDownloadStatus = UPDATE_LIST;
}
break;
case UPDATE_LIST:
if (mDownloadComplete) {
if (mMemoryBuffer != NULL)
{
// Tokenize and add each line separately
char *line = strtok(mMemoryBuffer, "\n");
while (line != NULL)
{
mFiles.push_back(line);
line = strtok(NULL, "\n");
}
mStoreInMemory = false;
mDownloadStatus = UPDATE_RESOURCES;
}
else {
logger->log("Unable to download resources.txt");
mDownloadStatus = UPDATE_ERROR;
}
}
break;
case UPDATE_RESOURCES:
if (mDownloadComplete)
{
if (mThread)
{
SDL_WaitThread(mThread, NULL);
mThread = NULL;
}
if (mFileIndex < mFiles.size())
{
mCurrentFile = mFiles[mFileIndex];
std::ifstream temp(
(mBasePath + "/updates/" + mCurrentFile).c_str());
if (!temp.is_open()) {
temp.close();
download();
}
else {
logger->log("%s already here", mCurrentFile.c_str());
}
mFileIndex++;
}
else {
// Download of updates completed
mDownloadStatus = UPDATE_COMPLETE;
}
}
break;
case UPDATE_COMPLETE:
enable();
setLabel("Completed");
break;
case UPDATE_IDLE:
break;
}
}