|
14 | 14 | @testable import Logging
|
15 | 15 | import XCTest
|
16 | 16 |
|
| 17 | +#if os(macOS) || os(iOS) || os(tvOS) || os(watchOS) |
| 18 | +import Darwin |
| 19 | +#else |
| 20 | +import Glibc |
| 21 | +#endif |
| 22 | + |
17 | 23 | class LoggingTest: XCTestCase {
|
18 | 24 | func testAutoclosure() throws {
|
19 | 25 | // bootstrap with our test logging impl
|
@@ -444,4 +450,66 @@ class LoggingTest: XCTestCase {
|
444 | 450 | XCTAssertTrue(messageSucceeded ?? false)
|
445 | 451 | XCTAssertEqual(interceptStream.strings.count, 1)
|
446 | 452 | }
|
| 453 | + |
| 454 | + func testStdioOutputStreamFlush() { |
| 455 | + // flush on every statement |
| 456 | + self.withWriteReadFDsAndReadBuffer { writeFD, readFD, readBuffer in |
| 457 | + let logStream = StdioOutputStream(file: writeFD, flushMode: .always) |
| 458 | + LoggingSystem.bootstrapInternal { StreamLogHandler(label: $0, stream: logStream) } |
| 459 | + Logger(label: "test").critical("test") |
| 460 | + |
| 461 | + let size = read(readFD, readBuffer, 256) |
| 462 | + XCTAssertGreaterThan(size, -1, "expected flush") |
| 463 | + |
| 464 | + logStream.flush() |
| 465 | + let size2 = read(readFD, readBuffer, 256) |
| 466 | + XCTAssertEqual(size2, -1, "expected no flush") |
| 467 | + } |
| 468 | + // default flushing |
| 469 | + self.withWriteReadFDsAndReadBuffer { writeFD, readFD, readBuffer in |
| 470 | + let logStream = StdioOutputStream(file: writeFD, flushMode: .undefined) |
| 471 | + LoggingSystem.bootstrapInternal { StreamLogHandler(label: $0, stream: logStream) } |
| 472 | + Logger(label: "test").critical("test") |
| 473 | + |
| 474 | + let size = read(readFD, readBuffer, 256) |
| 475 | + XCTAssertEqual(size, -1, "expected no flush") |
| 476 | + |
| 477 | + logStream.flush() |
| 478 | + let size2 = read(readFD, readBuffer, 256) |
| 479 | + XCTAssertGreaterThan(size2, -1, "expected flush") |
| 480 | + } |
| 481 | + } |
| 482 | + |
| 483 | + func withWriteReadFDsAndReadBuffer(_ body: (UnsafeMutablePointer<FILE>, CInt, UnsafeMutablePointer<Int8>) -> Void) { |
| 484 | + var fds: [Int32] = [-1, -1] |
| 485 | + fds.withUnsafeMutableBufferPointer { ptr in |
| 486 | + let err = pipe(ptr.baseAddress!) |
| 487 | + XCTAssertEqual(err, 0, "pipe faild \(err)") |
| 488 | + } |
| 489 | + |
| 490 | + let writeFD = fdopen(fds[1], "w") |
| 491 | + let writeBuffer = UnsafeMutablePointer<Int8>.allocate(capacity: 256) |
| 492 | + defer { |
| 493 | + writeBuffer.deinitialize(count: 256) |
| 494 | + writeBuffer.deallocate() |
| 495 | + } |
| 496 | + |
| 497 | + var err = setvbuf(writeFD, writeBuffer, _IOFBF, 256) |
| 498 | + XCTAssertEqual(err, 0, "setvbuf faild \(err)") |
| 499 | + |
| 500 | + let readFD = fds[0] |
| 501 | + err = fcntl(readFD, F_SETFL, fcntl(readFD, F_GETFL) | O_NONBLOCK) |
| 502 | + XCTAssertEqual(err, 0, "fcntl faild \(err)") |
| 503 | + |
| 504 | + let readBuffer = UnsafeMutablePointer<Int8>.allocate(capacity: 256) |
| 505 | + defer { |
| 506 | + readBuffer.deinitialize(count: 256) |
| 507 | + readBuffer.deallocate() |
| 508 | + } |
| 509 | + |
| 510 | + // the actual test |
| 511 | + body(writeFD!, readFD, readBuffer) |
| 512 | + |
| 513 | + fds.forEach { close($0) } |
| 514 | + } |
447 | 515 | }
|
0 commit comments