changed: upload function

select file type by the extension
         param: multi
fixed:   uptime function
         it showed incorrect uptime time (minuts were badly calculated)


git-svn-id: svn://ttmath.org/publicrep/winix/trunk@575 e52654a7-88a9-db11-a3e9-0013d4bc506e
This commit is contained in:
Tomasz Sowa 2010-02-12 23:16:13 +00:00
parent 796985472a
commit 3c5a7cd664
16 changed files with 275 additions and 81 deletions

View File

@ -476,7 +476,7 @@ void Content::ReadAdditionalInfo()
void Content::PrepareUrl(Item & item) void Content::PrepareUrl(Item & item)
{ {
TrimWhite(item.url); TrimWhite(item.url);
if( item.url.empty() ) if( item.url.empty() )
item.url = item.subject; // if the subject is empty then the url will be corrected by CorrectUrlOnlyAllowedChar() item.url = item.subject; // if the subject is empty then the url will be corrected by CorrectUrlOnlyAllowedChar()

View File

@ -110,8 +110,13 @@ class Content
void PostFunCreateThreadLogAndRedirect(); void PostFunCreateThreadLogAndRedirect();
void PostFunCreateThread(); void PostFunCreateThread();
void UploadSaveFile(); bool UploadCreatePath(std::string & path);
void UploadSaveFile(const std::string & tmp_filename, const std::string & destination);
void UploadMulti();
void UploadSingle();
bool FunUploadCheckAbuse();
void PostFunUpload(); void PostFunUpload();
std::string tmp_path, tmp_path2;
bool FunCreateTicketCheckAccess(); bool FunCreateTicketCheckAccess();
void FunCreateTicket(); void FunCreateTicket();

View File

@ -14,6 +14,7 @@
#include "content.h" #include "content.h"
#include "../core/request.h" #include "../core/request.h"
#include "../core/data.h" #include "../core/data.h"
#include "../core/misc.h"
@ -38,37 +39,35 @@ return true;
void Content::UploadSaveFile() bool Content::UploadCreatePath(std::string & path)
{ {
static std::string path; path.clear();
struct stat sb;
request.MakePath(path); request.MakePath(path);
if( stat(path.c_str(), &sb) < 0 ) if( !request.MakeDirsOnFS() )
{ {
if( !request.MakeDirsOnFS() ) request.status = Error::permission_denied;
{ return false;
request.status = Error::permission_denied;
return;
}
} }
return true;
}
path += '/';
path += request.item.url;
const std::string & tmp_filename = request.post_file_table.begin()->second.tmp_filename;
if( rename(tmp_filename.c_str(), path.c_str()) == 0 )
void Content::UploadSaveFile(const std::string & tmp_filename, const std::string & destination)
{
if( rename(tmp_filename.c_str(), destination.c_str()) == 0 )
{ {
log << log1 << "Content: uploaded a new file: " << path << logend; log << log1 << "Content: uploaded a new file: " << destination << logend;
} }
else else
{ {
int err = errno; int err = errno;
// !! skasowac takze plik z bazy danych // !! skasowac takze plik z bazy danych?
log << log1 << "Content: can't move the tmp file from: " << tmp_filename << ", to: " << path << ", "; log << log1 << "Content: can't move the tmp file from: " << tmp_filename << ", to: " << destination << ", ";
log.SystemErr(err); log.SystemErr(err);
log << logend; log << logend;
@ -78,7 +77,120 @@ struct stat sb;
bool Content::FunUploadCheckAbuse()
{
if( !CheckRebus() )
{
request.status = Error::incorrect_rebus;
request.session->done = Done::added_thread;
request.session->done_status = Error::incorrect_rebus;
return false;
}
CheckGetPostTimes(4);
if( request.session->spam_score > 0 )
{
request.status = Error::spam;
request.session->done = Done::added_thread;
request.session->done_status = Error::spam;
log << log1 << "Content: ignoring due to suspected spamming" << logend;
return false;
}
return true;
}
void Content::UploadMulti()
{
request.item.Clear(); // clearing and setting date
request.item.parent_id = request.dir_table.back()->id;
request.item.type = Item::file;
request.item.privileges = 0644; // !! tymczasowo
SetUser(request.item);
if( !UploadCreatePath(tmp_path) )
return;
tmp_path2 = tmp_path; // remember the path
PostFileTable::iterator i = request.post_file_table.begin();
for( ; i != request.post_file_table.end() ; ++i)
{
const char * file_name = i->second.filename.c_str();
request.item.subject = file_name;
request.item.url = file_name;
request.item.static_auth = SelectFileType(file_name);
PrepareUrl(request.item);
PostFunEmacsAdd(); // always adding a new item
if( request.session->done_status == Error::ok )
{
tmp_path += '/';
tmp_path += request.item.url; // item.url could have been changed
UploadSaveFile(i->second.tmp_filename, tmp_path);
tmp_path = tmp_path2; // restoring the oryginal path
}
}
RedirectToLastDir();
}
void Content::UploadSingle()
{
std::string * new_subject = request.PostVar("subject");
std::string * new_url = request.PostVar("url");
bool has_subject = (new_subject && (*new_subject)[0] != 0 );
bool has_url = (new_url && (*new_url)[0] != 0 );
ReadItem(request.item, Item::file); // ReadItem() changes the url if it is empty
request.item.privileges = 0644; // !! tymczasowo
const char * file_name = request.post_file_table.begin()->second.filename.c_str();
request.item.static_auth = SelectFileType(file_name);
if( !has_subject )
request.item.subject = file_name;
if( !has_url )
{
request.item.url = file_name;
PrepareUrl(request.item);
}
PostFunEmacsAdd(); // always adding a new item
if( request.session->done_status == Error::ok )
{
const std::string & tmp_filename = request.post_file_table.begin()->second.tmp_filename;
if( !UploadCreatePath(tmp_path) )
return;
tmp_path += '/';
tmp_path += request.item.url; // item.url could have been changed
UploadSaveFile(tmp_filename, tmp_path);
}
if( request.session->done_status == Error::ok )
RedirectTo(request.item);
}
// !! dodac usuwanie plikow statycznych przez rm
void Content::PostFunUpload() void Content::PostFunUpload()
{ {
if( !FunUploadCheckAccess() ) if( !FunUploadCheckAccess() )
@ -89,23 +201,14 @@ void Content::PostFunUpload()
request.status = Error::permission_denied; request.status = Error::permission_denied;
return; return;
} }
// !! moze request.session->done_status trzeba ustawic na Error::ok na poczatku? (i podobnie w innych metodach, mkdir, emacs ...)
ReadItem(request.item, Item::file);
// !! sprawdzanie rebusa? if( !FunUploadCheckAbuse() )
return;
// !! tutaj w zaleznosci od rozszerzenia dobrac odpowiedni static_auth if( request.post_file_table.size() > 1 )
request.item.static_auth = Item::static_other; UploadMulti();
else
PostFunEmacsAdd(); // always adding a new item UploadSingle();
if( request.session->done_status == Error::ok )
UploadSaveFile();
if( request.session->done_status == Error::ok )
RedirectTo(request.item);
} }

View File

@ -72,9 +72,10 @@ long content_id;
// the direct url is the same but the prefix is: base_url_static_auth // the direct url is the same but the prefix is: base_url_static_auth
enum StaticAuth enum StaticAuth
{ {
static_none = 0, static_none = 0,
static_image = 1, static_image = 1, /* png, gif, jpg - only types available to render by a web browser*/
static_other = 2 static_document = 2, /* pdf doc xls txt */
static_other = 3
}; };

View File

@ -50,6 +50,9 @@ return false;
} }
// this function checks how many dots there are in the url
// if there are more than one (last) dot then the first dots will be changed into '_'
void CorrectUrlDots(std::string & url) void CorrectUrlDots(std::string & url)
{ {
size_t i = url.size(); size_t i = url.size();
@ -73,7 +76,7 @@ void CorrectUrlChars(std::string & url)
{ {
std::string::iterator i; std::string::iterator i;
for(i = url.begin(); i!=url.end() ; ++i) for(i=url.begin(); i != url.end() ; ++i)
{ {
if( !CorrectUrlChar(*i) ) if( !CorrectUrlChar(*i) )
{ {
@ -215,6 +218,9 @@ void HtmlEscapeFormTxt(std::ostringstream & out, const std::string & in)
std::string::const_iterator i; std::string::const_iterator i;
int was_enter = 0; // how many enteres there were before int was_enter = 0; // how many enteres there were before
if( in.empty() )
return;
out << "<p>"; out << "<p>";
// skipping first new line characters // skipping first new line characters
@ -485,15 +491,15 @@ return false;
} }
bool EqualNoCase(const char * short_str, const char * long_str) bool EqualNoCase(const char * str1, const char * str2)
{ {
while( *short_str && *long_str && ToSmall(*short_str) == ToSmall(*long_str) ) while( *str1 && *str2 && ToSmall(*str1) == ToSmall(*str2) )
{ {
++short_str; ++str1;
++long_str; ++str2;
} }
if( *short_str == 0 && *long_str == 0 ) if( *str1 == 0 && *str2 == 0 )
return true; return true;
return false; return false;
@ -555,3 +561,54 @@ bool IsFile(const std::string & file)
{ {
return IsFile(file.c_str()); return IsFile(file.c_str());
} }
// if there is not an extension it returns a pointer to the last '\0' character
const char * GetFileExt(const char * name)
{
size_t i, ilast;
// looking for the end of the name
for(i=0 ; name[i] != 0 ; ++i);
if( i == 0 )
return name; // ops, the name is empty
// remember the end of the string
ilast = i;
// looking for the last dot
for(--i ; i>0 && name[i] != '.' ; --i);
if( name[i] != '.' )
return name + ilast; // ops, there is not a dot
// the extensions starts from i+1
// and can be empty (if the last character is a dot)
return name + i + 1;
}
Item::StaticAuth SelectFileType(const char * file_name)
{
const char * ext = GetFileExt(file_name);
// as an image we're using only those types which can be rendered
// by a web browser
if( EqualNoCase(ext, "jpg") ||
EqualNoCase(ext, "gif") ||
EqualNoCase(ext, "png") )
return Item::static_image;
if( EqualNoCase(ext, "pdf") ||
EqualNoCase(ext, "doc") ||
EqualNoCase(ext, "xls") ||
EqualNoCase(ext, "txt") ||
EqualNoCase(ext, "ods") ||
EqualNoCase(ext, "odt") )
return Item::static_document;
return Item::static_other;
}

View File

@ -57,12 +57,14 @@ const char * ToStr(int value);
bool IsSubString(const char * short_str, const char * long_str); bool IsSubString(const char * short_str, const char * long_str);
bool IsSubStringNoCase(const char * short_str, const char * long_str); bool IsSubStringNoCase(const char * short_str, const char * long_str);
bool EqualNoCase(const char * short_str, const char * long_str); bool EqualNoCase(const char * str1, const char * str2);
bool ValidateEmail(const std::string & email); bool ValidateEmail(const std::string & email);
bool IsFile(const char * file); bool IsFile(const char * file);
bool IsFile(const std::string & file); bool IsFile(const std::string & file);
const char * GetFileExt(const char * name);
Item::StaticAuth SelectFileType(const char * file_name);
#endif #endif

View File

@ -911,6 +911,7 @@ void Request::MakePath(std::string & path)
} }
} }
bool Request::MakeDirsOnFS() bool Request::MakeDirsOnFS()
{ {
size_t i; size_t i;
@ -922,11 +923,13 @@ bool Request::MakeDirsOnFS()
path += '/'; path += '/';
path += dir_table[i]->url; path += dir_table[i]->url;
if( mkdir(path.c_str(), 0755) < 0 ) if( !IsFile(path.c_str()) )
{ {
// oops if( mkdir(path.c_str(), 0755) < 0 )
log << log1 << "Request: can't create the directory on fs: " << path << logend; {
return false; log << log1 << "Request: can't create the directory on fs: " << path << logend;
return false;
}
} }
} }

View File

@ -1,9 +1,13 @@
[if-one mount_type_is_cms]<p class="withnext">{form_emacs_content_cms}</p>[end]
[if-one mount_type_is_thread]<p class="withnext">{form_emacs_content_thread}</p>[end]
[if-one mount_type_is_ticket]<p class="withnext">{form_emacs_content_ticket}</p>[end]
[is winix_function_is "upload"]
<p class="withnext">{upload_content}</p>
[else]
[if-one mount_type_is_cms]<p class="withnext">{form_emacs_content_cms}</p>[end]
[if-one mount_type_is_thread]<p class="withnext">{form_emacs_content_thread}</p>[end]
[if-one mount_type_is_ticket]<p class="withnext">{form_emacs_content_ticket}</p>[end]
[end]
<textarea class="multitext" rows="[if-one mount_type_is_cms]30[else]10[end]" cols="60" name="content">[item_content]</textarea> <textarea class="multitext" rows="[is winix_function_is "upload"]7[else][if-one mount_type_is_cms]30[else]10[end][end]" cols="60" name="content">[item_content]</textarea>
<p class="withnext">{form_emacs_content_type}</p> <p class="withnext">{form_emacs_content_type}</p>

View File

@ -1,4 +1,4 @@
[if-one item_is]<h1>{edit}</h1>[else]<h1>{add}</h1>[end] <h1>{upload_header}</h1>
[include "err_abuse.html"] [include "err_abuse.html"]
@ -6,30 +6,29 @@
<fieldset> <fieldset>
<legend>{upload_form_legend}</legend> <legend>{upload_form_legend}</legend>
<input type="file" name="fileupload"> [is winix_function_param_is "multi"]
<p class="withnext">{upload_form_file_multi}</p>
<input size="68" class="edit" type="file" name="fileupload1">
<input size="68" class="edit" type="file" name="fileupload2">
<input size="68" class="edit" type="file" name="fileupload3">
<input size="68" class="edit" type="file" name="fileupload4">
<input size="68" class="edit" type="file" name="fileupload5">
[if-one mount_type_is_cms] [else]
<p class="withnext">{upload_form_file}</p>
<input size="68" class="edit" type="file" name="fileupload">
<p class="withnext">{title}:</p> <p class="withnext">{title}:</p>
<input class="edit" type="text" name="subject" value="[item_subject]"> <input class="edit" type="text" name="subject" value="[item_subject]">
<p class="withnext">{url}:</p> <p class="withnext">{suggested_url}:</p>
<input class="edit" type="text" name="url" value="[item_url]"> <input class="edit" type="text" name="url" value="[item_url]">
[end]
[if-one mount_type_is_thread] [include "fun_emacs_post.html"]
[is mount_thread_is "subject"]
<p class="withnext">{title}:</p>
<input class="edit" type="text" name="subject" value="[item_subject]">
<p class="withnext">{url}:</p>
<input class="edit" type="text" name="url" value="[item_url]">
[end]
[end] [end]
<p class="withnext">{upload_content}:</p>
<textarea class="multitext" rows="[if-one thread_is]10[else]30[end]" cols="60" name="content">[item_content]</textarea>
[if-no user_logged] [if-no user_logged]
<p class="withnext">{nick}:</p> <p class="withnext">{nick}:</p>
@ -39,8 +38,7 @@
<input class="edit" type="text" name="rebus"> <input class="edit" type="text" name="rebus">
[end] [end]
<input class="submit" type="submit" value="{upload_submit}">
<input class="submit" type="submit" value="[if-one item_is]{change}[else]{add}[end]">
</fieldset> </fieldset>
</form> </form>

View File

@ -1,14 +1,16 @@
<h1>{uptime_header}</h1> <h1>{uptime_header}</h1>
<p>[winix_cur_time], {uptime_up} <p>{uptime_current_time}: [winix_cur_time]<br>
{uptime_up}:
[if-one uptime_more_than_one_day] [if-one uptime_more_than_one_day]
[uptime_days] {uptime_days}, [uptime_days] {uptime_days}
[else] [else]
[uptime_hours], [uptime_hours]
[end] [end]
<br>
[winix_users_logged] {uptime_users} {uptime_users}: [winix_users_logged]
</p> </p>

View File

@ -146,8 +146,12 @@ ticket_info_expected = Expected in
ticket_info_progress = Progress ticket_info_progress = Progress
ticket_reply_in_this_thread = Replay in this thread ticket_reply_in_this_thread = Replay in this thread
upload_header = Upload a file
upload_form_file = Browse for a file
upload_form_file_multi = Browse for files
upload_form_legend = Upload form upload_form_legend = Upload form
upload_content = File description upload_content = File description
upload_submit = Upload
who_header = Sessions who_header = Sessions
who_table_index = Ind. who_table_index = Ind.
@ -169,7 +173,8 @@ admin_upload = Upload an image or file (upload)
login_header = Login login_header = Login
uptime_header = Uptime uptime_header = Uptime
uptime_up = up uptime_current_time = current time
uptime_up = system running for
uptime_users = users uptime_users = users
uptime_days = days uptime_days = days

View File

@ -145,8 +145,13 @@ ticket_info_expected = Oczekiwany w
ticket_info_progress = Postêp prac ticket_info_progress = Postêp prac
ticket_reply_in_this_thread = Odpowiedz w tym w±tku ticket_reply_in_this_thread = Odpowiedz w tym w±tku
upload_header = Prześlij plik
upload_form_file = Wybierz plik
upload_form_file_multi = Wybierz pliki
upload_form_legend = Formularz do wys³ania pliku upload_form_legend = Formularz do wys³ania pliku
upload_content = Opis pliku upload_content = Opis pliku
upload_submit = Wyślij
who_header = Lista sesji who_header = Lista sesji
who_table_index = L.p. who_table_index = L.p.
@ -166,8 +171,9 @@ admin_upload = Wy
login_header = Logowanie login_header = Logowanie
uptime_header = Uptime uptime_header = Czas pracy systemu
uptime_up = uruchomiony przez uptime_current_time = aktualna godzina
uptime_up = system uruchomiony przez
uptime_users = zalogowanych u¿ytkowników uptime_users = zalogowanych u¿ytkowników
uptime_days = dni uptime_days = dni

View File

@ -107,7 +107,8 @@ void item_print_content(std::ostringstream & out, const std::string & content, I
else else
if( content_type == Item::ct_bbcode ) if( content_type == Item::ct_bbcode )
{ {
out << content; // !! tutaj bedzie parsowanie bbcodu i tworzenie odpowiadajacego mu html-a //out << content; // !! tutaj bedzie parsowanie bbcodu i tworzenie odpowiadajacego mu html-a
out << "bbcode is not implemented yet";
} }
else else
if( content_type == Item::ct_raw ) if( content_type == Item::ct_raw )
@ -117,6 +118,11 @@ void item_print_content(std::ostringstream & out, const std::string & content, I
} }
void item_content_is_empty(Info & i)
{
i.result = request.item.content.empty();
}
void item_print_content(Info & i) void item_print_content(Info & i)
{ {
item_print_content(i.out, request.item.content, request.item.content_type); item_print_content(i.out, request.item.content, request.item.content_type);

View File

@ -185,6 +185,7 @@ void Templates::CreateFunctions()
functions.Insert("item_id", item_id); functions.Insert("item_id", item_id);
functions.Insert("item_subject", item_subject); functions.Insert("item_subject", item_subject);
functions.Insert("item_subject_noescape", item_subject_noescape); functions.Insert("item_subject_noescape", item_subject_noescape);
functions.Insert("item_content_is_empty", item_content_is_empty);
functions.Insert("item_content", item_content); functions.Insert("item_content", item_content);
functions.Insert("item_content_noescape", item_content_noescape); functions.Insert("item_content_noescape", item_content_noescape);
functions.Insert("item_content_type_is", item_content_type_is); functions.Insert("item_content_type_is", item_content_type_is);

View File

@ -103,6 +103,7 @@ namespace TemplatesFunctions
void item_content(Info & i); void item_content(Info & i);
void item_content_noescape(Info & i); void item_content_noescape(Info & i);
void item_content_type_is(Info & i); void item_content_type_is(Info & i);
void item_content_is_empty(Info & i);
void item_print_content(std::ostringstream & out, const std::string & content, Item::ContentType content_type); void item_print_content(std::ostringstream & out, const std::string & content, Item::ContentType content_type);
void item_print_content(Info & i); void item_print_content(Info & i);
void item_privileges(Info & i); void item_privileges(Info & i);

View File

@ -44,9 +44,9 @@ char buf[50];
time_t hour = min / 60; time_t hour = min / 60;
if( hour == 0 && min == 0 ) if( hour == 0 && min == 0 )
sprintf(buf, "%d:%02d:%02d", (int)hour, (int)min, (int)sec); sprintf(buf, "%d:%02d:%02d", (int)hour, (int)(min%60), (int)(sec%60));
else else
sprintf(buf, "%d:%02d", (int)hour, (int)min); sprintf(buf, "%d:%02d", (int)hour, (int)(min%60));
i.out << buf; i.out << buf;
} }