forked from se-sic/coronet
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathutil-misc.R
915 lines (806 loc) · 39.4 KB
/
util-misc.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
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
## This file is part of coronet, which 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, version 2.
##
## 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, write to the Free Software Foundation, Inc.,
## 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
##
## Copyright 2016-2018 by Claus Hunsen <[email protected]>
## Copyright 2017 by Raphael Nömmer <[email protected]>
## Copyright 2017 by Christian Hechtl <[email protected]>
## Copyright 2017 by Felix Prasse <[email protected]>
## Copyright 2017-2018 by Thomas Bock <[email protected]>
## Copyright 2020-2021 by Thomas Bock <[email protected]>
## Copyright 2018-2019 by Jakob Kronawitter <[email protected]>
## Copyright 2021 by Niklas Schneider <[email protected]>
## All Rights Reserved.
## / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / /
## Libraries ---------------------------------------------------------------
requireNamespace("plyr") # for rbind.fill and dlply
requireNamespace("parallel") # for parallel computation
requireNamespace("igraph") # networks
requireNamespace("logging") # for logging
requireNamespace("lubridate") # for date conversion
## / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / /
## Network data ------------------------------------------------------------
#' Construct an edge list for the given network, with timestamps as an extra attribute column.
#'
#' The 'date' attribute has to be added during network construction as default edge attribute
#' in order to avoid problems accessing it.
#'
#' @param net the given network
#'
#' @return the new edgelist
get.edgelist.with.timestamps = function(net) {
## get edge list as data.frame
edges = as.data.frame(igraph::get.edgelist(net))
colnames(edges) = c("from", "to")
## get timestamps
dates = igraph::get.edge.attribute(net, "date")
## bind everything together
edges = cbind(edges, date = dates)
return(edges)
}
## / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / /
## Parameter verification --------------------------------------------------
#' Verify the actual arguments given for parameters, checking for two things:
#' - Is the argument missing?
#' - Does the argument inherit from the wrong class?
#'
#' If the checks fail for any reason, the program is stopped with an error message.
#'
#' @param argument the actual argument given to a function
#' @param allowed.classes the allowed classes of the argument
#' @param reference the reference string where this function is called from
#' (e.g., a function or class name)
#'
#' @return the argument if all checks are passed
verify.argument.for.parameter = function(argument, allowed.classes, reference) {
## get variable name of 'argument'
argument.variable = as.character(match.call())[2]
## check if the argument is missing
if (missing(argument)) {
error.message = sprintf("The parameter '%s' must not be missing when calling the function '%s'.",
argument.variable, reference)
logging::logerror(error.message)
stop(error.message)
}
## check if the argument inherits from the correct classes
if (!inherits(argument, allowed.classes)) {
error.message = sprintf(paste("The specified parameter '%s' of class [%s] inherits from the wrong class.",
"When calling '%s' the parameter must inherit from one of the following classes: %s"),
argument.variable, paste(class(argument), collapse = ", "), reference,
paste(allowed.classes, collapse = ", "))
logging::logerror(error.message)
stop(error.message)
}
return(argument)
}
#' Check whether an argument passed to a function partially matches the specified candidate values of this function.
#' If no argument is passed to that function or the passed argument consists of too many elements, a specified
#' \code{default} value for the argument is returned.
#' If \code{several.ok} is \code{TRUE}, then multiple elements are allowed and the \code{default} value is ignored.
#'
#' Notice: If *no* \code{default} value is specified, this function simply calls the \code{match.arg} function of
#' R-base. See further details in the documentation of \code{match.arg}: \code{?match.arg}
#'
#' @param arg the argument to check (has to be a character vector or \code{NULL})
#' @param choices the candidate values for \code{arg} (has to be a character vector)
#' (if this parameter is not passed, the candidate values are retrieved from the parent function)
#' @param default a valid default value for \code{arg} (used if \code{arg} is not passed to this function or
#' \code{arg} contains more than one element, unless \code{several.ok} is \code{TRUE})
#' or \code{NULL} (in case of \code{NULL}, the first element of \code{choices} is chosen)
#' [default: NULL]
#' @param several.ok logical indicating whether \code{arg} is allowed to have more than one element [default: FALSE]
#'
#' @return the unabbreviated match(es) out of \code{choices} or the \code{default} value
match.arg.or.default = function(arg, choices, default = NULL, several.ok = FALSE) {
## if no choices are given, extract them from the formal signature of the parent function
## (the following if-block is taken from https://svn.r-project.org/R/tags/R-3-4-4/src/library/base/R/match.R,
## which is also licensed under GPLv2 (or later))
if (missing(choices)) {
formal.args <- formals(sys.function(sys.parent()))
choices <- eval(formal.args[[as.character(substitute(arg))]])
}
## check whether default value is a valid choice
if (!is.null(default) && (length(default) != 1 || !default %in% choices)) {
stop(paste("'default' is not a valid choice. Valid choices: ", paste(dQuote(choices), collapse = ", ")))
}
## check whether to return the default value
if (length(arg) != 1 && !several.ok && !is.null(default)) {
return(default)
} else {
return(match.arg(arg, choices, several.ok))
}
}
## / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / /
## Empty dataframe creation-------------------------------------------------
#' Create an empty dataframe with the specified columns. Unless all columns should have the default datatype
#' \code{logical}, the second parameter \code{data.types} should specify the datatypes.
#'
#' @param columns a character vector containing all the column names
#' @param data.types a character vector of the same length as \code{columns}; the datatypes can be \code{integer},
#' \code{numeric}, \code{POSIXct}, \code{character}, \code{factor}, \code{logical}, and \code{list()}
#'
#' @return the newly created empty dataframe
create.empty.data.frame = function(columns, data.types = NULL) {
## if the vector data.types is specified, its length must match the length of the corresponding column names
if (!is.null(data.types) && length(data.types) != length(columns)) {
stop("If specified, the length of the two given vectors columns and data.types must be the same.")
}
## create the empty data frame (with zero rows), but the given number of columns
data.frame = data.frame(matrix(nrow = 0, ncol = length(columns)))
colnames(data.frame) = columns
## assign the datatypes to the data frame columns by indivdually swapping the columns with new columns that possess
## the correct data type
for (i in seq_along(data.types)) {
## get the column
column = data.frame[[i]]
## replace column with column of correct type
switch(tolower(data.types[i]),
"posixct" = {
column = lubridate::with_tz(as.POSIXct(column), tzone = TIMEZONE)
},
"integer" = {
column = as.integer(column)
},
"numeric" = {
column = as.numeric(column)
},
"logical" = {
column = as.logical(column)
},
"character" = {
column = as.character(column)
},
"factor" = {
column = as.factor(column)
},
"list()" = {
column = I(as.list(column))
},
{
stop(paste("Unknown datatype specified:", data.types[[i]]))
}
)
## set the column back into the dataframe
data.frame[[i]] = column
}
return(data.frame)
}
## / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / /
## Vector misc--------------------------------------------------------------
#' Get the second last element of a vector, if it has, at least, two elements.
#'
#' @param v the vector of which to retrieve the last element but one
#'
#' @return the second last element of \code{v}. If the vector has less than two elements, return \code{NA}.
get.second.last.element = function(v) {
if (length(v) >= 2) {
return(tail(v, n = 2)[[1]])
} else {
return(NA)
}
}
## / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / /
## Stacktrace --------------------------------------------------------------
#' Get the stacktrace.
#'
#' @param calls the calls of the stacktrace
#'
#' @return the built stacktrace
get.stacktrace = function(calls) {
lapply(calls, deparse)
}
## / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / /
## Intermediate data -------------------------------------------------------
#' This function is for repetitive runs of the same script: It saves intermediate data to disk and
#' loads it from a previous runs if possible. This way, computation time can be saved.
#'
#' In detail, there are two possibilities:
#' - The file with the path \code{dump.path} *does not exist*.
#'
#' Save the return value of \code{if.not.found} under the variable name \code{variable}
#' in the environment from which this function is called and, additionally, save the variable's
#' value in the file \code{dump.path}.
#'
#' - The file with the path \code{dump.path} *exists already*.
#'
#' Load the saved file and, thus, the inherently saved object and store the variable in the
#' calling environment.
#'
#' In both cases, the saved/loaded value is assigned to a variable named \code{variable}
#' in the parent frame, i.e., in the calling environment. This means, the return value of this function
#' does not need to be stored manually in a variable. However, for compatibility reasons, the
#' value is returned invisibly so that assignment is possible (although, this does not disable the
#' automatic storage in the parent environment!).
#'
#' Important: With the parameter \code{skip} set to \code{TRUE}, the check for data existance can be
#' skipped (i.e., force a re-save to disk).
#'
#' @param variable a character naming the data variable to be saved to disk or loaded
#' @param dump.path the path where the data is to be saved to or loaded from
#' @param if.not.found if the data does not exist on disk, run this function whose return value is to be
#' saved to disk into the file \code{dump.path}
#' @param skip re-save although data exists on the disk? [default: FALSE]
#'
#' @return the data named \code{variable}, either computed by \code{if.not.found} or loaded
#' from \code{dump.path}
save.and.load = function(variable, dump.path, if.not.found, skip = FALSE) {
if (!skip && file.exists(dump.path)) {
logging::logdebug("Load %s from previously dumped object: %s.", variable, dump.path)
## load the dumped object into the environment calling this very function
load(file = dump.path, envir = parent.frame())
} else {
res = if.not.found()
assign(variable, res, envir = parent.frame()) # rewrite to variable name
rm(res) # clear memory
logging::logdebug("Dumping object %s to %s.", variable, dump.path)
save(list = variable, file = dump.path, envir = parent.frame()) # save automatically
}
return(invisible(get0(variable, envir = parent.frame())))
}
## / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / /
## Date handling -----------------------------------------------------------
#' Parse a date with optional time
#'
#' Notice: Time-zone suffixes are ignored. The \code{input} is expected to be in UTC,
#' even if the \code{input} contains another time-zone suffix.
#'
#' @param input The date string, a vector of date strings, or a list of date strings
#'
#' @return The parsed date(s) as POSIXct object, without changing the underlying data structure
get.date.from.string = function(input) {
## re-usable function to parse date strings with lubridate
convert.text.to.date = function(text) {
date = lubridate::ymd_hms(text, truncated = 3)
return(date)
}
## Handle list manually as lubridate would
## emit warnings on lists containing NA
if (is.list(input)) {
result = lapply(input, convert.text.to.date)
} else {
result = convert.text.to.date(input)
}
return(result)
}
#' Convert UNIX timestamp to POSIXct
#'
#' @param timestmap The timestamp
#'
#' @return The parsed date as POSIXct object
get.date.from.unix.timestamp = function(timestamp) {
date = lubridate::as_datetime(timestamp)
return(date)
}
#' Formats a given date as string using the format "%Y-%m-%d %H:%M:%S"
#'
#' @param input The POSIXct object, a vector of such, or a list of such
#'
#' @return The formatted date(s), without changing the underlying data structure
get.date.string = function(input) {
## re-usable function to parse date strings with lubridate
convert.date.to.text = function(date) {
## if we do not have a POSIXct object here, do not convert
if (!lubridate::is.POSIXct(date)) {
return(date)
}
text = strftime(date, format = "%Y-%m-%d %H:%M:%S")
return(text)
}
## Handle list manually to not change the underlying
## data structure
if (is.list(input)) {
result = lapply(input, convert.date.to.text)
} else {
result = convert.date.to.text(input)
}
return(result)
}
#' Construct a date sequence on the given start time and end time, by either applying the given
#' time period between the sequentially generated dates or by creating the given number of
#' sequential windows.
#'
#' Note: You may want to use the function \code{ProjectData$get.data.timestamps} with this
#' function here.
#'
#' @param start The start time as string or POSIXct object
#' @param end The end time as string or POSIXct object
#' @param by The time period describing the length of time between dates, a character
#' string, e.g., "3 mins" or "15 days"
#' @param length.out The desired length of the sequence (an integer). If set, the
#' 'time.period' parameter is ignored. [default: NULL]
#'
#' @return the sequential dates as a vector
generate.date.sequence = function(start.date, end.date, by, length.out = NULL) {
## convert dates
start.date = get.date.from.string(start.date)
end.date = get.date.from.string(end.date)
## convert time.period to duration
if (is.null(length.out)) {
time.period = lubridate::duration(by)
} else {
time.complete = lubridate::as.duration(lubridate::interval(start.date, end.date))
time.period = time.complete / length.out
## to avoid rounding differences, we round the time period up
## (otherwise, we may end up with another unwanted date in the sequence)
time.period = ceiling(time.period)
}
## convenience function for next step
get.next.step = function(date) {
return(date + time.period)
}
## generate dates before end date:
## 1) initialize date sequence with first date
dates = c(start.date)
## 2) current date
current.date = start.date
## 3) iterate while smaller than end date
while (get.next.step(current.date) < end.date) {
## get next step
next.step = get.next.step(current.date)
## add next-step date to sequence
dates = c(dates, next.step)
current.date = next.step
}
## 4) add end date to sequence
dates = c(dates, end.date)
## 5) explicitly re-add time-zone attribute 'tzone' (as 'c.POSIXct' loses it)
dates = lubridate::with_tz(dates, tzone = TIMEZONE)
return(dates)
}
## / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / /
## Range construction and handling -----------------------------------------
#' Construct ranges from the given list/vector of revisions. If \code{raw} is
#' \code{FALSE} (the default), the ranges are constructed in the format "rev[n]-rev[n+1]".
#' Otherwise, pairs of range bounds are returned in list.
#'
#' @param revs the revisions
#' @param sliding.window whether sliding window splitting is enabled or not
#' [default: FALSE]
#' @param raw whether to return pairs of POSIXct objects or strings rather than
#' formatted strings [default: FALSE]
#'
#' @return the constructed ranges, either formatted or raw; the raw ranges are a named list,
#' for which the formatted ranges are the names
construct.ranges = function(revs, sliding.window = FALSE, raw = FALSE) {
## make sure that, at least, two revisions are provided
if (length(revs) < 2) {
error.message = "Cannot construct ranges from less than 2 revisions."
logging::logerror(error.message)
stop(error.message)
}
## setting offset to construct ranges, i.e.,
## combine each $offset revisions
offset = 1
## with sliding window, we combine each second revision
if (sliding.window)
offset = 2
## extract sequences of revisions
seq1 = revs[ 1:(length(revs) - offset) ]
if ((offset + 1) <= length(revs)) {
seq2 = revs[ (offset + 1):length(revs) ]
} else {
seq2 = revs[ length(revs) ]
}
## construct ranges
ranges = mapply(seq1, seq2, SIMPLIFY = FALSE, FUN = function(start, end) {
start.string = get.date.string(start)
end.string = get.date.string(end)
range = paste(start.string, end.string, sep = "-")
return(range)
})
ranges = unlist(ranges, use.names = FALSE)
## if raw is enabled, we need to compose seq1 and
## seq2 to appropriate tuples
if (raw) {
## compose tuples of range start and range end
ranges.raw = mapply(seq1, seq2, FUN = c, SIMPLIFY = FALSE)
## add formatted ranges as names
names(ranges.raw) = ranges
## set as return value
ranges = ranges.raw
}
return(ranges)
}
#' Construct consecutive ranges based on the given start time, end time, and time period for
#' each range. The ranges do not overlap, i.e., the end of any range is the start of the next one.
#'
#' With this function, it is possible to construct ranges like this:
#' > ++...
#' > ..++.
#' > ....+
#'
#' When the time difference between \code{start} and \code{end} is smaller than
#' \code{time.period}, just one range from \code{start} to \code{end} is constructed.
#'
#' Important: As the start of each range is supposed to be inclusive and the end of each range
#' exclusive, 1 second is added to \code{end}. This way, the date \code{end} will be *included*
#' in the last range. To avoid that behavior, set parameter \code{include.end.date} to \code{FALSE}.
#'
#' Note: You may want to use the function \code{ProjectData$get.data.timestamps} with this
#' function here.
#'
#' @param start The start time as string or POSIXct object
#' @param end The end time as string or POSIXct object; the last time to be *included* in the
#' last range (see above)
#' @param time.period The time period describing the length of the ranges, a character
#' string, e.g., "3 mins" or "15 days"
#' @param imperfect.range.ratio The ratio of the \code{time.period} the last range has to last at least.
#' That is, if the last range would be shorter than ratio * \code{time.period},
#' the last range will be combined with the second last range.
#' A ratio of 0.0 means that the last range could be as small as possible.
#' A ratio of 1.0 means that the last range has to last at least \code{time.period}.
#' [default: 0.0]
#' @param include.end.date whether to include the end date or not [default: TRUE]
#' @param raw whether to return pairs of POSIXct objects or strings rather than
#' formatted strings [default: FALSE]
#'
#' @return the constructed ranges, either formatted or raw; the raw ranges are a named list,
#' for which the formatted ranges are the names
construct.consecutive.ranges = function(start, end, time.period, imperfect.range.ratio = 0.0,
include.end.date = TRUE, raw = FALSE) {
## just construct overlapping ranges without any overlap ;)
ranges = construct.overlapping.ranges(start, end, time.period, overlap = 0,
imperfect.range.ratio = imperfect.range.ratio,
include.end.date = include.end.date, raw)
return(ranges)
}
#' Construct ranges based on the given start time, end time, time period, and overlap.
#'
#' With this function, it is possible to construct ranges like this:
#' > ++++
#' > .++++
#' > ..++++
#'
#' With \code{overlap} being the half of \code{time.period}, we basically obtain half-
#' overlapping ranges as in the function \code{construct.ranges} when \code{sliding.window}
#' is set to \code{TRUE}.
#'
#' When the time difference between \code{start} and \code{end} is smaller than
#' \code{time.period}, just one range from \code{start} to \code{end} is constructed.
#'
#' Important: As the start of each range is supposed to be inclusive and the end of each range
#' exclusive, 1 second is added to \code{end}. This way, the date \code{end} will be *included*
#' in the last range. To avoid that behavior, set parameter \code{include.end.date} to \code{FALSE}.
#'
#' Note: You may want to use the function \code{ProjectData$get.data.timestamps} with this
#' function here.
#'
#' @param start The start time as string or POSIXct object
#' @param end The end time as string or POSIXct object; the last time to be *included* in the
#' last range (see above)
#' @param time.period The time period describing the length of the ranges, a character
#' string, e.g., "3 mins" or "15 days"
#' @param overlap The time period describing the length of the overlap, a character string
#' (e.g., "3 mins" or "15 days") or a numeric indication the percentage of
#' overlap (e.g., 1/4). Should be more than 0 seconds and must not be larger
#' than the given \code{time.period}.
#' @param imperfect.range.ratio The ratio of the \code{time.period} the last range has to last at least.
#' That is, if the last range would be shorter than ratio * \code{time.period},
#' the last range will be combined with the second last range.
#' A ratio of 0.0 means that the last range could be as small as possible.
#' A ratio of 1.0 means that the last range has to last at least \code{time.period}.
#' [default: 0.0]
#' @param include.end.date whether to include the end date or not [default: TRUE]
#' @param raw whether to return pairs of POSIXct objects or strings rather than
#' formatted strings [default: FALSE]
#'
#' @return the constructed ranges, either formatted or raw; the raw ranges are a named list,
#' for which the formatted ranges are the names
construct.overlapping.ranges = function(start, end, time.period, overlap, imperfect.range.ratio = 0.0,
include.end.date = TRUE, raw = FALSE) {
## convert given periods to lubridate stuff:
## 1) time period
time.period = lubridate::duration(time.period)
## 2) overlap as character string or percent of time.period
if (is.character(overlap)) {
overlap = lubridate::duration(overlap)
} else {
overlap = time.period * overlap
}
## 3) the dates for theirselves
start.date = get.date.from.string(start)
if (include.end.date) {
end.date = get.date.from.string(end) + 1 ## add 1 for inclusion of end.date
} else {
end.date = get.date.from.string(end)
}
## check the breaking case
if (overlap >= time.period) {
logging::logerror("The overlap (%s) is exceeding the given time period (%s).",
overlap, time.period)
stop("Stopping due to illegally specified overlap for overlapping ranges.")
}
## compute overall duration
bins.duration = lubridate::as.duration(lubridate::interval(start.date, end.date))
## compute negative overlap
overlap.negative = time.period - overlap
## compute number of complete bins
bins.number = floor(bins.duration / overlap.negative)
if (bins.number < 1) {
bins.number = 1
}
## generate a approximate sequence of dates which can be streamlined later
seq.start = start.date + overlap
seq.end = seq.start + (bins.number) * overlap.negative
ranges.approx = generate.date.sequence(seq.start, seq.end, by = overlap.negative)
## handle end date properly
if (end.date > seq.end) {
bins.number = bins.number + 1
if (seq.end == seq.start) {
ranges.approx = c(seq.start, end.date)
} else {
ranges.approx = c(ranges.approx, end.date)
}
}
## construct the raw ranges from the approximate ones
ranges.raw = lapply(seq_len(bins.number), function(bin.index) {
## combine start and end dates
bin.start = ranges.approx[[bin.index]] - overlap
bin.end = ranges.approx[[bin.index + 1]]
## check if we hit the end already
if (bin.end > end.date) {
bin.end = end.date
}
## construct current bin as the tuple of bin start and bin end
current.bin = c(bin.start, bin.end)
## explicitly set time-zone attribute 'tzone' again (as 'c.POSIXct' loses it)
current.bin = lubridate::with_tz(current.bin, tzone = TIMEZONE)
return(current.bin)
})
# if wanted, check for imperfect range in the end:
if (imperfect.range.ratio > 0) {
## 1) get the last range
last.range = ranges.raw[[bins.number]]
## 2) get the last range's duration
last.range.duration = lubridate::as.duration(lubridate::interval(last.range[1], last.range[2]))
## 3) check if the last range is too short
is.too.short = last.range.duration < imperfect.range.ratio * time.period
## 4) combine the last range with the second-last one, if the last one is too short
if (bins.number > 1 && is.too.short) {
## extend second-last range until end.date
ranges.raw[[bins.number - 1]][2] = end.date
## remove last range
ranges.raw = ranges.raw[-bins.number]
}
}
## construct actual range strings (without names)
ranges = sapply(ranges.raw, construct.ranges, sliding.window = FALSE, raw = FALSE)
ranges = unname(ranges)
## if raw is enabled, we need to attach proper names
if (raw) {
## add formatted ranges as names
names(ranges.raw) = ranges
## set as return value
ranges = ranges.raw
}
return(ranges)
}
#' Construct cumulative ranges based on the given start time, end time, and time period.
#' Each range starts at \code{start}; the first range lasts exactly \code{time.period}-long,
#' the second two times as long, etc.
#'
#' With this function, it is possible to construct ranges like this:
#' > +...
#' > ++..
#' > +++.
#' > ++++
#'
#' When the time difference between \code{start} and \code{end} is smaller than
#' \code{time.period}, just one range from \code{start} to \code{end} is constructed.
#'
#' Important: As the start of each range is supposed to be inclusive and the end of each range
#' exclusive, 1 second is added to \code{end}. This way, the date \code{end} will be *included*
#' in the last range. To avoid that behavior, set parameter \code{include.end.date} to \code{FALSE}.
#'
#' Note: You may want to use the function \code{ProjectData$get.data.timestamps} with this
#' function here.
#'
#' @param start The start time as string or POSIXct object
#' @param end The end time as string or POSIXct object; the last time to be *included* in the
#' last range (see above)
#' @param time.period The time period describing the length of the ranges, a character
#' string, e.g., "3 mins" or "15 days"
#' @param imperfect.range.ratio The ratio of the \code{time.period} the last range has to last at least.
#' That is, if the last range would be shorter than ratio * \code{time.period},
#' the last range will be combined with the second last range.
#' A ratio of 0.0 means that the last range could be as small as possible.
#' A ratio of 1.0 means that the last range has to last at least \code{time.period}.
#' [default: 0.0]
#' @param include.end.date whether to include the end date or not [default: TRUE]
#' @param raw whether to return pairs of POSIXct objects or strings rather than
#' formatted strings [default: FALSE]
#'
#' @return the constructed ranges, either formatted or raw; the raw ranges are a named list,
#' for which the formatted ranges are the names
construct.cumulative.ranges = function(start, end, time.period, imperfect.range.ratio = 0.0,
include.end.date = TRUE, raw = FALSE) {
## get the consecutive ranges to alter them afterwards
ranges.consecutive = construct.overlapping.ranges(start, end, time.period, overlap = 0,
imperfect.range.ratio = imperfect.range.ratio,
include.end.date = include.end.date, raw = TRUE)
## set the start of each range to global start date
ranges.raw = lapply(ranges.consecutive, function(range.bounds) {
## start of each range is the global start date
range.bounds[1] = start
return(range.bounds)
})
## construct actual range strings (without names)
ranges = sapply(ranges.raw, construct.ranges, sliding.window = FALSE, raw = FALSE)
ranges = unname(ranges)
## if raw is enabled, we need to attach proper names
if (raw) {
## add formatted ranges as names
names(ranges.raw) = ranges
## set as return value
ranges = ranges.raw
}
return(ranges)
}
#' Aggregate a given list/vector of ranges to specific levels, configurable through the
#' the parameter \code{aggregation.level} (see below for more details).
#'
#' Using different aggregation levels given by the parameter \code{aggregation.level},
#' it is possible to configure the exact treatment of range bounds and, thus, the
#' re-arrangement of the given list of ranges. The various aggregation levels work
#' as follows:
#' - \code{"range"}: The ranges will be kept exactly as given.
#' - \code{"cumulative"}: The ranges will be re-arranged in a cumulative manner.
#' - \code{"all.ranges"}: The ranges will be re-arranged to exactly to the time range
#' specified by the start of the first range and end of the last
#' range. All ranges will be exactly the same.
#' - \code{"project.cumulative"}: The same re-arrangement as for \code{"cumulative"}, but
#' all ranges will start at \code{project.start} and *not* at the
#' beginning of the first range.
#' - \code{"project.all.ranges"}: The same re-arrangement as for \code{"all.ranges"}, but
#' all ranges will start at \code{project.start} and *not* at
#' the beginning of the first range. All ranges will be exactly the same.
#' - \code{"complete"}: The same re-arrangement as for \code{"all.ranges"}, but all ranges
#' will start at \code{project.start} and end at \code{project.end}. All
#' ranges will be exactly the same.
#'
#' Note: You may want to use the function \code{ProjectData$get.data.timestamps} with this
#' function here, to pass proper values for \code{project.start} and \code{project.end}.
#'
#' Important: As the start of each range is supposed to be inclusive and the end of each range
#' exclusive, 1 second is added to \code{project.end}. All other range bounds are supposed to
#' be correctly constructed upfront, but if \code{project.end} comes from the function
#' \code{ProjectData$get.data.timestamps}, this is not respected directly. This way, the date
#' \code{project.end} will be *included* in the last range for the aggregation level
#' \code{"complete"}.
#'
#' @param ranges the list or vector of ranges to aggregate
#' @param project.start the project start time as string or POSIXct object
#' @param project.end the project end time as string or POSIXct object
#' @param aggregation.level One of \code{"range"}, \code{"cumulative"}, \code{"all.ranges"},
#' \code{"project.cumulative"}, \code{"project.all.ranges"}, and
#' \code{"complete"}. See above for more details. [default: "range"]
#' @param raw whether to return pairs of POSIXct objects or strings rather than
#' formatted strings [default: FALSE]
#'
#' @return the constructed ranges, either formatted or raw; the raw ranges are a named list,
#' for which the ranges from \code{ranges} are the names
aggregate.ranges = function(ranges, project.start, project.end,
aggregation.level = c("range", "cumulative", "all.ranges",
"project.cumulative", "project.all.ranges",
"complete"),
raw = FALSE) {
## get the chosen aggregation level
aggregation.level = match.arg(aggregation.level)
## get the timestamp data from the project data (needed for some aggr. levels)
project.start = get.date.from.string(project.start)
project.end = get.date.from.string(project.end) + 1 ## add 1 for inclusion of project.end
## with aggregation level "complete"
## loop over all ranges and split the data for each range accordingly:
list.of.range.bounds = lapply(ranges, get.range.bounds)
ranges.raw = lapply(ranges, function(range) {
## 1) get the range bounds to work with
start.end = get.range.bounds(range)
## 2) adjust the range bounds for the respective aggregation levels
## (if nothing else is stated below, the respective range bounds stay unchanged)
switch(aggregation.level,
range = {
## use the exact range bounds
},
cumulative = {
## the start is always at the first network's start bound
start.end[1] = list.of.range.bounds[[1]][1]
},
all.ranges = {
## the start is always at the first network's start bound
start.end[1] =list.of.range.bounds[[1]][1]
## the end is always at the last network's ending bound
start.end[2] = list.of.range.bounds[[length(ranges)]][2]
},
project.cumulative = {
## the start is always at the project data's start
start.end[1] = project.start
},
project.all.ranges = {
## the start is always at the project data's start
start.end[1] = project.start
## the end is always at the last network's ending bound
start.end[2] = list.of.range.bounds[[length(ranges)]][2]
},
complete = {
## the start is always at the project data's start
start.end[1] = project.start
## the start is always at the project data's ending
start.end[2] = project.end
}
)
return(start.end)
})
## construct actual range strings (without names)
ranges.new = sapply(ranges.raw, construct.ranges, sliding.window = FALSE, raw = FALSE)
ranges.new = unname(ranges.new)
## if raw is enabled, we need to attach proper names
if (raw) {
## add formatted original(!) ranges as names
if (is.list(ranges)) {
names(ranges.raw) = names(ranges)
} else {
names(ranges.raw) = ranges
}
## set as return value
ranges.new = ranges.raw
}
return(ranges.new)
}
#' Calculate the bounds of a range from its name.
#'
#' @param range The range name
#'
#' @return Returns a vector with two entries (start, end) of type POSIXct if input was a date;
#' or of type character if input was a commit hash or version;
#' or the unaltered given range if the string could not be parsed
get.range.bounds = function(range) {
## the patterns to test with appropriate conversions (if any)
tests = list(
## date format (assuming dates are GMT)
c("\\d{4}-\\d{2}-\\d{2}(\\s\\d{2}:\\d{2}:\\d{2})?", get.date.from.string),
## commit format
c("[A-F0-9a-f]{40}", identity),
## version format
c("([A-Za-z0-9]+[\\._]?)+", identity)
)
for (pattern in tests) {
start.end = regmatches(range, gregexpr(pattern = pattern[[1]], range))[[1]]
if (length(start.end) == 2) {
return (pattern[[2]](start.end))
}
}
return (range)
}
#' Get the data from a data frame in a specific range.
#'
#' @param range The range object that specifies the range
#' @param data The data frame that the data shall be extracted from. It must have a 'date' column.
#'
#' @return A data frame holding the data corresponding to the given range
get.data.from.range = function(range, data) {
range.bounds = get.range.bounds(range)
range.bounds.date = get.date.from.string(range.bounds)
df.bins = findInterval(data[["date"]], range.bounds.date, all.inside = FALSE)
## split data by this bin; this gives a list of three data frames, "0" contains the data before the range, "1" the
## data within the range and "2" the holds the data after the range
split.data = split.data.by.bins(data, df.bins)
## look for the element with name "1", as we are interested in the data within the range
## if there is no data, return an empty data frame corresponding to the data we want to cut
data.between = split.data["1"][[1]]
if (is.null(data.between)) {
## in order to get an empty data frame, get the data with all rows removed
empty.data = data[0, ]
return(empty.data)
} else {
return(data.between)
}
}