OLD | NEW |
(Empty) | |
| 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file |
| 2 // for details. All rights reserved. Use of this source code is governed by a |
| 3 // BSD-style license that can be found in the LICENSE file. |
| 4 |
| 5 library logging_test; |
| 6 |
| 7 import 'dart:async'; |
| 8 |
| 9 import 'package:logging/logging.dart'; |
| 10 import 'package:test/test.dart'; |
| 11 |
| 12 void main() { |
| 13 test('level comparison is a valid comparator', () { |
| 14 var level1 = const Level('NOT_REAL1', 253); |
| 15 expect(level1 == level1, isTrue); |
| 16 expect(level1 <= level1, isTrue); |
| 17 expect(level1 >= level1, isTrue); |
| 18 expect(level1 < level1, isFalse); |
| 19 expect(level1 > level1, isFalse); |
| 20 |
| 21 var level2 = const Level('NOT_REAL2', 455); |
| 22 expect(level1 <= level2, isTrue); |
| 23 expect(level1 < level2, isTrue); |
| 24 expect(level2 >= level1, isTrue); |
| 25 expect(level2 > level1, isTrue); |
| 26 |
| 27 var level3 = const Level('NOT_REAL3', 253); |
| 28 expect(level1, isNot(same(level3))); // different instances |
| 29 expect(level1, equals(level3)); // same value. |
| 30 }); |
| 31 |
| 32 test('default levels are in order', () { |
| 33 final levels = Level.LEVELS; |
| 34 |
| 35 for (int i = 0; i < levels.length; i++) { |
| 36 for (int j = i + 1; j < levels.length; j++) { |
| 37 expect(levels[i] < levels[j], isTrue); |
| 38 } |
| 39 } |
| 40 }); |
| 41 |
| 42 test('levels are comparable', () { |
| 43 final unsorted = [ |
| 44 Level.INFO, |
| 45 Level.CONFIG, |
| 46 Level.FINE, |
| 47 Level.SHOUT, |
| 48 Level.OFF, |
| 49 Level.FINER, |
| 50 Level.ALL, |
| 51 Level.WARNING, |
| 52 Level.FINEST, |
| 53 Level.SEVERE, |
| 54 ]; |
| 55 |
| 56 final sorted = Level.LEVELS; |
| 57 |
| 58 expect(unsorted, isNot(orderedEquals(sorted))); |
| 59 |
| 60 unsorted.sort(); |
| 61 expect(unsorted, orderedEquals(sorted)); |
| 62 }); |
| 63 |
| 64 test('levels are hashable', () { |
| 65 var map = new Map<Level, String>(); |
| 66 map[Level.INFO] = 'info'; |
| 67 map[Level.SHOUT] = 'shout'; |
| 68 expect(map[Level.INFO], same('info')); |
| 69 expect(map[Level.SHOUT], same('shout')); |
| 70 }); |
| 71 |
| 72 test('logger name cannot start with a "." ', () { |
| 73 expect(() => new Logger('.c'), throwsArgumentError); |
| 74 }); |
| 75 |
| 76 test('logger naming is hierarchical', () { |
| 77 Logger c = new Logger('a.b.c'); |
| 78 expect(c.name, equals('c')); |
| 79 expect(c.parent.name, equals('b')); |
| 80 expect(c.parent.parent.name, equals('a')); |
| 81 expect(c.parent.parent.parent.name, equals('')); |
| 82 expect(c.parent.parent.parent.parent, isNull); |
| 83 }); |
| 84 |
| 85 test('logger full name', () { |
| 86 Logger c = new Logger('a.b.c'); |
| 87 expect(c.fullName, equals('a.b.c')); |
| 88 expect(c.parent.fullName, equals('a.b')); |
| 89 expect(c.parent.parent.fullName, equals('a')); |
| 90 expect(c.parent.parent.parent.fullName, equals('')); |
| 91 expect(c.parent.parent.parent.parent, isNull); |
| 92 }); |
| 93 |
| 94 test('logger parent-child links are correct', () { |
| 95 Logger a = new Logger('a'); |
| 96 Logger b = new Logger('a.b'); |
| 97 Logger c = new Logger('a.c'); |
| 98 expect(a, same(b.parent)); |
| 99 expect(a, same(c.parent)); |
| 100 expect(a.children['b'], same(b)); |
| 101 expect(a.children['c'], same(c)); |
| 102 }); |
| 103 |
| 104 test('loggers are singletons', () { |
| 105 Logger a1 = new Logger('a'); |
| 106 Logger a2 = new Logger('a'); |
| 107 Logger b = new Logger('a.b'); |
| 108 Logger root = Logger.root; |
| 109 expect(a1, same(a2)); |
| 110 expect(a1, same(b.parent)); |
| 111 expect(root, same(a1.parent)); |
| 112 expect(root, same(new Logger(''))); |
| 113 }); |
| 114 |
| 115 test('cannot directly manipulate Logger.children', () { |
| 116 var loggerAB = new Logger('a.b'); |
| 117 var loggerA = loggerAB.parent; |
| 118 |
| 119 expect(loggerA.children['b'], same(loggerAB), reason: 'can read Children'); |
| 120 |
| 121 expect(() { |
| 122 loggerAB.children['test'] = null; |
| 123 }, throwsUnsupportedError, reason: 'Children is read-only'); |
| 124 }); |
| 125 |
| 126 test('stackTrace gets throw to LogRecord', () { |
| 127 Logger.root.level = Level.INFO; |
| 128 |
| 129 var records = new List<LogRecord>(); |
| 130 |
| 131 var sub = Logger.root.onRecord.listen(records.add); |
| 132 |
| 133 try { |
| 134 throw new UnsupportedError('test exception'); |
| 135 } catch (error, stack) { |
| 136 Logger.root.log(Level.SEVERE, 'severe', error, stack); |
| 137 Logger.root.warning('warning', error, stack); |
| 138 } |
| 139 |
| 140 Logger.root.log(Level.SHOUT, 'shout'); |
| 141 |
| 142 sub.cancel(); |
| 143 |
| 144 expect(records, hasLength(3)); |
| 145 |
| 146 var severe = records[0]; |
| 147 expect(severe.message, 'severe'); |
| 148 expect(severe.error is UnsupportedError, isTrue); |
| 149 expect(severe.stackTrace is StackTrace, isTrue); |
| 150 |
| 151 var warning = records[1]; |
| 152 expect(warning.message, 'warning'); |
| 153 expect(warning.error is UnsupportedError, isTrue); |
| 154 expect(warning.stackTrace is StackTrace, isTrue); |
| 155 |
| 156 var shout = records[2]; |
| 157 expect(shout.message, 'shout'); |
| 158 expect(shout.error, isNull); |
| 159 expect(shout.stackTrace, isNull); |
| 160 }); |
| 161 |
| 162 group('zone gets recorded to LogRecord', () { |
| 163 test('root zone', () { |
| 164 var root = Logger.root; |
| 165 |
| 166 var recordingZone = Zone.current; |
| 167 var records = new List<LogRecord>(); |
| 168 root.onRecord.listen(records.add); |
| 169 root.info('hello'); |
| 170 |
| 171 expect(records, hasLength(1)); |
| 172 expect(records.first.zone, equals(recordingZone)); |
| 173 }); |
| 174 |
| 175 test('child zone', () { |
| 176 var root = Logger.root; |
| 177 |
| 178 var recordingZone; |
| 179 var records = new List<LogRecord>(); |
| 180 root.onRecord.listen(records.add); |
| 181 |
| 182 runZoned(() { |
| 183 recordingZone = Zone.current; |
| 184 root.info('hello'); |
| 185 }); |
| 186 |
| 187 expect(records, hasLength(1)); |
| 188 expect(records.first.zone, equals(recordingZone)); |
| 189 }); |
| 190 |
| 191 test('custom zone', () { |
| 192 var root = Logger.root; |
| 193 |
| 194 var recordingZone; |
| 195 var records = new List<LogRecord>(); |
| 196 root.onRecord.listen(records.add); |
| 197 |
| 198 runZoned(() { |
| 199 recordingZone = Zone.current; |
| 200 }); |
| 201 |
| 202 runZoned(() => root.log(Level.INFO, 'hello', null, null, recordingZone)); |
| 203 |
| 204 expect(records, hasLength(1)); |
| 205 expect(records.first.zone, equals(recordingZone)); |
| 206 }); |
| 207 }); |
| 208 |
| 209 group('detached loggers', () { |
| 210 test("create new instances of Logger", () { |
| 211 Logger a1 = new Logger.detached("a"); |
| 212 Logger a2 = new Logger.detached("a"); |
| 213 Logger a = new Logger("a"); |
| 214 |
| 215 expect(a1, isNot(a2)); |
| 216 expect(a1, isNot(a)); |
| 217 expect(a2, isNot(a)); |
| 218 }); |
| 219 |
| 220 test("parent is null", () { |
| 221 Logger a = new Logger.detached("a"); |
| 222 expect(a.parent, null); |
| 223 }); |
| 224 |
| 225 test("children is empty", () { |
| 226 Logger a = new Logger.detached("a"); |
| 227 expect(a.children, {}); |
| 228 }); |
| 229 }); |
| 230 |
| 231 group('mutating levels', () { |
| 232 Logger root = Logger.root; |
| 233 Logger a = new Logger('a'); |
| 234 Logger b = new Logger('a.b'); |
| 235 Logger c = new Logger('a.b.c'); |
| 236 Logger d = new Logger('a.b.c.d'); |
| 237 Logger e = new Logger('a.b.c.d.e'); |
| 238 |
| 239 setUp(() { |
| 240 hierarchicalLoggingEnabled = true; |
| 241 root.level = Level.INFO; |
| 242 a.level = null; |
| 243 b.level = null; |
| 244 c.level = null; |
| 245 d.level = null; |
| 246 e.level = null; |
| 247 root.clearListeners(); |
| 248 a.clearListeners(); |
| 249 b.clearListeners(); |
| 250 c.clearListeners(); |
| 251 d.clearListeners(); |
| 252 e.clearListeners(); |
| 253 hierarchicalLoggingEnabled = false; |
| 254 root.level = Level.INFO; |
| 255 }); |
| 256 |
| 257 test('cannot set level if hierarchy is disabled', () { |
| 258 expect(() { |
| 259 a.level = Level.FINE; |
| 260 }, throwsUnsupportedError); |
| 261 }); |
| 262 |
| 263 test('loggers effective level - no hierarchy', () { |
| 264 expect(root.level, equals(Level.INFO)); |
| 265 expect(a.level, equals(Level.INFO)); |
| 266 expect(b.level, equals(Level.INFO)); |
| 267 |
| 268 root.level = Level.SHOUT; |
| 269 |
| 270 expect(root.level, equals(Level.SHOUT)); |
| 271 expect(a.level, equals(Level.SHOUT)); |
| 272 expect(b.level, equals(Level.SHOUT)); |
| 273 }); |
| 274 |
| 275 test('loggers effective level - with hierarchy', () { |
| 276 hierarchicalLoggingEnabled = true; |
| 277 expect(root.level, equals(Level.INFO)); |
| 278 expect(a.level, equals(Level.INFO)); |
| 279 expect(b.level, equals(Level.INFO)); |
| 280 expect(c.level, equals(Level.INFO)); |
| 281 |
| 282 root.level = Level.SHOUT; |
| 283 b.level = Level.FINE; |
| 284 |
| 285 expect(root.level, equals(Level.SHOUT)); |
| 286 expect(a.level, equals(Level.SHOUT)); |
| 287 expect(b.level, equals(Level.FINE)); |
| 288 expect(c.level, equals(Level.FINE)); |
| 289 }); |
| 290 |
| 291 test('isLoggable is appropriate', () { |
| 292 hierarchicalLoggingEnabled = true; |
| 293 root.level = Level.SEVERE; |
| 294 c.level = Level.ALL; |
| 295 e.level = Level.OFF; |
| 296 |
| 297 expect(root.isLoggable(Level.SHOUT), isTrue); |
| 298 expect(root.isLoggable(Level.SEVERE), isTrue); |
| 299 expect(root.isLoggable(Level.WARNING), isFalse); |
| 300 expect(c.isLoggable(Level.FINEST), isTrue); |
| 301 expect(c.isLoggable(Level.FINE), isTrue); |
| 302 expect(e.isLoggable(Level.SHOUT), isFalse); |
| 303 }); |
| 304 |
| 305 test('add/remove handlers - no hierarchy', () { |
| 306 int calls = 0; |
| 307 var handler = (_) { |
| 308 calls++; |
| 309 }; |
| 310 final sub = c.onRecord.listen(handler); |
| 311 root.info("foo"); |
| 312 root.info("foo"); |
| 313 expect(calls, equals(2)); |
| 314 sub.cancel(); |
| 315 root.info("foo"); |
| 316 expect(calls, equals(2)); |
| 317 }); |
| 318 |
| 319 test('add/remove handlers - with hierarchy', () { |
| 320 hierarchicalLoggingEnabled = true; |
| 321 int calls = 0; |
| 322 var handler = (_) { |
| 323 calls++; |
| 324 }; |
| 325 c.onRecord.listen(handler); |
| 326 root.info("foo"); |
| 327 root.info("foo"); |
| 328 expect(calls, equals(0)); |
| 329 }); |
| 330 |
| 331 test('logging methods store appropriate level', () { |
| 332 root.level = Level.ALL; |
| 333 var rootMessages = []; |
| 334 root.onRecord.listen((record) { |
| 335 rootMessages.add('${record.level}: ${record.message}'); |
| 336 }); |
| 337 |
| 338 root.finest('1'); |
| 339 root.finer('2'); |
| 340 root.fine('3'); |
| 341 root.config('4'); |
| 342 root.info('5'); |
| 343 root.warning('6'); |
| 344 root.severe('7'); |
| 345 root.shout('8'); |
| 346 |
| 347 expect(rootMessages, equals([ |
| 348 'FINEST: 1', |
| 349 'FINER: 2', |
| 350 'FINE: 3', |
| 351 'CONFIG: 4', |
| 352 'INFO: 5', |
| 353 'WARNING: 6', |
| 354 'SEVERE: 7', |
| 355 'SHOUT: 8' |
| 356 ])); |
| 357 }); |
| 358 |
| 359 test('logging methods store exception', () { |
| 360 root.level = Level.ALL; |
| 361 var rootMessages = []; |
| 362 root.onRecord.listen((r) { |
| 363 rootMessages.add('${r.level}: ${r.message} ${r.error}'); |
| 364 }); |
| 365 |
| 366 root.finest('1'); |
| 367 root.finer('2'); |
| 368 root.fine('3'); |
| 369 root.config('4'); |
| 370 root.info('5'); |
| 371 root.warning('6'); |
| 372 root.severe('7'); |
| 373 root.shout('8'); |
| 374 root.finest('1', 'a'); |
| 375 root.finer('2', 'b'); |
| 376 root.fine('3', ['c']); |
| 377 root.config('4', 'd'); |
| 378 root.info('5', 'e'); |
| 379 root.warning('6', 'f'); |
| 380 root.severe('7', 'g'); |
| 381 root.shout('8', 'h'); |
| 382 |
| 383 expect(rootMessages, equals([ |
| 384 'FINEST: 1 null', |
| 385 'FINER: 2 null', |
| 386 'FINE: 3 null', |
| 387 'CONFIG: 4 null', |
| 388 'INFO: 5 null', |
| 389 'WARNING: 6 null', |
| 390 'SEVERE: 7 null', |
| 391 'SHOUT: 8 null', |
| 392 'FINEST: 1 a', |
| 393 'FINER: 2 b', |
| 394 'FINE: 3 [c]', |
| 395 'CONFIG: 4 d', |
| 396 'INFO: 5 e', |
| 397 'WARNING: 6 f', |
| 398 'SEVERE: 7 g', |
| 399 'SHOUT: 8 h' |
| 400 ])); |
| 401 }); |
| 402 |
| 403 test('message logging - no hierarchy', () { |
| 404 root.level = Level.WARNING; |
| 405 var rootMessages = []; |
| 406 var aMessages = []; |
| 407 var cMessages = []; |
| 408 c.onRecord.listen((record) { |
| 409 cMessages.add('${record.level}: ${record.message}'); |
| 410 }); |
| 411 a.onRecord.listen((record) { |
| 412 aMessages.add('${record.level}: ${record.message}'); |
| 413 }); |
| 414 root.onRecord.listen((record) { |
| 415 rootMessages.add('${record.level}: ${record.message}'); |
| 416 }); |
| 417 |
| 418 root.info('1'); |
| 419 root.fine('2'); |
| 420 root.shout('3'); |
| 421 |
| 422 b.info('4'); |
| 423 b.severe('5'); |
| 424 b.warning('6'); |
| 425 b.fine('7'); |
| 426 |
| 427 c.fine('8'); |
| 428 c.warning('9'); |
| 429 c.shout('10'); |
| 430 |
| 431 expect(rootMessages, equals([ |
| 432 // 'INFO: 1' is not loggable |
| 433 // 'FINE: 2' is not loggable |
| 434 'SHOUT: 3', |
| 435 // 'INFO: 4' is not loggable |
| 436 'SEVERE: 5', |
| 437 'WARNING: 6', |
| 438 // 'FINE: 7' is not loggable |
| 439 // 'FINE: 8' is not loggable |
| 440 'WARNING: 9', |
| 441 'SHOUT: 10' |
| 442 ])); |
| 443 |
| 444 // no hierarchy means we all hear the same thing. |
| 445 expect(aMessages, equals(rootMessages)); |
| 446 expect(cMessages, equals(rootMessages)); |
| 447 }); |
| 448 |
| 449 test('message logging - with hierarchy', () { |
| 450 hierarchicalLoggingEnabled = true; |
| 451 |
| 452 b.level = Level.WARNING; |
| 453 |
| 454 var rootMessages = []; |
| 455 var aMessages = []; |
| 456 var cMessages = []; |
| 457 c.onRecord.listen((record) { |
| 458 cMessages.add('${record.level}: ${record.message}'); |
| 459 }); |
| 460 a.onRecord.listen((record) { |
| 461 aMessages.add('${record.level}: ${record.message}'); |
| 462 }); |
| 463 root.onRecord.listen((record) { |
| 464 rootMessages.add('${record.level}: ${record.message}'); |
| 465 }); |
| 466 |
| 467 root.info('1'); |
| 468 root.fine('2'); |
| 469 root.shout('3'); |
| 470 |
| 471 b.info('4'); |
| 472 b.severe('5'); |
| 473 b.warning('6'); |
| 474 b.fine('7'); |
| 475 |
| 476 c.fine('8'); |
| 477 c.warning('9'); |
| 478 c.shout('10'); |
| 479 |
| 480 expect(rootMessages, equals([ |
| 481 'INFO: 1', |
| 482 // 'FINE: 2' is not loggable |
| 483 'SHOUT: 3', |
| 484 // 'INFO: 4' is not loggable |
| 485 'SEVERE: 5', |
| 486 'WARNING: 6', |
| 487 // 'FINE: 7' is not loggable |
| 488 // 'FINE: 8' is not loggable |
| 489 'WARNING: 9', |
| 490 'SHOUT: 10' |
| 491 ])); |
| 492 |
| 493 expect(aMessages, equals([ |
| 494 // 1,2 and 3 are lower in the hierarchy |
| 495 // 'INFO: 4' is not loggable |
| 496 'SEVERE: 5', |
| 497 'WARNING: 6', |
| 498 // 'FINE: 7' is not loggable |
| 499 // 'FINE: 8' is not loggable |
| 500 'WARNING: 9', |
| 501 'SHOUT: 10' |
| 502 ])); |
| 503 |
| 504 expect(cMessages, equals([ |
| 505 // 1 - 7 are lower in the hierarchy |
| 506 // 'FINE: 8' is not loggable |
| 507 'WARNING: 9', |
| 508 'SHOUT: 10' |
| 509 ])); |
| 510 }); |
| 511 |
| 512 test('message logging - lazy functions', () { |
| 513 root.level = Level.INFO; |
| 514 var messages = []; |
| 515 root.onRecord.listen((record) { |
| 516 messages.add('${record.level}: ${record.message}'); |
| 517 }); |
| 518 |
| 519 var callCount = 0; |
| 520 var myClosure = () => "${++callCount}"; |
| 521 |
| 522 root.info(myClosure); |
| 523 root.finer(myClosure); // Should not get evaluated. |
| 524 root.warning(myClosure); |
| 525 |
| 526 expect(messages, equals(['INFO: 1', 'WARNING: 2',])); |
| 527 }); |
| 528 |
| 529 test('message logging - calls toString', () { |
| 530 root.level = Level.INFO; |
| 531 var messages = []; |
| 532 root.onRecord.listen((record) { |
| 533 messages.add('${record.level}: ${record.message}'); |
| 534 }); |
| 535 |
| 536 root.info(5); |
| 537 root.info(false); |
| 538 root.info([1, 2, 3]); |
| 539 root.info(() => 10); |
| 540 |
| 541 expect(messages, |
| 542 equals(['INFO: 5', 'INFO: false', 'INFO: [1, 2, 3]', 'INFO: 10',])); |
| 543 }); |
| 544 }); |
| 545 |
| 546 group('recordStackTraceAtLevel', () { |
| 547 var root = Logger.root; |
| 548 tearDown(() { |
| 549 recordStackTraceAtLevel = Level.OFF; |
| 550 root.clearListeners(); |
| 551 }); |
| 552 |
| 553 test('no stack trace by default', () { |
| 554 var records = new List<LogRecord>(); |
| 555 root.onRecord.listen(records.add); |
| 556 root.severe('hello'); |
| 557 root.warning('hello'); |
| 558 root.info('hello'); |
| 559 expect(records, hasLength(3)); |
| 560 expect(records[0].stackTrace, isNull); |
| 561 expect(records[1].stackTrace, isNull); |
| 562 expect(records[2].stackTrace, isNull); |
| 563 }); |
| 564 |
| 565 test('trace recorded only on requested levels', () { |
| 566 var records = new List<LogRecord>(); |
| 567 recordStackTraceAtLevel = Level.WARNING; |
| 568 root.onRecord.listen(records.add); |
| 569 root.severe('hello'); |
| 570 root.warning('hello'); |
| 571 root.info('hello'); |
| 572 expect(records, hasLength(3)); |
| 573 expect(records[0].stackTrace, isNotNull); |
| 574 expect(records[1].stackTrace, isNotNull); |
| 575 expect(records[2].stackTrace, isNull); |
| 576 }); |
| 577 |
| 578 test('provided trace is used if given', () { |
| 579 var trace; |
| 580 try { |
| 581 throw 'trace'; |
| 582 } catch(e, t) { |
| 583 trace = t; |
| 584 } |
| 585 var records = new List<LogRecord>(); |
| 586 recordStackTraceAtLevel = Level.WARNING; |
| 587 root.onRecord.listen(records.add); |
| 588 root.severe('hello'); |
| 589 root.warning('hello', 'a', trace); |
| 590 expect(records, hasLength(2)); |
| 591 expect(records[0].stackTrace, isNot(equals(trace))); |
| 592 expect(records[1].stackTrace, trace); |
| 593 }); |
| 594 |
| 595 test('error also generated when generating a trace', () { |
| 596 var records = new List<LogRecord>(); |
| 597 recordStackTraceAtLevel = Level.WARNING; |
| 598 root.onRecord.listen(records.add); |
| 599 root.severe('hello'); |
| 600 root.warning('hello'); |
| 601 root.info('hello'); |
| 602 expect(records, hasLength(3)); |
| 603 expect(records[0].error, isNotNull); |
| 604 expect(records[1].error, isNotNull); |
| 605 expect(records[2].error, isNull); |
| 606 }); |
| 607 }); |
| 608 } |
OLD | NEW |