| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  | ! Copyright (C) 2008, 2010 Slava Pestov. | 
					
						
							|  |  |  | ! See http://factorcode.org/license.txt for BSD license. | 
					
						
							|  |  |  | USING: accessors peg peg.parsers memoize kernel sequences | 
					
						
							|  |  |  | logging arrays words strings vectors io io.files | 
					
						
							|  |  |  | io.encodings.utf8 namespaces make combinators logging.server | 
					
						
							|  |  |  | calendar calendar.format assocs prettyprint ;
 | 
					
						
							|  |  |  | IN: logging.parser | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TUPLE: log-entry date level word-name message ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : string-of ( quot -- parser )
 | 
					
						
							|  |  |  |     satisfy repeat0 [ >string ] action ; inline
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | SYMBOL: multiline | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-15 21:10:13 -04:00
										 |  |  | : date-parser ( -- parser )
 | 
					
						
							| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  |     [ "]" member? not ] string-of [ | 
					
						
							|  |  |  |         dup multiline-header =
 | 
					
						
							|  |  |  |         [ drop multiline ] [ rfc3339>timestamp ] if
 | 
					
						
							|  |  |  |     ] action | 
					
						
							|  |  |  |     "[" "]" surrounded-by ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-15 21:10:13 -04:00
										 |  |  | : log-level-parser ( -- parser )
 | 
					
						
							| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  |     log-levels keys [ | 
					
						
							|  |  |  |         [ name>> token ] keep [ nip ] curry action | 
					
						
							|  |  |  |     ] map choice ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-15 21:10:13 -04:00
										 |  |  | : word-name-parser ( -- parser )
 | 
					
						
							| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  |     [ " :" member? not ] string-of ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | SYMBOL: malformed | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-15 21:10:13 -04:00
										 |  |  | : malformed-line-parser ( -- parser )
 | 
					
						
							| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  |     [ drop t ] string-of | 
					
						
							|  |  |  |     [ log-entry new swap >>message malformed >>level ] action ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-15 21:10:13 -04:00
										 |  |  | : log-message-parser ( -- parser )
 | 
					
						
							| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  |     [ drop t ] string-of | 
					
						
							|  |  |  |     [ 1vector ] action ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-15 21:10:13 -04:00
										 |  |  | : log-line-parser ( -- parser )
 | 
					
						
							| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  |     [ | 
					
						
							| 
									
										
										
										
											2015-08-15 21:10:13 -04:00
										 |  |  |         date-parser , | 
					
						
							| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  |         " " token hide , | 
					
						
							| 
									
										
										
										
											2015-08-15 21:10:13 -04:00
										 |  |  |         log-level-parser , | 
					
						
							| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  |         " " token hide , | 
					
						
							| 
									
										
										
										
											2015-08-15 21:10:13 -04:00
										 |  |  |         word-name-parser , | 
					
						
							| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  |         ": " token hide , | 
					
						
							| 
									
										
										
										
											2015-08-15 21:10:13 -04:00
										 |  |  |         log-message-parser , | 
					
						
							| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  |     ] seq* [ first4 log-entry boa ] action | 
					
						
							| 
									
										
										
										
											2015-08-15 21:10:13 -04:00
										 |  |  |     malformed-line-parser 2choice ;
 | 
					
						
							| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-15 21:10:13 -04:00
										 |  |  | PEG: parse-log-line ( string -- entry ) log-line-parser ;
 | 
					
						
							| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | : malformed? ( line -- ? )
 | 
					
						
							|  |  |  |     level>> malformed eq? ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : multiline? ( line -- ? )
 | 
					
						
							|  |  |  |     level>> multiline eq? ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : malformed-line ( line -- )
 | 
					
						
							|  |  |  |     "Warning: malformed log line:" print
 | 
					
						
							|  |  |  |     message>> print ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : add-multiline ( line -- )
 | 
					
						
							|  |  |  |     building get empty? [ | 
					
						
							|  |  |  |         "Warning: log begins with multiline entry" print drop
 | 
					
						
							|  |  |  |     ] [ | 
					
						
							|  |  |  |         message>> first building get last message>> push
 | 
					
						
							|  |  |  |     ] if ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : parse-log ( lines -- entries )
 | 
					
						
							|  |  |  |     [ | 
					
						
							|  |  |  |         [ | 
					
						
							|  |  |  |             parse-log-line { | 
					
						
							|  |  |  |                 { [ dup malformed? ] [ malformed-line ] } | 
					
						
							|  |  |  |                 { [ dup multiline? ] [ add-multiline ] } | 
					
						
							|  |  |  |                 [ , ] | 
					
						
							|  |  |  |             } cond
 | 
					
						
							|  |  |  |         ] each
 | 
					
						
							|  |  |  |     ] { } make ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : parse-log-file ( service -- entries )
 | 
					
						
							|  |  |  |     log-path 1 log# dup exists? | 
					
						
							|  |  |  |     [ utf8 file-lines parse-log ] [ drop f ] if ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GENERIC: log-timestamp. ( date -- )
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: timestamp log-timestamp. (timestamp>string) ;
 | 
					
						
							|  |  |  | M: word log-timestamp. drop "multiline" write ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : log-entry. ( entry -- )
 | 
					
						
							|  |  |  |     "====== " write
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         [ date>> log-timestamp. bl ] | 
					
						
							|  |  |  |         [ level>> pprint bl ] | 
					
						
							|  |  |  |         [ word-name>> write nl ] | 
					
						
							|  |  |  |         [ message>> "\n" join print ] | 
					
						
							|  |  |  |     } cleave ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : log-entries. ( errors -- )
 | 
					
						
							|  |  |  |     [ log-entry. ] each ;
 |