| 12
 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
 
 | .TH disk_log 3 "kernel  2.12.3" "Ericsson AB" "ERLANG MODULE DEFINITION"
.SH MODULE
disk_log \- A disk based term logging facility
.SH DESCRIPTION
.LP
\fIdisk_log\fR is a disk based term logger which makes it possible to efficiently log items on files\&. Two types of logs are supported, \fIhalt logs\fR and \fIwrap logs\fR\&. A halt log appends items to a single file, the size of which may or may not be limited by the disk log module, whereas a wrap log utilizes a sequence of wrap log files of limited size\&. As a wrap log file has been filled up, further items are logged onto to the next file in the sequence, starting all over with the first file when the last file has been filled up\&. For the sake of efficiency, items are always written to files as binaries\&. 
.LP
Two formats of the log files are supported, the \fIinternal format\fR and the \fIexternal format\fR\&. The internal format supports automatic repair of log files that have not been properly closed, and makes it possible to efficiently read logged items in \fIchunks\fR using a set of functions defined in this module\&. In fact, this is the only way to read internally formatted logs\&. The external format leaves it up to the user to read the logged deep byte lists\&. The disk log module cannot repair externally formatted logs\&. An item logged to an internally formatted log must not occupy more than 4 GB of disk space (the size must fit in 4 bytes)\&. 
.LP
For each open disk log there is one process that handles requests made to the disk log; the disk log process is created when \fIopen/1\fR is called, provided there exists no process handling the disk log\&. A process that opens a disk log can either be an \fIowner\fR or an anonymous \fIuser\fR of the disk log\&. Each owner is linked to the disk log process, and the disk log is closed by the owner should the owner terminate\&. Owners can subscribe to \fInotifications\fR, messages of the form \fI{disk_log, Node, Log, Info}\fR that are sent from the disk log process when certain events occur, see the commands below and in particular the \fIopen/1\fR option notify\&. There can be several owners of a log, but a process cannot own a log more than once\&. One and the same process may, however, open the log as a user more than once\&. For a disk log process to properly close its file and terminate, it must be closed by its owners and once by some non-owner process for each time the log was used anonymously; the users are counted, and there must not be any users left when the disk log process terminates\&. 
.LP
Items can be logged \fIsynchronously\fR by using the functions \fIlog/2\fR, \fIblog/2\fR, \fIlog_terms/2\fR and \fIblog_terms/2\fR\&. For each of these functions, the caller is put on hold until the items have been logged (but not necessarily written, use \fIsync/1\fR to ensure that)\&. By adding an \fIa\fR to each of the mentioned function names we get functions that log items \fIasynchronously\fR\&. Asynchronous functions do not wait for the disk log process to actually write the items to the file, but return the control to the caller more or less immediately\&. 
.LP
When using the internal format for logs, the functions \fIlog/2\fR, \fIlog_terms/2\fR, \fIalog/2\fR, and \fIalog_terms/2\fR should be used\&. These functions log one or more Erlang terms\&. By prefixing each of the functions with a \fIb\fR (for "binary") we get the corresponding \fIblog\fR functions for the external format\&. These functions log one or more deep lists of bytes or, alternatively, binaries of deep lists of bytes\&. For example, to log the string \fI"hello"\fR in ASCII format, we can use \fIdisk_log:blog(Log, "hello")\fR, or \fIdisk_log:blog(Log, list_to_binary("hello"))\fR\&. The two alternatives are equally efficient\&. The \fIblog\fR functions can be used for internally formatted logs as well, but in this case they must be called with binaries constructed with calls to \fIterm_to_binary/1\fR\&. There is no check to ensure this, it is entirely the responsibility of the caller\&. If these functions are called with binaries that do not correspond to Erlang terms, the \fIchunk/2, 3\fR and automatic repair functions will fail\&. The corresponding terms (not the binaries) will be returned when \fIchunk/2, 3\fR is called\&. 
.LP
A collection of open disk logs with the same name running on different nodes is said to be a \fIa distributed disk log\fR if requests made to any one of the logs are automatically made to the other logs as well\&. The members of such a collection will be called individual distributed disk logs, or just distributed disk logs if there is no risk of confusion\&. There is no order between the members of such a collection\&. For instance, logged terms are not necessarily written onto the node where the request was made before written onto the other nodes\&. One could note here that there are a few functions that do not make requests to all members of distributed disk logs, namely \fIinfo\fR, \fIchunk\fR, \fIbchunk\fR, \fIchunk_step\fR and \fIlclose\fR\&. An open disk log that is not a distributed disk log is said to be a \fIlocal disk log\fR\&. A local disk log is accessible only from the node where the disk log process runs, whereas a distributed disk log is accessible from all nodes in the Erlang system, with exception for those nodes where a local disk log with the same name as the distributed disk log exists\&. All processes on nodes that have access to a local or distributed disk log can log items or otherwise change, inspect or close the log\&. 
.LP
It is not guaranteed that all log files of a distributed disk log contain the same log items; there is no attempt made to synchronize the contents of the files\&. However, as long as at least one of the involved nodes is alive at each time, all items will be logged\&. When logging items to a distributed log, or otherwise trying to change the log, the replies from individual logs are ignored\&. If all nodes are down, the disk log functions reply with a \fInonode\fR error\&. 
.SS Note:
.LP
In some applications it may not be acceptable that replies from individual logs are ignored\&. An alternative in such situations is to use several local disk logs instead of one distributed disk log, and implement the distribution without use of the disk log module\&.
.LP
Errors are reported differently for asynchronous log attempts and other uses of the disk log module\&. When used synchronously the disk log module replies with an error message, but when called asynchronously, the disk log module does not know where to send the error message\&. Instead owners subscribing to notifications will receive an \fIerror_status\fR message\&. 
.LP
The disk log module itself does not report errors to the \fIerror_logger\fR module; it is up to the caller to decide whether the error logger should be employed or not\&. The function \fIformat_error/1\fR can be used to produce readable messages from error replies\&. Information events are however sent to the error logger in two situations, namely when a log is repaired, or when a file is missing while reading chunks\&. 
.LP
The error message \fIno_such_log\fR means that the given disk log is not currently open\&. Nothing is said about whether the disk log files exist or not\&. 
.SS Note:
.LP
If an attempt to reopen or truncate a log fails (see \fIreopen\fR and \fItruncate\fR) the disk log process immediately terminates\&. Before the process terminates links to to owners and blocking processes (see \fIblock\fR) are removed\&. The effect is that the links work in one direction only; any process using a disk log has to check for the error message \fIno_such_log\fR if some other process might truncate or reopen the log simultaneously\&.
.SH EXPORTS
.LP
.B
accessible_logs() -> {[LocalLog], [DistributedLog]}
.br
.RS
.TP
Types
LocalLog = DistributedLog = term()
.br
.RE
.RS
.LP
The \fIaccessible_logs/0\fR function returns the names of the disk logs accessible on the current node\&. The first list contains local disk logs, and the second list contains distributed disk logs\&. 
.RE
.LP
.B
alog(Log, Term)
.br
.B
balog(Log, Bytes) -> ok | {error, Reason}
.br
.RS
.TP
Types
Log = term()
.br
Term = term()
.br
Bytes = binary() | [Byte]
.br
Byte = [Byte] | 0 =< integer() =< 255
.br
Reason = no_such_log
.br
.RE
.RS
.LP
The \fIalog/2\fR and \fIbalog/2\fR functions asynchronously append an item to a disk log\&. The function \fIalog/2\fR is used for internally formatted logs, and the function \fIbalog/2\fR for externally formatted logs\&. \fIbalog/2\fR can be used for internally formatted logs as well provided the binary was constructed with a call to \fIterm_to_binary/1\fR\&. 
.LP
The owners that subscribe to notifications will receive the message \fIread_only\fR, \fIblocked_log\fR or \fIformat_external\fR in case the item cannot be written on the log, and possibly one of the messages \fIwrap\fR, \fIfull\fR and \fIerror_status\fR if an item was written on the log\&. The message \fIerror_status\fR is sent if there is something wrong with the header function or a file error occurred\&. 
.RE
.LP
.B
alog_terms(Log, TermList)
.br
.B
balog_terms(Log, BytesList) -> ok | {error, Reason}
.br
.RS
.TP
Types
Log = term()
.br
TermList = [term()]
.br
BytesList = [Bytes]
.br
Bytes = binary() | [Byte]
.br
Byte = [Byte] | 0 =< integer() =< 255
.br
Reason = no_such_log
.br
.RE
.RS
.LP
The \fIalog_terms/2\fR and \fIbalog_terms/2\fR functions asynchronously append a list of items to a disk log\&. The function \fIalog_terms/2\fR is used for internally formatted logs, and the function \fIbalog_terms/2\fR for externally formatted logs\&. \fIbalog_terms/2\fR can be used for internally formatted logs as well provided the binaries were constructed with calls to \fIterm_to_binary/1\fR\&. 
.LP
The owners that subscribe to notifications will receive the message \fIread_only\fR, \fIblocked_log\fR or \fIformat_external\fR in case the items cannot be written on the log, and possibly one or more of the messages \fIwrap\fR, \fIfull\fR and \fIerror_status\fR if items were written on the log\&. The message \fIerror_status\fR is sent if there is something wrong with the header function or a file error occurred\&. 
.RE
.LP
.B
block(Log)
.br
.B
block(Log, QueueLogRecords) -> ok | {error, Reason}
.br
.RS
.TP
Types
Log = term()
.br
QueueLogRecords = bool()
.br
Reason = no_such_log | nonode | {blocked_log, Log}
.br
.RE
.RS
.LP
With a call to \fIblock/1, 2\fR a process can block a log\&. If the blocking process is not an owner of the log, a temporary link is created between the disk log process and the blocking process\&. The link is used to ensure that the disk log is unblocked should the blocking process terminate without first closing or unblocking the log\&. 
.LP
Any process can probe a blocked log with \fIinfo/1\fR or close it with \fIclose/1\fR\&. The blocking process can also use the functions \fIchunk/2, 3\fR, \fIbchunk/2, 3\fR, \fIchunk_step/3\fR, and \fIunblock/1\fR without being affected by the block\&. Any other attempt than those hitherto mentioned to update or read a blocked log suspends the calling process until the log is unblocked or returns an error message \fI{blocked_log, Log}\fR, depending on whether the value of \fIQueueLogRecords\fR is \fItrue\fR or \fIfalse\fR\&. The default value of \fIQueueLogRecords\fR is \fItrue\fR, which is used by \fIblock/1\fR\&. 
.RE
.LP
.B
change_header(Log, Header) -> ok | {error, Reason}
.br
.RS
.TP
Types
Log = term()
.br
Header = {head, Head} | {head_func, {M, F, A}}
.br
Head = none | term() | binary() | [Byte]
.br
Byte = [Byte] | 0 =< integer() =< 255
.br
Reason = no_such_log | nonode | {read_only_mode, Log} | {blocked_log, Log} | {badarg, head}
.br
.RE
.RS
.LP
The \fIchange_header/2\fR function changes the value of the \fIhead\fR or \fIhead_func\fR option of a disk log\&.
.RE
.LP
.B
change_notify(Log, Owner, Notify) -> ok | {error, Reason}
.br
.RS
.TP
Types
Log = term()
.br
Owner = pid()
.br
Notify = bool()
.br
Reason = no_such_log | nonode | {blocked_log, Log} | {badarg, notify} | {not_owner, Owner}
.br
.RE
.RS
.LP
The \fIchange_notify/3\fR function changes the value of the \fInotify\fR option for an owner of a disk log\&. 
.RE
.LP
.B
change_size(Log, Size) -> ok | {error, Reason}
.br
.RS
.TP
Types
Log = term()
.br
Size = integer() > 0 | infinity | {MaxNoBytes, MaxNoFiles}
.br
MaxNoBytes = integer() > 0
.br
MaxNoFiles = integer() > 0
.br
Reason = no_such_log | nonode | {read_only_mode, Log} | {blocked_log, Log} | {new_size_too_small, CurrentSize} | {badarg, size} | {file_error, FileName, FileError}
.br
.RE
.RS
.LP
The \fIchange_size/2\fR function changes the size of an open log\&. For a halt log it is always possible to increase the size, but it is not possible to decrease the size to something less than the current size of the file\&. 
.LP
For a wrap log it is always possible to increase both the size and number of files, as long as the number of files does not exceed 65000\&. If the maximum number of files is decreased, the change will not be valid until the current file is full and the log wraps to the next file\&. The redundant files will be removed next time the log wraps around, i\&.e\&. starts to log to file number 1\&. 
.LP
As an example, assume that the old maximum number of files is 10 and that the new maximum number of files is 6\&. If the current file number is not greater than the new maximum number of files, the files 7 to 10 will be removed when file number 6 is full and the log starts to write to file number 1 again\&. Otherwise the files greater than the current file will be removed when the current file is full (e\&.g\&. if the current file is 8, the files 9 and 10); the files between new maximum number of files and the current file (i\&.e\&. files 7 and 8) will be removed next time file number 6 is full\&. 
.LP
If the size of the files is decreased the change will immediately affect the current log\&. It will not of course change the size of log files already full until next time they are used\&. 
.LP
If the log size is decreased for instance to save space, the function \fIinc_wrap_file/1\fR can be used to force the log to wrap\&. 
.RE
.LP
.B
chunk(Log, Continuation)
.br
.B
chunk(Log, Continuation, N) -> {Continuation2, Terms} | {Continuation2, Terms, Badbytes} | eof | {error, Reason}
.br
.B
bchunk(Log, Continuation)
.br
.B
bchunk(Log, Continuation, N) -> {Continuation2, Binaries} | {Continuation2, Binaries, Badbytes} | eof | {error, Reason}
.br
.RS
.TP
Types
Log = term()
.br
Continuation = start | cont()
.br
N = integer() > 0 | infinity
.br
Continuation2 = cont()
.br
Terms = [term()]
.br
Badbytes = integer()
.br
Reason = no_such_log | {format_external, Log} | {blocked_log, Log} | {badarg, continuation} | {not_internal_wrap, Log} | {corrupt_log_file, FileName} | {file_error, FileName, FileError}
.br
Binaries = [binary()]
.br
.RE
.RS
.LP
The \fIchunk/2, 3\fR and \fIbchunk/2, 3\fR functions make it possible to efficiently read the terms which have been appended to an internally formatted log\&. It minimizes disk I/O by reading 64 kilobyte chunks from the file\&. The \fIbchunk/2, 3\fR functions return the binaries read from the file; they do not call \fIbinary_to_term\fR\&. Otherwise the work just like \fIchunk/2, 3\fR\&. 
.LP
The first time \fIchunk\fR (or \fIbchunk\fR) is called, an initial continuation, the atom \fIstart\fR, must be provided\&. If there is a disk log process running on the current node, terms are read from that log, otherwise an individual distributed log on some other node is chosen, if such a log exists\&. 
.LP
When \fIchunk/3\fR is called, \fIN\fR controls the maximum number of terms that are read from the log in each chunk\&. Default is \fIinfinity\fR, which means that all the terms contained in the 64 kilobyte chunk are read\&. If less than \fIN\fR terms are returned, this does not necessarily mean that the end of the file has been reached\&. 
.LP
The \fIchunk\fR function returns a tuple \fI{Continuation2, Terms}\fR, where \fITerms\fR is a list of terms found in the log\&. \fIContinuation2\fR is yet another continuation which must be passed on to any subsequent calls to \fIchunk\fR\&. With a series of calls to \fIchunk\fR it is possible to extract all terms from a log\&. 
.LP
The \fIchunk\fR function returns a tuple \fI{Continuation2, Terms, Badbytes}\fR if the log is opened in read-only mode and the read chunk is corrupt\&. \fIBadbytes\fR is the number of bytes in the file which were found not to be Erlang terms in the chunk\&. Note also that the log is not repaired\&. When trying to read chunks from a log opened in read-write mode, the tuple \fI{corrupt_log_file, FileName}\fR is returned if the read chunk is corrupt\&. 
.LP
\fIchunk\fR returns \fIeof\fR when the end of the log is reached, or \fI{error, Reason}\fR if an error occurs\&. Should a wrap log file be missing, a message is output on the error log\&. 
.LP
When \fIchunk/2, 3\fR is used with wrap logs, the returned continuation may or may not be valid in the next call to \fIchunk\fR\&. This is because the log may wrap and delete the file into which the continuation points\&. To make sure this does not happen, the log can be blocked during the search\&. 
.RE
.LP
.B
chunk_info(Continuation) -> InfoList | {error, Reason}
.br
.RS
.TP
Types
Continuation = cont()
.br
Reason = {no_continuation, Continuation}
.br
.RE
.RS
.LP
The \fIchunk_info/1\fR function returns the following pair describing the chunk continuation returned by \fIchunk/2, 3\fR, \fIbchunk/2, 3\fR, or \fIchunk_step/3\fR: 
.RS 2
.TP 2
*
\fI{node, Node}\fR\&. Terms are read from the disk log running on \fINode\fR\&.
.RE
.RE
.LP
.B
chunk_step(Log, Continuation, Step) -> {ok, Continuation2} | {error, Reason}
.br
.RS
.TP
Types
Log = term()
.br
Continuation = start | cont()
.br
Step = integer()
.br
Continuation2 = cont()
.br
Reason = no_such_log | end_of_log | {format_external, Log} | {blocked_log, Log} | {badarg, continuation} | {file_error, FileName, FileError}
.br
.RE
.RS
.LP
The function \fIchunk_step\fR can be used in conjunction with \fIchunk/2, 3\fR and \fIbchunk/2, 3\fR to search through an internally formatted wrap log\&. It takes as argument a continuation as returned by \fIchunk/2, 3\fR, \fIbchunk/2, 3\fR, or \fIchunk_step/3\fR, and steps forward (or backward) \fIStep\fR files in the wrap log\&. The continuation returned points to the first log item in the new current file\&. 
.LP
If the atom \fIstart\fR is given as continuation, a disk log to read terms from is chosen\&. A local or distributed disk log on the current node is preferred to an individual distributed log on some other node\&. 
.LP
If the wrap log is not full because all files have not been used yet, \fI{error, end_of_log}\fR is returned if trying to step outside the log\&. 
.RE
.LP
.B
close(Log) -> ok | {error, Reason}
.br
.RS
.TP
Types
Reason = no_such_log | nonode | {file_error, FileName, FileError}
.br
.RE
.RS
.LP
 The function \fIclose/1\fR closes a local or distributed disk log properly\&. An internally formatted log must be closed before the Erlang system is stopped, otherwise the log is regarded as unclosed and the automatic repair procedure will be activated next time the log is opened\&. 
.LP
The disk log process in not terminated as long as there are owners or users of the log\&. It should be stressed that each and every owner must close the log, possibly by terminating, and that any other process - not only the processes that have opened the log anonymously - can decrement the \fIusers\fR counter by closing the log\&. Attempts to close a log by a process that is not an owner are simply ignored if there are no users\&. 
.LP
If the log is blocked by the closing process, the log is also unblocked\&. 
.RE
.LP
.B
format_error(Error) -> Chars
.br
.RS
.TP
Types
Chars = [char() | Chars]
.br
.RE
.RS
.LP
Given the error returned by any function in this module, the function \fIformat_error\fR returns a descriptive string of the error in English\&. For file errors, the function \fIformat_error/1\fR in the \fIfile\fR module is called\&.
.RE
.LP
.B
inc_wrap_file(Log) -> ok | {error, Reason}
.br
.RS
.TP
Types
Reason = no_such_log | nonode | {read_only_mode, Log} | {blocked_log, Log} | {halt_log, Log} | {invalid_header, InvalidHeader} | {file_error, FileName, FileError}
.br
.RE
.RS
.LP
The \fIinc_wrap_file/1\fR function forces the internally formatted disk log to start logging to the next log file\&. It can be used, for instance, in conjunction with \fIchange_size/2\fR to reduce the amount of disk space allocated by the disk log\&. 
.LP
The owners that subscribe to notifications will normally receive a \fIwrap\fR message, but in case of an error with a reason tag of \fIinvalid_header\fR or \fIfile_error\fR an \fIerror_status\fR message will be sent\&.
.RE
.LP
.B
info(Log) -> InfoList | {error, no_such_log}
.br
.RS
.LP
The \fIinfo/1\fR function returns a list of \fI{Tag, Value}\fR pairs describing the log\&. If there is a disk log process running on the current node, that log is used as source of information, otherwise an individual distributed log on some other node is chosen, if such a log exists\&. 
.LP
The following pairs are returned for all logs: 
.RS 2
.TP 2
*
\fI{name, Log}\fR, where \fILog\fR is the name of the log as given by the \fIopen/1\fR option \fIname\fR\&.
.TP 2
*
\fI{file, File}\fR\&. For halt logs \fIFile\fR is the filename, and for wrap logs \fIFile\fR is the base name\&.
.TP 2
*
\fI{type, Type}\fR, where \fIType\fR is the type of the log as given by the \fIopen/1\fR option \fItype\fR\&.
.TP 2
*
\fI{format, Format}\fR, where \fIFormat\fR is the format of the log as given by the \fIopen/1\fR option \fIformat\fR\&.
.TP 2
*
\fI{size, Size}\fR, where \fISize\fR is the size of the log as given by the \fIopen/1\fR option \fIsize\fR, or the size set by \fIchange_size/2\fR\&. The value set by \fIchange_size/2\fR is reflected immediately\&.
.TP 2
*
\fI{mode, Mode}\fR, where \fIMode\fR is the mode of the log as given by the \fIopen/1\fR option \fImode\fR\&.
.TP 2
*
\fI{owners, [{pid(), Notify}]}\fR where \fINotify\fR is the value set by the \fIopen/1\fR option \fInotify\fR or the function \fIchange_notify/3\fR for the owners of the log\&.
.TP 2
*
\fI{users, Users}\fR where \fIUsers\fR is the number of anonymous users of the log, see the \fIopen/1\fR option linkto\&.
.TP 2
*
\fI{status, Status}\fR, where \fIStatus\fR is \fIok\fR or \fI{blocked, QueueLogRecords}\fR as set by the functions \fIblock/1, 2\fR and \fIunblock/1\fR\&.
.TP 2
*
\fI{node, Node}\fR\&. The information returned by the current invocation of the \fIinfo/1\fR function has been gathered from the disk log process running on \fINode\fR\&.
.TP 2
*
\fI{distributed, Dist}\fR\&. If the log is local on the current node, then \fIDist\fR has the value \fIlocal\fR, otherwise all nodes where the log is distributed are returned as a list\&.
.RE
.LP
The following pairs are returned for all logs opened in \fIread_write\fR mode: 
.RS 2
.TP 2
*
\fI{head, Head}\fR\&. Depending of the value of the \fIopen/1\fR options \fIhead\fR and \fIhead_func\fR or set by the function \fIchange_header/2\fR, the value of \fIHead\fR is \fInone\fR (default), \fI{head, H}\fR (\fIhead\fR option) or \fI{M, F, A}\fR (\fIhead_func\fR option)\&.
.TP 2
*
\fI{no_written_items, NoWrittenItems}\fR, where \fINoWrittenItems\fR is the number of items written to the log since the disk log process was created\&.
.RE
.LP
The following pair is returned for halt logs opened in \fIread_write\fR mode: 
.RS 2
.TP 2
*
\fI{full, Full}\fR, where \fIFull\fR is \fItrue\fR or \fIfalse\fR depending on whether the halt log is full or not\&.
.RE
.LP
The following pairs are returned for wrap logs opened in \fIread_write\fR mode: 
.RS 2
.TP 2
*
\fI{no_current_bytes, integer() >= 0}\fR is the number of bytes written to the current wrap log file\&.
.TP 2
*
\fI{no_current_items, integer() >= 0}\fR is the number of items written to the current wrap log file, header inclusive\&.
.TP 2
*
\fI{no_items, integer() >= 0}\fR is the total number of items in all wrap log files\&.
.TP 2
*
\fI{current_file, integer()}\fR is the ordinal for the current wrap log file in the range \fI1\&.\&.MaxNoFiles\fR, where \fIMaxNoFiles\fR is given by the \fIopen/1\fR option \fIsize\fR or set by \fIchange_size/2\fR\&.
.TP 2
*
\fI{no_overflows, {SinceLogWasOpened, SinceLastInfo}}\fR, where \fISinceLogWasOpened\fR (\fISinceLastInfo\fR) is the number of times a wrap log file has been filled up and a new one opened or \fIinc_wrap_file/1\fR has been called since the disk log was last opened (\fIinfo/1\fR was last called)\&. The first time \fIinfo/2\fR is called after a log was (re)opened or truncated, the two values are equal\&.
.RE
.LP
Note that the \fIchunk/2, 3\fR, \fIbchunk/2, 3\fR, and \fIchunk_step/3\fR functions do not affect any value returned by \fIinfo/1\fR\&. 
.RE
.LP
.B
lclose(Log)
.br
.B
lclose(Log, Node) -> ok | {error, Reason}
.br
.RS
.TP
Types
Node = node()
.br
Reason = no_such_log | {file_error, FileName, FileError}
.br
.RE
.RS
.LP
The function \fIlclose/1\fR closes a local log or an individual distributed log on the current node\&. The function \fIlclose/2\fR closes an individual distributed log on the specified node if the node is not the current one\&. \fIlclose(Log)\fR is equivalent to \fIlclose(Log, node())\fR\&. See also close/1\&. 
.LP
If there is no log with the given name on the specified node, \fIno_such_log\fR is returned\&. 
.RE
.LP
.B
log(Log, Term)
.br
.B
blog(Log, Bytes) -> ok | {error, Reason}
.br
.RS
.TP
Types
Log = term()
.br
Term = term()
.br
Bytes = binary() | [Byte]
.br
Byte = [Byte] | 0 =< integer() =< 255
.br
Reason = no_such_log | nonode | {read_only_mode, Log} | {format_external, Log} | {blocked_log, Log} | {full, Log} | {invalid_header, InvalidHeader} | {file_error, FileName, FileError}
.br
.RE
.RS
.LP
The \fIlog/2\fR and \fIblog/2\fR functions synchronously append a term to a disk log\&. They return \fIok\fR or \fI{error, Reason}\fR when the term has been written to disk\&. If the log is distributed, \fIok\fR is always returned, unless all nodes are down\&. Terms are written by means of the ordinary \fIwrite()\fR function of the operating system\&. Hence, there is no guarantee that the term has actually been written to the disk, it might linger in the operating system kernel for a while\&. To make sure the item is actually written to disk, the \fIsync/1\fR function must be called\&. 
.LP
The \fIlog/2\fR function is used for internally formatted logs, and \fIblog/2\fR for externally formatted logs\&. \fIblog/2\fR can be used for internally formatted logs as well provided the binary was constructed with a call to \fIterm_to_binary/1\fR\&. 
.LP
The owners that subscribe to notifications will be notified of an error with an \fIerror_status\fR message if the error reason tag is \fIinvalid_header\fR or \fIfile_error\fR\&. 
.RE
.LP
.B
log_terms(Log, TermList)
.br
.B
blog_terms(Log, BytesList) -> ok | {error, Reason}
.br
.RS
.TP
Types
Log = term()
.br
TermList = [term()]
.br
BytesList = [Bytes]
.br
Bytes = binary() | [Byte]
.br
Byte = [Byte] | 0 =< integer() =< 255
.br
Reason = no_such_log | nonode | {read_only_mode, Log} | {format_external, Log} | {blocked_log, Log} | {full, Log} | {invalid_header, InvalidHeader} | {file_error, FileName, FileError}
.br
.RE
.RS
.LP
The \fIlog_terms/2\fR and \fIblog_terms/2\fR functions synchronously append a list of items to the log\&. The benefit of using these functions rather than the \fIlog/2\fR and \fIblog/2\fR functions is that of efficiency: the given list is split into as large sublists as possible (limited by the size of wrap log files), and each sublist is logged as one single item, which reduces the overhead\&. 
.LP
The \fIlog_terms/2\fR function is used for internally formatted logs, and \fIblog_terms/2\fR for externally formatted logs\&. \fIblog_terms/2\fR can be used for internally formatted logs as well provided the binaries were constructed with calls to \fIterm_to_binary/1\fR\&. 
.LP
The owners that subscribe to notifications will be notified of an error with an \fIerror_status\fR message if the error reason tag is \fIinvalid_header\fR or \fIfile_error\fR\&. 
.RE
.LP
.B
open(ArgL) -> OpenRet | DistOpenRet
.br
.RS
.TP
Types
ArgL = [Opt]
.br
Opt = {name, term()} | {file, FileName}, {linkto, LinkTo} | {repair, Repair} | {type, Type} | {format, Format} | {size, Size} | {distributed, [Node]} | {notify, bool()} | {head, Head} | {head_func, {M, F, A}} | {mode, Mode}
.br
FileName = string() | atom()
.br
LinkTo = pid() | none
.br
Repair = true | false | truncate
.br
Type = halt | wrap
.br
Format = internal | external
.br
Size = integer() > 0 | infinity | {MaxNoBytes, MaxNoFiles}
.br
MaxNoBytes = integer() > 0
.br
MaxNoFiles = 0 < integer() < 65000
.br
Rec = integer()
.br
Bad = integer()
.br
Head = none | term() | binary() | [Byte]
.br
Byte = [Byte] | 0 =< integer() =< 255
.br
Mode = read_write | read_only
.br
OpenRet = Ret | {error, Reason}
.br
DistOpenRet = {[{Node, Ret}], [{BadNode, {error, DistReason}}]}
.br
Node = BadNode = atom()
.br
Ret = {ok, Log} | {repaired, Log, {recovered, Rec}, {badbytes, Bad}}
.br
DistReason = nodedown | Reason
.br
Reason = no_such_log | {badarg, Arg} | {size_mismatch, CurrentSize, NewSize} | {arg_mismatch, OptionName, CurrentValue, Value} | {name_already_open, Log} | {open_read_write, Log} | {open_read_only, Log} | {need_repair, Log} | {not_a_log_file, FileName} | {invalid_index_file, FileName} | {invalid_header, InvalidHeader} | {file_error, FileName, FileError} | {node_already_open, Log}
.br
.RE
.RS
.LP
The \fIArgL\fR parameter is a list of options which have the following meanings:
.RS 2
.TP 2
*
\fI{name, Log}\fR specifies the name of the log\&. This is the name which must be passed on as a parameter in all subsequent logging operations\&. A name must always be supplied\&. 
.TP 2
*
\fI{file, FileName}\fR specifies the name of the file which will be used for logged terms\&. If this value is omitted and the name of the log is either an atom or a string, the file name will default to \fIlists:concat([Log, "\&.LOG"])\fR for halt logs\&. For wrap logs, this will be the base name of the files\&. Each file in a wrap log will be called \fI<base_name>\&.N\fR, where \fIN\fR is an integer\&. Each wrap log will also have two files called \fI<base_name>\&.idx\fR and \fI<base_name>\&.siz\fR\&. 
.TP 2
*
\fI{linkto, LinkTo}\fR\&.  If \fILinkTo\fR is a pid, that pid becomes an owner of the log\&. If \fILinkTo\fR is \fInone\fR the log records that it is used anonymously by some process by incrementing the \fIusers\fR counter\&. By default, the process which calls \fIopen/1\fR owns the log\&. 
.TP 2
*
\fI{repair, Repair}\fR\&. If \fIRepair\fR is \fItrue\fR, the current log file will be repaired, if needed\&. As the restoration is initiated, a message is output on the error log\&. If \fIfalse\fR is given, no automatic repair will be attempted\&. Instead, the tuple \fI{error, {need_repair, Log}}\fR is returned if an attempt is made to open a corrupt log file\&. If \fItruncate\fR is given, the log file will be truncated, creating an empty log\&. Default is \fItrue\fR, which has no effect on logs opened in read-only mode\&. 
.TP 2
*
\fI{type, Type}\fR is the type of the log\&. Default is \fIhalt\fR\&. 
.TP 2
*
\fI{format, Format}\fR specifies the format of the disk log\&. Default is \fIinternal\fR\&. 
.TP 2
*
\fI{size, Size}\fR specifies the size of the log\&. When a halt log has reached its maximum size, all attempts to log more items are rejected\&. The default size is \fIinfinity\fR, which for halt implies that there is no maximum size\&. For wrap logs, the \fISize\fR parameter may be either a pair \fI{MaxNoBytes, MaxNoFiles}\fR or \fIinfinity\fR\&. In the latter case, if the files of an already existing wrap log with the same name can be found, the size is read from the existing wrap log, otherwise an error is returned\&. Wrap logs write at most \fIMaxNoBytes\fR bytes on each file and use \fIMaxNoFiles\fR files before starting all over with the first wrap log file\&. Regardless of \fIMaxNoBytes\fR, at least the header (if there is one) and one item is written on each wrap log file before wrapping to the next file\&. When opening an existing wrap log, it is not necessary to supply a value for the option \fISize\fR, but any supplied value must equal the current size of the log, otherwise the tuple \fI{error, {size_mismatch, CurrentSize, NewSize}}\fR is returned\&. 
.TP 2
*
\fI{distributed, Nodes}\fR\&. This option can be used for adding members to a distributed disk log\&. The default value is \fI[]\fR, which means that the log is local on the current node\&. 
.TP 2
*
 
.RS 2
.LP
\fI{notify, bool()}\fR\&. If \fItrue\fR, the owners of the log are notified when certain events occur in the log\&. Default is \fIfalse\fR\&. The owners are sent one of the following messages when an event occurs: 
.LP
.RS 2
.TP 2
-
\fI{disk_log, Node, Log, {wrap, NoLostItems}}\fR is sent when a wrap log has filled up one of its files and a new file is opened\&. \fINoLostItems\fR is the number of previously logged items that have been lost when truncating existing files\&. 
.TP 2
-
\fI{disk_log, Node, Log, {truncated, NoLostItems}}\fR is sent when a log has been truncated or reopened\&. For halt logs \fINoLostItems\fR is the number of items written on the log since the disk log process was created\&. For wrap logs \fINoLostItems\fR is the number of items on all wrap log files\&. 
.TP 2
-
\fI{disk_log, Node, Log, {read_only, Items}}\fR is sent when an asynchronous log attempt is made to a log file opened in read-only mode\&. \fIItems\fR is the items from the log attempt\&. 
.TP 2
-
\fI{disk_log, Node, Log, {blocked_log, Items}}\fR is sent when an asynchronous log attempt is made to a blocked log that does not queue log attempts\&. \fIItems\fR is the items from the log attempt\&. 
.TP 2
-
\fI{disk_log, Node, Log, {format_external, Items}}\fR is sent when \fIalog/2\fR or \fIalog_terms/2\fR is used for internally formatted logs\&. \fIItems\fR is the items from the log attempt\&. 
.TP 2
-
\fI{disk_log, Node, Log, full}\fR is sent when an attempt to log items to a wrap log would write more bytes than the limit set by the \fIsize\fR option\&. 
.TP 2
-
\fI{disk_log, Node, Log, {error_status, Status}}\fR is sent when the error status changes\&. The error status is defined by the outcome of the last attempt to log items to a the log or to truncate the log or the last use of \fIsync/1\fR, \fIinc_wrap_file/1\fR or \fIchange_size/2\fR\&. \fIStatus\fR is one of \fIok\fR and \fI{error, Error}\fR, the former being the initial value\&. 
.RE
.RE
.TP 2
*
\fI{head, Head}\fR specifies a header to be written first on the log file\&. If the log is a wrap log, the item \fIHead\fR is written first in each new file\&. \fIHead\fR should be a term if the format is \fIinternal\fR, and a deep list of bytes (or a binary) otherwise\&. Default is \fInone\fR, which means that no header is written first on the file\&. 
.TP 2
*
\fI{head_func, {M, F, A}}\fR specifies a function to be called each time a new log file is opened\&. The call \fIM:F(A)\fR is assumed to return \fI{ok, Head}\fR\&. The item \fIHead\fR is written first in each file\&. \fIHead\fR should be a term if the format is \fIinternal\fR, and a deep list of bytes (or a binary) otherwise\&. 
.TP 2
*
\fI{mode, Mode}\fR specifies if the log is to be opened in read-only or read-write mode\&. It defaults to \fIread_write\fR\&. 
.RE
.LP
The \fIopen/1\fR function returns \fI{ok, Log}\fR if the log file was successfully opened\&. If the file was successfully repaired, the tuple \fI{repaired, Log, {recovered, Rec}, {badbytes, Bad}}\fR is returned, where \fIRec\fR is the number of whole Erlang terms found in the file and \fIBad\fR is the number of bytes in the file which were non-Erlang terms\&. If the \fIdistributed\fR parameter was given, \fIopen/1\fR returns a list of successful replies and a list of erroneous replies\&. Each reply is tagged with the node name\&. 
.LP
When a disk log is opened in read-write mode, any existing log file is checked for\&. If there is none a new empty log is created, otherwise the existing file is opened at the position after the last logged item, and the logging of items will commence from there\&. If the format is \fIinternal\fR and the existing file is not recognized as an internally formatted log, a tuple \fI{error, {not_a_log_file, FileName}}\fR is returned\&. 
.LP
The \fIopen/1\fR function cannot be used for changing the values of options of an already open log; when there are prior owners or users of a log, all option values except \fIname\fR, \fIlinkto\fR and \fInotify\fR are just checked against the values that have been supplied before as option values to \fIopen/1\fR, \fIchange_header/2\fR, \fIchange_notify/3\fR or \fIchange_size/2\fR\&. As a consequence, none of the options except \fIname\fR is mandatory\&. If some given value differs from the current value, a tuple \fI{error, {arg_mismatch, OptionName, CurrentValue, Value}}\fR is returned\&. Caution: an owner\&'s attempt to open a log as owner once again is acknowledged with the return value \fI{ok, Log}\fR, but the state of the disk log is not affected in any way\&. 
.LP
If a log with a given name is local on some node, and one tries to open the log distributed on the same node, then the tuple \fI{error, {node_already_open, Name}}\fR is returned\&. The same tuple is returned if the log is distributed on some node, and one tries to open the log locally on the same node\&. Opening individual distributed disk logs for the first time adds those logs to a (possibly empty) distributed disk log\&. The option values supplied are used on all nodes mentioned by the \fIdistributed\fR option\&. Individual distributed logs know nothing about each other\&'s option values, so each node can be given unique option values by creating a distributed log with several calls to \fIopen/1\fR\&. 
.LP
It is possible to open a log file more than once by giving different values to the option \fIname\fR or by using the same file when distributing a log on different nodes\&. It is up to the user of the \fIdisk_log\fR module to ensure that no more than one disk log process has write access to any file, or the the file may be corrupted\&. 
.LP
If an attempt to open a log file for the first time fails, the disk log process terminates with the EXIT message \fI{{failed, Reason}, [{disk_log, open, 1}]}\fR\&. The function returns \fI{error, Reason}\fR for all other errors\&. 
.RE
.LP
.B
pid2name(Pid) -> {ok, Log} | undefined
.br
.RS
.TP
Types
Log = term()
.br
Pid = pid()
.br
.RE
.RS
.LP
The \fIpid2name/1\fR function returns the name of the log given the pid of a disk log process on the current node, or \fIundefined\fR if the given pid is not a disk log process\&. 
.LP
This function is meant to be used for debugging only\&. 
.RE
.LP
.B
reopen(Log, File)
.br
.B
reopen(Log, File, Head)
.br
.B
breopen(Log, File, BHead) -> ok | {error, Reason}
.br
.RS
.TP
Types
Log = term()
.br
File = string()
.br
Head = term()
.br
BHead = binary() | [Byte]
.br
Byte = [Byte] | 0 =< integer() =< 255
.br
Reason = no_such_log | nonode | {read_only_mode, Log} | {blocked_log, Log} | {same_file_name, Log} | {invalid_index_file, FileName} | {invalid_header, InvalidHeader} | {file_error, FileName, FileError}
.br
.RE
.RS
.LP
The \fIreopen\fR functions first rename the log file to \fIFile\fR and then re-create a new log file\&. In case of a wrap log, \fIFile\fR is used as the base name of the renamed files\&. By default the header given to \fIopen/1\fR is written first in the newly opened log file, but if the \fIHead\fR or the \fIBHead\fR argument is given, this item is used instead\&. The header argument is used once only; next time a wrap log file is opened, the header given to \fIopen/1\fR is used\&. 
.LP
The \fIreopen/2, 3\fR functions are used for internally formatted logs, and \fIbreopen/3\fR for externally formatted logs\&. 
.LP
The owners that subscribe to notifications will receive a \fItruncate\fR message\&. 
.LP
Upon failure to reopen the log, the disk log process terminates with the EXIT message \fI{{failed, Error}, [{disk_log, Fun, Arity}]}\fR, and other processes that have requests queued receive the message \fI{disk_log, Node, {error, disk_log_stopped}}\fR\&. 
.RE
.LP
.B
sync(Log) -> ok | {error, Reason}
.br
.RS
.TP
Types
Log = term()
.br
Reason = no_such_log | nonode | {read_only_mode, Log} | {blocked_log, Log} | {file_error, FileName, FileError}
.br
.RE
.RS
.LP
The \fIsync/1\fR function ensures that the contents of the log are actually written to the disk\&. This is usually a rather expensive operation\&. 
.RE
.LP
.B
truncate(Log)
.br
.B
truncate(Log, Head)
.br
.B
btruncate(Log, BHead) -> ok | {error, Reason}
.br
.RS
.TP
Types
Log = term()
.br
Head = term()
.br
BHead = binary() | [Byte]
.br
Byte = [Byte] | 0 =< integer() =< 255
.br
Reason = no_such_log | nonode | {read_only_mode, Log} | {blocked_log, Log} | {invalid_header, InvalidHeader} | {file_error, FileName, FileError}
.br
.RE
.RS
.LP
The \fItruncate\fR functions remove all items from a disk log\&. If the \fIHead\fR or the \fIBHead\fR argument is given, this item is written first in the newly truncated log, otherwise the header given to \fIopen/1\fR is used\&. The header argument is only used once; next time a wrap log file is opened, the header given to \fIopen/1\fR is used\&. 
.LP
The \fItruncate/1, 2\fR functions are used for internally formatted logs, and \fIbtruncate/2\fR for externally formatted logs\&. 
.LP
The owners that subscribe to notifications will receive a \fItruncate\fR message\&. 
.LP
If the attempt to truncate the log fails, the disk log process terminates with the EXIT message \fI{{failed, Reason}, [{disk_log, Fun, Arity}]}\fR, and other processes that have requests queued receive the message \fI{disk_log, Node, {error, disk_log_stopped}}\fR\&. 
.RE
.LP
.B
unblock(Log) -> ok | {error, Reason}
.br
.RS
.TP
Types
Log = term()
.br
Reason = no_such_log | nonode | {not_blocked, Log} | {not_blocked_by_pid, Log}
.br
.RE
.RS
.LP
The \fIunblock/1\fR function unblocks a log\&. A log can only be unblocked by the blocking process\&. 
.RE
.SH SEE ALSO
.LP
file(3), pg2(3), wrap_log_reader(3)
 |