-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathIBMfunctionsS3.R
568 lines (442 loc) · 23.8 KB
/
IBMfunctionsS3.R
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
# Functions for wildlife watching IBM ####
# Author: Francesca Mancini
# Date created: 2018-05-15
# Date modified: 2018-07-20
library(dplyr)
# Wildlife functions ####
# Calculate the effect that tourism had on the population.
# The effect of the time spent with the animals is sigmoid:
# if less than maxx, effect ~ 0, otherwise it increases.
# When time is maxx + 1/5maxx, then effect is medium
# = 0.05 change in probability of encounter.
# Parameter wide determines the range of values over which effect goes from 0 to the maximum (0.1).
# Lower value of this parameter (i.e. when population is bigger)
# corresponds to a wider range over which we observe a change.
# Parameter withanimals is the time tour operators spend with animals.
# maxx is the maximum amout of time operators can spend with animals before seeing an effect.
# Effect on wildlife is also dependent on number of tourists:
# similar sigmoid relationship, if capacity is the same effect is 0,
# if industry doubles in size then effect is maximum 0.1
# and this effect is additive.
tourism_effect <- function(slope_time, slope_capacity, withanimals, init_capacity, new_capacity, maxx) {
0.1 + ((- 0.1) / (1 + exp(slope_time * (withanimals - (maxx + maxx / 5))))) +
0.1 + ((- 0.1) / (1 + exp(-slope_capacity * (init_capacity/new_capacity * 100 - 50))))
}
# testing
# withanimals <- seq(80000, 150000, 1000)
# slope_time <- 0.00025
# slope_capacity <- 0.2
# max_fixed <- 100000
# init_capacity <- 100
# new_capacity_exp <- 200
# new_capacity <- 100
#
# all_effects <- tourism_effect(slope_time, 0.2, withanimals, init_capacity, new_capacity, max_fixed)
# all_effects_exp <- tourism_effect(slope_time, slope_capacity, withanimals, init_capacity, new_capacity_exp, max_fixed)
#
# par(mfrow = c(2,1))
# plot(all_effects ~ withanimals)
# plot(all_effects_exp ~ withanimals)
# Calculate the probability of encounter due to effect on population
# The effect reduced the annual growth rate of the population
# (here: by affecting the probability of encounter)
p_encounter <- function(p_e, effect) {
p_e <- p_e * (1.01 - effect)
p_e <- ifelse(p_e > 1, 1, p_e) # probability cannot be > 1
}
# Calculate new maximum time allowed with animals
time_with_animals <- function(maxx, effect) {
maxx <- maxx * (1.01 - effect) #updates threshold of effect with increase in abundance
}
# testing wildlife functions
#
# withanimals <- seq(0, 300000, 1000)
#
# maxx <- rep(NA, length(withanimals))
# maxx[1] <- 100000
#
# wide <- rep(NA, length(withanimals))
# wide[1] <- 0.00025 / (maxx[1] / 100000)
#
# p_e <- rep(NA, length(withanimals))
# p_e[1] <- 0.8
#
# effect <- rep(NA, length(withanimals))
#
# for(i in 1:seq_along(withanimals)){
# effect[i] <- tourism_effect(wide[i], withanimals[i], maxx[i])
# if(i < length(withanimals)){
# p_e[i + 1] <- p_encounter(p_e[i], effect[i])
# curve <- time_with_animals(maxx[i], effect[i])
# maxx[i + 1] <- curve[1]
# wide[i + 1] <- curve[2]
# }
# }
#
# max_fixed <- 100000
# wide_fixed <- 0.00025
#
# for(i in 1:seq_along(withanimals)){
# effect[i] <- tourism_effect(wide_fixed, withanimals[i], max_fixed)
# }
#
# for(i in 1:seq_along(withanimals)){
# if(i < length(withanimals)){
# p_e[i + 1] <- p_encounter(p_e[i], effect[i])
# }
# }
#
# for(i in 1:seq_along(withanimals)){
# if(i < length(withanimals)){
# curve <- time_with_animals(maxx[i], effect[i])
# maxx[i + 1] <- curve[1]
# wide[i + 1] <- curve[2]
# }
# }
tourists_max <- function(tmax, timeout, tour_ops){
t_pertour <- tmax/365/dim(tour_ops)[1] # calculate n of time allowed with animals per tour
t_mean <- mean(withanimals, na.rm = T) # calculate average time spent with animals per tour in the previous year
if(t_pertour >= t_mean) { # if time allowed is more or equal the mean
(sum(tour_ops$capacity)*365) * dim(tour_ops)[1]} # max tourists is equal to maximum capacity
else{prop_mean <- (t_mean - t_pertour)/timeout # othewise calculate the proportion of a trip that needs to be reduced
n_tours <- (365 - (prop_mean * 365)) * dim(tour_ops)[1] # transform this proportion into number of tours that can be run
n_tours * (sum(tour_ops$capacity))} # multiply the number of tours by tour operators' capacity to obtain maximum tourists
}
# Tourists functions ####
# From a 2010 survey the number of tourists passing through the Moray Firth dolphin watching locations
# are 35500 in Peak season and 27500 in off-peak season
# Jan Feb Mar Apr May June July Aug Sept Oct Nov Dec
# 315 315 1260 5030 7680 9640 11620 14240 7145 4830 610 315
# Percentage of these that go on a boat tour
# 16% 16% 16% 16% 16% 19% 19% 19% 16% 16% 16% 16%
# 50 50 200 800 1230 1830 2210 2700 1140 773 97 50
# Booking tours
booking <- function(tourists, tour_ops){ # the booking function has 2 arguments, a dataset for tourists and one for tour operators
for(i in seq_len(nrow(tourists))) { # loop trhough each tourist
tourist <- tourists[i, ]
inbudget <- subset(tour_ops, tour_ops$price <= tourist$price_max) # extract the subset of tours that are within the budget of the tourist
preferred <- inbudget[which.max(inbudget$rating), "id"] # extract the preferred tour (higher rating)
if(length(preferred) != 0 && # if there is a preferred tour and
tour_ops[which(tour_ops$id == preferred), "rating"] > tourist["rating_min"] && # if the rating of this tour is higher than the the tourist's minimum rating and
tour_ops[which(tour_ops$id == preferred), "capacity"] > tour_ops[which(tour_ops$id == preferred), "bookings"]) { # if the tour operator is not fully booked
tour_ops[which(tour_ops$id == preferred), "bookings"] <- tour_ops[which(tour_ops$id == preferred), "bookings"] + 1 # add a booking to the preferred tour operator
tourists[i, "going"] <- tour_ops[which(tour_ops$id == preferred), "id"] # the tourist is going on the tour
tourists[i, "sample_p"] <- 0.01
} else {inbudget <- subset(inbudget, inbudget$id != preferred) # otherwise delete the preferred tour from the inbudget vector and
while(nrow(inbudget) != 0 && # for as long as there are tours in the inbudget vector and
length(inbudget$rating[inbudget$rating > tourist$rating_min]) > 0) { # there are tours with higher rating than the tourist's minimum
preferred <- as.numeric(inbudget[which.max(inbudget$rating), "id"]) # select the second preferred tour and
if (length(preferred) != 0 && # if all conditions are met
tour_ops[which(tour_ops$id == preferred), "rating"] > tourist["rating_min"] &&
tour_ops[which(tour_ops$id == preferred), "capacity"] > tour_ops[which(tour_ops$id == preferred), "bookings"]) {
tour_ops[which(tour_ops$id == preferred), "bookings"] <- tour_ops[which(tour_ops$id == preferred), "bookings"] + 1 # add a booking to the tour and
tourists[i, "going"] <- tour_ops[which(tour_ops$id == preferred), "id"] # the tourist is going on the tour
tourists[i, "sample_p"] <- 0.01
break
} else {
inbudget <- subset(inbudget, inbudget$id != preferred) # otherwise try the third preferred tour and then the fourth etc...
}
}
if(is.na(tourists[which(tourists$id == tourist$id), "going"]) == TRUE) { # if the tourist has not found a tour
tourists[i, "waiting"] <- tourists[i, "waiting"] + 1 # add 1 day to the tourists's waiting time
tourists[i, "sample_p"] <- 1
}
}
}
invisible(list(tourists,tour_ops)) # return the two dataframes in a list
}
# Testing
# create dataframes to hold preferences for tourists and caracteristics for tour operators
# tour_ops <- data.frame(id = seq(1, 10, 1), price = rnorm(10, 15, 5), rating = rnorm(10, 3, 1.5),
# capacity = as.integer(runif(10, 10, 30)), bookings = rep(0, 10),
# investment_infra = runif(10, 0, 50), investment_ot = runif(10, 0, 50),
# time_with = rnorm(10, 20, 10), profit = rep(0, 10), profit_year = rep(0, 10))
#
# tourists <- data.frame(id = seq(1, 1000, 1), price_max = rnorm(1000, 15, 5),
# rating_min = rnorm(1000, 3, 0.5), going = rep(NA, 1000),
# waiting = rep(0, 1000), satisfaction = rep(NA, 1000),
# satis_animals = rep(NA, 1000), satis_price = rep(NA, 1000),
# satis_infr = rep(NA, 1000), satis_invest = rep(NA, 1000),
# satis_wait = rep(NA, 1000), stringsAsFactors=FALSE)
#
# run the function
# bookings <- booking(tourists, tour_ops)
# extract dataframes from list
# tourists <- bookings[[1]]
# tour_ops <- bookings[[2]]
# Satisfaction
# tourist satisfaction is affected by different caracteristics:
# time spent with the animals, price/quality ratio, waiting time for booking,
# investment in infrastructure and other services
# satisfaction is calculated as a sigmoid curve
# the proportion of tour time spent with animals determines tourist satisfaction
# with an inflectioon point of 0.3 and a variable slope.
satisfaction_animals <- function(withanimals, timeout, slope) {
1 + (0.01 - 1) / (1 + exp(slope * ((withanimals / timeout) - 0.3)))}
# testing
# withanimals <- seq(10, 90, 1)
# timeout <- 90
#
# satisfaction <- satisfaction_animals(withanimals, timeout, 15)
#
# plot(satisfaction~withanimals)
# satisfaction regarding price/quality ratio is also expressed as
# a sigmoid relationship.
satisfaction_price <- function(price, max_price, rating, max_rating, slope, infl) {
1 + (0.5 - 1) / (1 + exp(-slope * (((price/max_price) / (rating/max_rating)) - infl)))
}
# testing
# rating <- rep(5, 41)
# price <- seq(10, 30, 0.5)
#
# satisfaction <- satisfaction_price(price, max(price), rating, max(rating), slope = 15, infl = 0.7)
# plot(satisfaction~c(((price/max(price)) / (rating/max(rating)))))
# different shape of the function for price/rating satisfaction
# this time it is a linear relationship
satisfaction_price_lin <- function(price, max_price, rating, max_rating, slope){
slope * ((price/max_price) / (rating/max_rating))
}
# testing
# price <- 15
# rating <- seq(0, 5, 0.1)
# satisfaction <- satisfaction_price_lin(price, rating, slope = 3)
# plot(satisfaction~rating)
# the proportion of the year a tourist had to wait before being able
# to book a tour determines their satisfaction in a similar way
satisfaction_waiting <- function(waiting, slope, infl) {
1 + (0.01 - 1) / (1 + exp(slope * ((waiting / 365) - infl)))
}
# testing
# waiting <- seq(1, 365, 1)
# satisfaction <- satisfaction_waiting(waiting, -60,0.1)
# plot(satisfaction~waiting)
satisfaction_waiting_lin <- function(waiting, slope){
1 + slope * (waiting/365)
}
# waiting <- seq(1, 365, 1)
# satisfaction <- satisfaction_waiting_lin(waiting, -1)
# plot(satisfaction~waiting) # not sure a linear relationship makes sense
# the proportion of tour operator's profits that is reinvested into
# infrastructure (here) or other services (below) influences tourist satisfaction
satisfaction_infr_investment <- function(infr_investment, max_infr_investment, slope, infl){
1 + (0.01 - 1) / (1 + exp(slope * ((infr_investment / max_infr_investment) - infl)))
}
# testing
# investment <- seq(0, 10000, 100)
# satisfaction <- satisfaction_infr_investment(investment, max(investment), 10, 0.3)
# prop_investment <- investment/max(investment)
# plot(satisfaction ~ prop_investment)
satisfaction_infr_investment_lin <- function(infr_investment, max_infr_investment, slope){
slope * (infr_investment / max_infr_investment)
}
# investment <- seq(0, 10000, 100)
# profit <- 10000
# satisfaction <- satisfaction_infr_investment_lin(investment, 1)
# prop_investment <- investment/profit
# plot(satisfaction ~ prop_investment)
satisfaction_other_investment <- function(investment_other, max_investment_other, slope, infl){
1 + (0.01 - 1) / (1 + exp(slope * ((investment_other / max_investment_other) - infl)))
}
satisfaction_other_investment_lin <- function(investment_other, max_investment_other, slope){
slope * (investment_other / max_investment_other)
}
# overall satisfaction
# tourists <- tourists %>%
# group_by(going) %>%
# mutate(satis_animals = ifelse(is.na(going), as.integer(NA),
# satisfaction_animals(tour_ops[which(tour_ops$id == unique(going)), "time_with"], 90, 15)),
# satis_price = ifelse(is.na(going), as.integer(NA),
# satisfaction_price(tour_ops[which(tour_ops$id == unique(going)), "price"],
# tour_ops[which(tour_ops$id == unique(going)), "rating"], 15, 0.3)),
# satis_wait = ifelse(is.na(going), as.integer(NA), satisfaction_waiting(waiting, 10, 0.4)),
# satis_infr = ifelse(is.na(going), as.integer(NA),
# satisfaction_infr_investment(tour_ops[which(tour_ops$id == unique(going)), "investment_infra"],
# tour_ops[which(tour_ops$id == unique(going)), "profit"], 10, 0.1)),
# satis_invest = ifelse(is.na(going), as.integer(NA),
# satisfaction_other_investment(tour_ops[which(tour_ops$id == unique(going)), "investment_ot"],
# tour_ops[which(tour_ops$id == unique(going)), "profit"], 10, 0.1))) %>%
# ungroup() %>%
# rowwise() %>%
# mutate(satisfaction = ifelse(is.na(going), as.integer(NA),
# sum(satis_animals, satis_price, satis_wait, satis_infr, satis_invest, na.rm = TRUE)))
#
# generate daily time series of tourists
# days <- 365 # number of days in a year
# days_tot <- years* 365 # number of days in total
#
# #effect sizes
# trend <- 0.005 # trends in demand
# eff.season <- 80 # seasonal fluctuation
# const <- 120 # intercept
# # sampling noise
# sampling.sd <-10
#
# # creates a vector holding the day of the year for all of the days
# season <- rep(1:days, length = days_tot)
# # creates a vector holding the days
# day <- 1:days_tot
#
# # calculate the number of tourists as a linear function of the annual trend
# # and a sinusoidal function of the day of year
# # plus some noise
#
# alt.season <- ((season * 2/days)-0.5)*pi
# n_tourists <- round(rnorm(days_tot, mean = const + days_tot *trend + eff.season*sin (alt.season), sd = sampling.sd ))
#
# plot(n_tourists~day, type = "n")
# lines(n_tourists~day, col = "black")
# Tour operators ####
# Tour operators can modify the price of their tours on a yearly basis.
# the change in the price of the ticket is dependent on
# the ratio between demand and supply so that when demand
# is higher than supply price increases
price_change <- function(ticket, demand, supply){
ticket * (demand / supply)}
# encounters
# the time that tour operators can spend in encounters is given by:
# the number of animals encountered (a series of binomial draws with
# probability calculated each year by p_encounter()) multiplied by the
# maximum time per encounter suggested by the code of conduct
library(RGeode)
encounter_time <- function(n_ops, p_e, code = 10, max = 5) {
replicate(n_ops,{sum(rbinom(max,rep(1,10),p_e) * rexptr(max,1/code,c(1,code)))})}
# if encounter time is > the maximum allowed then the tour operators
# need to make the choice whether to defect or cooperate
# stay within the maximum time allowed or use all the time available
# payoffs
# the choice whether to cooperate or defect depends on the tour ops
# behavioural phenotype and on the payoff matrix
# the payoffs are calculated for each behavioural strategy:
# cooperate whent he others cooperate
# cooperate when the others defect
# defect when the others cooperate
# and defet when the others defect
# when defecting while the others cooperate, the tour ops have a
# competitive advantage. the defecting operator will have higher
# rating than the cooperating ones thus attracting more tourists,
# but also attracting those lost by the cooperating competitors
# this advantage is calculated as the difference between the tourists attracted by
# a defecting cooperator and those attracted by a cooperating operator multiplied by the ticket price
competitive_adv <- function(ticket, capacity, allowed, t_encounter, timeout, slope, n_tourops){
ticket * sum(rbinom(capacity, 1, satisfaction_animals(withanimals = t_encounter, timeout, slope))) -
ticket * sum(rbinom(capacity, 1, satisfaction_animals(allowed, timeout, slope)))
}
# the payoff for the strategy of cooperating when the others cooperate is given by:
# the number of tourists attracted by spending the maximum time allowed with animals
# multiplied by the ticket price
CC <- function(ticket, capacity, allowed, timeout, slope, n_tourops){
ticket * sum(rbinom(capacity, 1, satisfaction_animals(allowed, timeout, slope)))}
# the payoff for the strategy of defecting when the others cooperate is given by:
# the number of tourists attracted by spending all the time with animals
# multiplied by the ticket price
# plus the competitive advantage
# minus the possible fine for defecting
DC <- function(ticket, capacity, t_encounter, allowed, timeout, slope, p_detection, fine, n_tourops) {
ticket * sum(rbinom(capacity, 1, satisfaction_animals(withanimals = t_encounter, timeout, slope))) +
competitive_adv(ticket, capacity, allowed, t_encounter, timeout, slope, n_tourops) - (rbinom(1, 1, p_detection) * fine)}
# the payoff for the strategy of cooperating when the others defect is given by:
# the number of tourists attracted by spending the maximum time allowed with animals
# multiplied by the ticket price
# minus the competitive disadvantage
CD <- function(ticket, capacity, allowed, t_encounter, timeout, slope, n_tourops) {
ticket * sum(rbinom(capacity, 1, satisfaction_animals(allowed, timeout, slope))) -
competitive_adv(ticket, capacity, allowed, t_encounter, timeout, slope, n_tourops)}
# the payoff for the strategy of defecting when the others defect is given by:
# the number of tourists attracted by spending all the time with animals
# multiplied by the ticket price
# minus the possible fine for defecting
DD <- function(ticket, capacity, t_encounter, timeout, slope, p_detection, fine) {
ticket * sum(rbinom(capacity, 1, satisfaction_animals(withanimals = t_encounter, timeout, slope))) -
(rbinom(1, 1, p_detection) * fine)}
# calculation of payoffs for management scenario 3 is similar
# but it does not include fines
CC.3 <- function(ticket, capacity, allowed, timeout, slope, n_tourops) {
ticket * sum(rbinom(capacity, 1, satisfaction_animals(allowed, timeout, slope)))}
DC.3 <- function(ticket, capacity, t_encounter, allowed, timeout, slope, n_tourops) {
ticket * sum(rbinom(capacity, 1, satisfaction_animals(withanimals = t_encounter, timeout, slope))) +
competitive_adv(ticket, capacity, allowed, t_encounter, timeout, slope, n_tourops)}
CD.3 <- function(ticket, capacity, allowed, t_encounter, timeout, slope, n_tourops) {
ticket * sum(rbinom(capacity, 1, satisfaction_animals(withanimals = allowed, timeout, slope))) -
competitive_adv(ticket, capacity, allowed, t_encounter, timeout, slope, n_tourops)}
DD.3 <- function(ticket, capacity, t_encounter, timeout, slope) {
ticket * sum(rbinom(capacity, 1, satisfaction_animals(withanimals = t_encounter, timeout, slope)))}
# behavioural strategy
# this function determines the behavioural choice for each tour operators
# according to the payoffs just calculated and their behavioural phenotype
behaviour_choice <- function(tour_ops, payoff_CC, payoff_CD, payoff_DC, payoff_DD, allowed){
tour_ops %>%
mutate(., behaviour = with(., case_when(
time_with < allowed ~ "no choice",
phenotype == "trustful" ~ "cooperate",
phenotype == "optimist" & payoff_DC < payoff_CC ~ "cooperate",
phenotype == "pessimist" & payoff_CD > payoff_DD ~ "cooperate",
phenotype == "envious" & payoff_CD - payoff_DC >= 0 ~ "cooperate",
phenotype == "undefined" ~ sample(c("defect", "cooperate"), 1, prob = c(0.5, 0.5)),
TRUE ~ "defect")))
}
# testing
# phenotypes <- c("trustful", "optimist", "pessimist", "envious")
#
# tour_ops <- data.frame(id = seq(1, 10, 1), price = rnorm(10, 15, 5), rating = rnorm(10, 3, 1.5),
# capacity = as.integer(runif(10, 10, 30)), bookings = rep(0, 10),
# phenotype = sample(phenotypes, 10, replace = T), behaviour = rep(NA, 10))
#
# tour_ops <- behaviour_choice(tour_ops, payoff_CC, payoff_CD, payoff_DC, payoff_DD)
# profit
# calculate daily and annual profit
# tour_ops <- tour_ops %>%
# mutate(profit = ifelse(bookings == 0, 0, (bookings * price) - (0.7 * 90)),
# profit_year = profit_year + profit)
#
# rating
# update rating according to tourists satisfaction
# tour_ops <- tour_ops %>%
# group_by(id) %>%
# mutate(rating = ifelse(bookings ==0, rating, mean(c(colMeans(tourists[which(tourists$going == id), "satisfaction"], na.rm = T), rating), na.rm = T))) %>%
# ungroup()
# Investment decisions
# Investment in services
# The probability of investing in services is a logistic function of
# the proportion of the operator's rating in relation to the maximum rating:
# it is parameterised so that when the rating is up to 0.7 of the maximum
# (on a scale from 1 to 5, a rating of 3.5) the probability of investing is high
# and it gets quickly smaller after reaching 3.75.
# The decision to invest is then a binomial draw based on this probability.
# The amount invested is a random proportion between 1 and 100 % of the available money
# which is the profits for that year.
invest_services <- function(slope = 20, rating, max_rating, profit, infl = 0.75){
p_services <- 1 / (1 + exp(slope * ((rating/max_rating) - infl)))
ifelse(rbinom(length(p_services), 1, p_services) == 1 & profit > 0, runif(1, 0.1, 1) * profit, 0.001)
}
# testing
# rat <- seq(1, 5, 0.1)
# max_r <- 5
# p_services <- invest_services(rating = rat, max_rating = max_r)
# prop_rating <- rat/max_r
# plot(p_services ~ prop_rating)
# Investment in infrastructure
# The probability of investing in infrastructure is a logistic function of
# the proportion of the operator's profit in relation to their own maximum profits:
# it is parameterised so that when the profit is up to 0.8 of the maximum
# (a profit of 80000 over a maximum of 100000) the probability of investing is low
# and it gets quickly higher after reaching 85000.
# The decision to invest is then a binomial draw based on this probability.
# The amount invested is then given by the amount of money available
# and the maximum number of extra tourists that the operator can afford.
# The price per extra tourist is given by the equivalent of 2 weeks of work at full capacity.
invest_infrastructure <- function(slope = 30, profit, max_profit, capacity, ticket,infl = 0.7){
p_infrastructure <- 1 / (1 + exp(-slope * ((profit/(max_profit/2)) - infl)))
ifelse(rbinom(length(p_infrastructure), 1, p_infrastructure) == 1,
as.integer(profit / (capacity * ticket * 14)) * (capacity * ticket * 14), 0)
}
# testing
#
# profits <- seq(0, 10000, 1000)
# max_p <- 100000
# p_infrastructure <- invest_infrastructure(slope = 25, profit = profits, max_profit = max_p, capacity = 20, ticket = 25, infl = 0.7)
# prop_profit <- profits/max_p
# plot(p_infrastructure ~ prop_profit)
# Management #####
# fine defectors
fines <- function(p_detection, penalty){
rbinom(1, 1, p_detection) * penalty}