150 lines
4.4 KiB
C++
150 lines
4.4 KiB
C++
/*
|
|
* Copyright (c) 2003-2009, John Wiegley. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are
|
|
* met:
|
|
*
|
|
* - Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
*
|
|
* - Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* - Neither the name of New Artisans LLC nor the names of its
|
|
* contributors may be used to endorse or promote products derived from
|
|
* this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#include "timelog.h"
|
|
|
|
namespace ledger {
|
|
|
|
namespace {
|
|
void clock_out_from_timelog(std::list<time_entry_t>& time_entries,
|
|
const datetime_t& when,
|
|
account_t * account,
|
|
const char * desc,
|
|
journal_t& journal)
|
|
{
|
|
time_entry_t event;
|
|
|
|
if (time_entries.size() == 1) {
|
|
event = time_entries.back();
|
|
time_entries.clear();
|
|
}
|
|
else if (time_entries.empty()) {
|
|
throw parse_error("Timelog check-out event without a check-in");
|
|
}
|
|
else if (! account) {
|
|
throw parse_error
|
|
("When multiple check-ins are active, checking out requires an account");
|
|
}
|
|
else {
|
|
bool found = false;
|
|
|
|
for (std::list<time_entry_t>::iterator i = time_entries.begin();
|
|
i != time_entries.end();
|
|
i++)
|
|
if (account == (*i).account) {
|
|
event = *i;
|
|
found = true;
|
|
time_entries.erase(i);
|
|
break;
|
|
}
|
|
|
|
if (! found)
|
|
throw parse_error
|
|
("Timelog check-out event does not match any current check-ins");
|
|
}
|
|
|
|
if (desc && event.desc.empty()) {
|
|
event.desc = desc;
|
|
desc = NULL;
|
|
}
|
|
|
|
std::auto_ptr<entry_t> curr(new entry_t);
|
|
curr->_date = when.date();
|
|
curr->code = desc ? desc : "";
|
|
curr->payee = event.desc;
|
|
|
|
if (when < event.checkin)
|
|
throw parse_error
|
|
("Timelog check-out date less than corresponding check-in");
|
|
|
|
char buf[32];
|
|
std::sprintf(buf, "%lds", long((when - event.checkin).total_seconds()));
|
|
amount_t amt;
|
|
amt.parse(buf);
|
|
assert(amt.valid());
|
|
|
|
xact_t * xact = new xact_t(event.account, amt, XACT_VIRTUAL);
|
|
xact->set_state(item_t::CLEARED);
|
|
curr->add_xact(xact);
|
|
|
|
if (! journal.add_entry(curr.get()))
|
|
throw parse_error("Failed to record 'out' timelog entry");
|
|
else
|
|
curr.release();
|
|
}
|
|
} // unnamed namespace
|
|
|
|
time_log_t::~time_log_t()
|
|
{
|
|
TRACE_DTOR(time_log_t);
|
|
|
|
if (! time_entries.empty()) {
|
|
std::list<account_t *> accounts;
|
|
|
|
foreach (time_entry_t& time_entry, time_entries)
|
|
accounts.push_back(time_entry.account);
|
|
|
|
foreach (account_t * account, accounts)
|
|
clock_out_from_timelog(time_entries, CURRENT_TIME(), account, NULL,
|
|
journal);
|
|
|
|
assert(time_entries.empty());
|
|
}
|
|
}
|
|
|
|
void time_log_t::clock_in(const datetime_t& checkin,
|
|
account_t * account,
|
|
const string& desc)
|
|
{
|
|
time_entry_t event(checkin, account, desc);
|
|
|
|
if (! time_entries.empty()) {
|
|
foreach (time_entry_t& time_entry, time_entries) {
|
|
if (event.account == time_entry.account)
|
|
throw parse_error("Cannot double check-in to the same account");
|
|
}
|
|
}
|
|
|
|
time_entries.push_back(event);
|
|
}
|
|
|
|
void time_log_t::clock_out(const datetime_t& checkin,
|
|
account_t * account,
|
|
const string& desc)
|
|
{
|
|
if (time_entries.empty())
|
|
throw std::logic_error("Timelog check-out event without a check-in");
|
|
|
|
clock_out_from_timelog(time_entries, checkin, account, desc.c_str(),
|
|
journal);
|
|
}
|
|
|
|
} // namespace ledger
|