(root)/src/abe/scheduler.cpp - Rev 1222
Rev 1221 |
Blame |
Compare with Previous |
Last modification |
View Log
| RSS feed
/********************************************************************************************
** iLand - an individual based forest landscape and disturbance model
** http://iland.boku.ac.at
** Copyright (C) 2009- Werner Rammer, Rupert Seidl
**
** This program 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 3 of the License, or
** (at your option) any later version.
**
** 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 <http://www.gnu.org/licenses/>.
********************************************************************************************/
#include "abe_global.h"
#include "scheduler.h"
#include "fmstand.h"
#include "activity.h"
#include "fmunit.h"
#include "fmstp.h"
#include "forestmanagementengine.h"
#include "agent.h"
#include "agenttype.h"
#include "mapgrid.h"
#include "expression.h"
namespace ABE {
/** @class Scheduler
@ingroup abe
The Scheduler class implements the logic of scheduling the when and what of activties.
*/
void Scheduler::addTicket(FMStand *stand, ActivityFlags *flags, double prob_schedule, double prob_execute)
{
if (FMSTP::verbose())
qCDebug(abe)<< "ticked added for stand" << stand->id();
flags->setIsPending(true);
SchedulerItem *item = new SchedulerItem();
item->stand = stand;
item->flags = flags;
item->enterYear = ForestManagementEngine::instance()->currentYear();
item->optimalYear = item->enterYear + flags->activity()->optimalSchedule(stand->U())- stand->absoluteAge();
item->scheduledYear = item->optimalYear;
// estimate growth from now to the optimal time - we assume that growth of the last decade continues
int t = item->optimalYear - item->enterYear; // in t years harvest is optimal
double time_factor = 0.;
if (stand->volume()>0.)
time_factor = t* stand->meanAnnualIncrement()/stand->volume();
item->harvest = stand->scheduledHarvest() * (1. + time_factor);
item->harvestPerHa = item->harvest / stand->area();
item->harvestType = flags->isFinalHarvest()? EndHarvest : Thinning;
item->scheduleScore = prob_schedule;
item->harvestScore = prob_execute;
item->forbiddenTo = 0;
item->calculate(); // set score
mItems.push_back(item);
}
void Scheduler::run()
{
// update the plan if necessary...
if (FMSTP::verbose() && mItems.size()>0)
qCDebug(abe) << "running scheduler for unit" << mUnit->id() << ". # of active items:" << mItems.size();
double harvest_in_queue = 0.;
double total_final_harvested = mExtraHarvest;
double total_thinning_harvested = 0.;
//mExtraHarvest = 0.;
if (FMSTP::verbose() && total_final_harvested>0.)
qCDebug(abe) << "Got extra harvest (e.g. salvages), m3=" << total_final_harvested;
int current_year = ForestManagementEngine::instance()->currentYear();
// update the schedule probabilities....
QList<SchedulerItem*>::iterator it = mItems.begin();
while (it!=mItems.end()) {
SchedulerItem *item = *it;
double p_sched = item->flags->activity()->scheduleProbability(item->stand);
item->scheduleScore = p_sched;
item->calculate();
if (item->stand->trace())
qCDebug(abe) << item->stand->context() << "scheduler scores (harvest schedule total): " << item->harvestScore << item->scheduleScore << item->score;
// drop item if no schedule to happen any more
if (item->score == 0.) {
if (item->stand->trace())
qCDebug(abe) << item->stand->context() << "dropped activity" << item->flags->activity()->name() << "from scheduler.";
item->flags->setIsPending(false);
item->flags->setActive(false);
item->stand->afterExecution(true); // execution canceled
it = mItems.erase(it);
delete item;
} else {
// handle item
harvest_in_queue += item->harvest;
++it;
}
}
if (mUnit->agent()->schedulerOptions().useScheduler)
updateCurrentPlan();
// sort the probabilities, highest probs go first....
//qSort(mItems);
//qSort(mItems.begin(), mItems.end(), )
std::sort(mItems.begin(), mItems.end(), ItemComparator());
if (FMSTP::verbose())
dump();
int no_executed = 0;
double harvest_scheduled = 0.;
// now execute the activities with the highest ranking...
it = mItems.begin();
while (it!=mItems.end()) {
SchedulerItem *item = *it;
// ignore stands that are currently banned (only for final harvests)
if (item->forbiddenTo > current_year && item->flags->isFinalHarvest()) {
++it;
continue;
}
if (item->scheduledYear > current_year)
break; // finished! TODO: check if this works ok ;)
bool remove = false;
bool final_harvest = item->flags->isFinalHarvest();
//
double rel_harvest;
if (final_harvest)
rel_harvest = total_final_harvested / mUnit->area() / mFinalCutTarget;
else
rel_harvest = total_thinning_harvested/mUnit->area() / mThinningTarget;
double min_exec_probability = 0; // calculateMinProbability( rel_harvest );
rel_harvest = (total_final_harvested+total_thinning_harvested)/ mUnit->area() / (mFinalCutTarget+mThinningTarget);
if (rel_harvest > mUnit->agent()->schedulerOptions().maxHarvestLevel)
break;
if (rel_harvest + item->harvest/mUnit->area()/(mFinalCutTarget+mThinningTarget) > mUnit->agent()->schedulerOptions().maxHarvestLevel) {
// including the *current* harvest, the threshold would be exceeded -> draw a random number
if (drandom() <0.5)
break;
}
if (item->score >= min_exec_probability) {
// execute activity:
if (item->stand->trace())
qCDebug(abe) << item->stand->context() << "execute activity" << item->flags->activity()->name() << "score" << item->score << "planned harvest:" << item->harvest;
harvest_scheduled += item->harvest;
bool executed = item->flags->activity()->execute(item->stand);
if (final_harvest)
total_final_harvested += item->stand->totalHarvest();
else
total_thinning_harvested += item->stand->totalHarvest();
item->flags->setIsPending(false);
if (!item->flags->activity()->isRepeatingActivity()) {
item->flags->setActive(false);
item->stand->afterExecution(!executed); // check what comes next for the stand
}
no_executed++;
// flag neighbors of the stand, if a clearcut happened
// this is to avoid large unforested areas
if (executed && final_harvest) {
if (FMSTP::verbose()) qCDebug(abe) << item->stand->context() << "ran final harvest -> flag neighbors";
// simple rule: do not allow harvests for neighboring stands for 7 years
item->forbiddenTo = current_year + 7;
QList<int> neighbors = ForestManagementEngine::instance()->standGrid()->neighborsOf(item->stand->id());
for (QList<SchedulerItem*>::iterator nit = mItems.begin(); nit!=mItems.end(); ++nit)
if (neighbors.contains((*nit)->stand->id()))
(*nit)->forbiddenTo = current_year + 7;
}
remove = true;
}
if (remove) {
// removing item from scheduler
if (item->stand->trace())
qCDebug(abe) << item->stand->context() << "removing activity" << item->flags->activity()->name() << "from scheduler.";
it = mItems.erase(it);
delete item;
} else {
++it;
}
}
if (FMSTP::verbose() && no_executed>0)
qCDebug(abe) << "scheduler finished for" << mUnit->id() << ". # of items executed (n/volume):" << no_executed << "(" << harvest_scheduled << "m3), total:" << mItems.size() << "(" << harvest_in_queue << "m3)";
}
bool Scheduler::forceHarvest(const FMStand *stand, const int max_years)
{
// check if we have the stand in the list:
for (QList<SchedulerItem*>::const_iterator nit = mItems.constBegin(); nit!=mItems.constEnd(); ++nit) {
const SchedulerItem *item = *nit;
if (item->stand == stand)
if (abs(item->optimalYear - GlobalSettings::instance()->currentYear()) < max_years ) {
item->flags->setExecuteImmediate(true);
return true;
}
}
return false;
}
void Scheduler::addExtraHarvest(const FMStand *stand, const double volume, Scheduler::HarvestType type)
{
Q_UNUSED(stand); Q_UNUSED(type); // at least for now
mExtraHarvest += volume;
}
double Scheduler::plannedHarvests(double &rFinal, double &rThinning)
{
rFinal = 0.; rThinning = 0.;
int current_year = ForestManagementEngine::instance()->currentYear();
for (QList<SchedulerItem*>::const_iterator nit = mItems.constBegin(); nit!=mItems.constEnd(); ++nit)
if ((*nit)->optimalYear < current_year + 10) {
if ((*nit)->flags->isFinalHarvest()) {
rFinal += (*nit)->harvest; // scheduled harvest in m3
} else {
rThinning += (*nit)->harvest;
}
}
return rFinal + rThinning;
}
double Scheduler::scoreOf(const int stand_id) const
{
// lookup stand in scheduler list
SchedulerItem *item = 0;
for (QList<SchedulerItem*>::const_iterator nit = mItems.constBegin(); nit!=mItems.constEnd(); ++nit)
if ((*nit)->stand->id() == stand_id) {
item = *nit;
break;
}
if (!item)
return -1;
return item->score;
}
QStringList Scheduler::info(const int stand_id) const
{
SchedulerItem *si = item(stand_id);
if (!si)
return QStringList();
QStringList lines = QStringList();
lines << "-";
lines << QString("type: %1").arg(si->harvestType==Thinning?QStringLiteral("Thinning"):QStringLiteral("End harvest"));
lines << QString("schedule score: %1").arg(si->scheduleScore);
lines << QString("total score: %1").arg(si->score);
lines << QString("scheduled vol/ha: %1").arg(si->harvestPerHa);
lines << QString("postponed to year: %1").arg(si->forbiddenTo);
lines << QString("in scheduler since: %1").arg(si->enterYear);
lines << "/-";
return lines;
}
void Scheduler::updateCurrentPlan()
{
if (mItems.isEmpty())
return;
double scheduled_harvest[MAX_YEARS];
double state[MAX_YEARS];
for (int i=0;i<MAX_YEARS;++i) {
scheduled_harvest[i]=0.;
state[i] = 0.;
}
scheduled_harvest[0] = mExtraHarvest; // salvaging
mSchedule.clear();
int current_year = ForestManagementEngine::instance()->currentYear();
int max_year = 0;
double total_plan = mExtraHarvest;
for (QList<SchedulerItem*>::const_iterator it=mItems.begin(); it!=mItems.end(); ++it) {
SchedulerItem *item = *it;
mSchedule.insert(qMax(item->optimalYear, current_year), item);
total_plan += item->harvest;
int year_index = qMin(qMax(0, item->optimalYear-current_year),MAX_YEARS-1);
scheduled_harvest[ year_index ] += item->harvest;
max_year = qMax(max_year, year_index);
}
double mean_harvest = total_plan / (max_year + 1.);
double level = (mFinalCutTarget + mThinningTarget) * mUnit->area();
level = qMax(level, mean_harvest);
for (int i=0;i<MAX_YEARS;++i)
state[i] = scheduled_harvest[i]>level? 1. : 0.;
int max_iter = mItems.size() * 10;
bool updated = false;
do {
updated = false;
do {
// look for a relocate candidate and relocate
// look for the highest planned harvest
int year=-1; double max_harvest = -1.;
for (int i=0;i<MAX_YEARS;++i) {
if (scheduled_harvest[i]>max_harvest && state[i] == 1.) {
year = i;
max_harvest = scheduled_harvest[i];
}
}
// if no further slot is found, then stop
if (year==-1)
break;
// if the maximum harvest in the next x years is below the current plan,
// then we simply call it a day (and execute everything on its "optimal" point in time)
if (max_harvest < level)
break;
state[year] = -1.; // processed
// pick an element of that year and try to find another year
int pick = irandom(0, mSchedule.count(year + current_year));
QMultiHash<int, SchedulerItem*>::iterator i = mSchedule.find(year + current_year);
while (i!=mSchedule.end() && i.key()==year+current_year) {
if (pick--==0) // select 'pick'ed element
break;
++i;
}
if (i==mSchedule.end() || i.key()!=year+current_year) {
qCDebug(abe) << "updateCurrentPlan(): no item found for year" << year << ", #elements:" << mSchedule.count(year + current_year);
break;
}
SchedulerItem *item = i.value();
// try to change something only if the years' schedule is above the level without the focal item
if (scheduled_harvest[year]-item->harvest > level ) {
//
int calendar_year = year + current_year;
int dist = -1;
do {
double value = item->flags->activity()->scheduleProbability(item->stand, calendar_year + dist);
if (value>0. && year+dist>=0 && year+dist<MAX_YEARS) {
if (state[year+dist] == 0.) {
// simple: finish!
mSchedule.erase(i);
scheduled_harvest[year] -= item->harvest;
scheduled_harvest[year+dist] += item->harvest;
mSchedule.insert(calendar_year+dist, item);
updated = true;
// reset also the processed flag
state[year] = scheduled_harvest[year]>level? 1. : 0.;
state[year+dist] = scheduled_harvest[year+dist]>level? 1. : 0.;
break;
}
}
// series of: -1 +1 -2 +2 -3 +3 ...
if (dist<0)
dist = -dist; // switch sign
else
dist = - (dist+1); // switch sign and add 1
} while (dist<MAX_YEARS);
if (updated)
break;
} // end if
if (--max_iter<0) {
qCDebug(abe) << "scheduler: max iterations reached in updateCurrentPlan()";
break;
}
} while (1==1); // continue until no further candidate exists or a relocate happened
} while (updated); // stop when no new candidate is found
// write back the execution plan....
for (QMultiHash<int, SchedulerItem*>::iterator it=mSchedule.begin(); it!=mSchedule.end(); ++it)
it.value()->scheduledYear = it.key();
if (FMSTP::verbose())
dump();
}
void Scheduler::dump() const
{
if(mItems.isEmpty())
return;
qCDebug(abe)<< "***** Scheduler items **** Unit:" << mUnit->id();
qCDebug(abe)<< "stand.id, scheduled.year, score, opt.year, act.name, planned.harvest";
QList<SchedulerItem*>::const_iterator it = mItems.begin();
while (it!=mItems.end()) {
SchedulerItem *item = *it;
qCDebug(abe) << QString("%1, %2, %3, %4, %5, %6").arg(item->stand->id()).arg(item->scheduledYear).arg(item->score).arg(item->optimalYear)
.arg(item->flags->activity()->name())
.arg(item->harvest);
++it;
}
}
Scheduler::SchedulerItem *Scheduler::item(const int stand_id) const
{
for (QList<SchedulerItem*>::const_iterator nit = mItems.constBegin(); nit!=mItems.constEnd(); ++nit)
if ((*nit)->stand->id() == stand_id) {
return *nit;
}
return 0;
}
bool Scheduler::SchedulerItem::operator<(const Scheduler::SchedulerItem &item)
{
// sort *descending*, i.e. after sorting, the item with the highest score is in front.
// if (this->score == item.score)
// return this->enterYear < item.enterYear; // higher prob. for items that entered earlier TODO: change to due/overdue
if (this->scheduledYear==item.scheduledYear)
return this->score > item.score;
else
return this->scheduledYear < item.scheduledYear;
}
void Scheduler::SchedulerItem::calculate()
{
if (flags->isExecuteImmediate())
score = 1.1; // above 1
else
score = scheduleScore * harvestScore;
if (score<0.)
score = 0.;
}
// **************************************************************************************
QStringList SchedulerOptions::mAllowedProperties = QStringList()
<< "minScheduleHarvest" << "maxScheduleHarvest" << "maxHarvestLevel"
<< "useSustainableHarvest" << "scheduleRebounceDuration" << "deviationDecayRate"
<< "enabled" << "harvestIntensity";
void SchedulerOptions::setup(QJSValue jsvalue)
{
useScheduler = false;
if (!jsvalue.isObject()) {
qCDebug(abeSetup) << "Scheduler options are not an object:" << jsvalue.toString();
return;
}
FMSTP::checkObjectProperties(jsvalue, mAllowedProperties, "setup of scheduler options");
minScheduleHarvest = FMSTP::valueFromJs(jsvalue, "minScheduleHarvest","0").toNumber();
maxScheduleHarvest = FMSTP::valueFromJs(jsvalue, "maxScheduleHarvest","10000").toNumber();
maxHarvestLevel = FMSTP::valueFromJs(jsvalue, "maxHarvestLevel","2").toNumber();
qCDebug(abe) << "maxHarvestLevel" << maxHarvestLevel;
useSustainableHarvest = FMSTP::valueFromJs(jsvalue, "useSustainableHarvest", "1").toNumber();
if (useSustainableHarvest<0. || useSustainableHarvest>1.)
throw IException("Setup of scheduler-options: invalid value for 'useSustainableHarvest' (0..1 allowed).");
harvestIntensity = FMSTP::valueFromJs(jsvalue, "harvestIntensity", "1").toNumber();
scheduleRebounceDuration = FMSTP::valueFromJs(jsvalue, "scheduleRebounceDuration", "5").toNumber();
if (scheduleRebounceDuration==0.)
throw IException("Setup of scheduler-options: '0' is not a valid value for 'scheduleRebounceDuration'!");
// calculate the "tau" of a exponential decay function based on the provided half-time
scheduleRebounceDuration = scheduleRebounceDuration / log(2.);
deviationDecayRate = FMSTP::valueFromJs(jsvalue, "deviationDecayRate","0").toNumber();
if (deviationDecayRate==1.)
throw IException("Setup of scheduler-options: '0' is not a valid value for 'deviationDecayRate'!");
deviationDecayRate = 1. - deviationDecayRate; // if eg value is 0.05 -> multiplier 0.95
useScheduler = FMSTP::boolValueFromJs(jsvalue, "enabled", true);
}
bool Scheduler::ItemComparator::operator()(const Scheduler::SchedulerItem *lx, const Scheduler::SchedulerItem *rx) const
{
if (lx->scheduledYear==rx->scheduledYear)
return lx->score > rx->score;
else
return lx->scheduledYear < rx->scheduledYear;
}
} // namespace