/* * 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_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::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 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 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