2 * Empire - A multi-player, client/server Internet based war game.
3 * Copyright (C) 1986-2006, Dave Pare, Jeff Bailey, Thomas Ruschak,
4 * Ken Stevens, Steve McClure
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
22 * See files README, COPYING and CREDITS in the root of the source
23 * tree for related information and legal notices. It is expected
24 * that future projects/authors will amend these files as needed.
28 * nreport.c: File a news report. Downgrade relations if things get hostile.
30 * Known contributors to this file:
37 #include "prototypes.h"
49 static struct newscache cache[MAXNOC][SLOTS];
52 static struct newscache *
53 ncache(int actor, int event, int victim, int times);
56 nreport(natid actor, int event, natid victim, int times)
60 struct newscache *ncp;
62 if (CANT_HAPPEN((unsigned)event > N_MAX_VERB
63 || rpt[event].r_newstory[0] == rpt[0].r_newstory[0]))
66 ncp = ncache(actor, event, victim, times);
67 putnews(ncp->id, &ncp->news);
70 * this is probably pretty expensive, but hopefully we
71 * don't fire zillions of these things off every second.
73 if (victim == 0 || (nice = rpt[event].r_good_will) >= 0)
76 * Pretty schlocky to put it here, but
77 * I guess it can't go anywhere else.
81 if (!chance((double)-nice * times / 20.0))
83 if ((natp = getnatp(victim)) == 0)
85 if (getrel(natp, actor) < HOSTILE)
88 setrel(victim, actor, HOSTILE);
92 * Delete news articles that have expired.
101 /* skip over expired news */
102 expiry_time = time(NULL) - days(news_keep_days);
103 for (i = 0; getnews(i, &news); i++) {
104 if (news.nws_when == 0 || news.nws_when >= expiry_time)
107 /* news id 0..I-1 have expired */
108 CANT_HAPPEN(i > news_tail);
109 /* no items to delete if I is equal zero */
113 /* move unexpired news I.. to 0.., overwriting expired news */
114 for (j = 0; getnews(i + j, &news); j++) {
115 if (news.nws_when == 0)
119 CANT_HAPPEN(i + j != news_tail);
122 /* mark slots no longer in use */
123 memset(&news, 0, sizeof(news));
124 for (k = 0; k < i; k++)
125 putnews(j + k, &news);
127 /* clear cache because moving news invalidated it */
128 memset(&cache, 0, sizeof(cache));
132 * Initialize news reporting.
133 * Must run between open of file EF_NEWS and first nreport().
141 for (newest_item = 0; getnews(newest_item, &news); newest_item++) {
142 if (news.nws_when == 0)
145 news_tail = newest_item;
149 * Look to see if the same message has been generated
150 * in the last 5 minutes, if so just increment the times
151 * field instead of creating a new message.
153 static struct newscache *
154 ncache(int actor, int event, int victim, int times)
156 register struct newscache *np;
160 time_t now = time(NULL);
163 oldtime = 0x7fffffff;
164 for (i = 0; i < SLOTS; i++) {
165 np = &cache[actor][i];
166 if (np->news.nws_when < oldtime) {
168 oldtime = np->news.nws_when;
172 if ((now - np->news.nws_when) > minutes(5))
174 if (np->news.nws_vrb == event && np->news.nws_vno == victim &&
175 np->news.nws_ntm + times <= 127) {
176 np->news.nws_ntm += times;
180 if (CANT_HAPPEN(oldslot < 0))
182 if (CANT_HAPPEN(!strstr(rpt[event].r_newstory[0], "%s") && victim != 0))
184 np = &cache[actor][oldslot];
185 np->news.nws_ano = actor;
186 np->news.nws_vno = victim;
187 np->news.nws_when = now;
188 np->news.nws_vrb = event;
189 np->news.nws_ntm = times;
190 ef_ensure_space(EF_NEWS, news_tail, 100);
191 np->id = news_tail++;