File: PreviewHTTPHandler.swift

package info (click to toggle)
swiftlang 6.0.3-2
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 2,519,992 kB
  • sloc: cpp: 9,107,863; ansic: 2,040,022; asm: 1,135,751; python: 296,500; objc: 82,456; f90: 60,502; lisp: 34,951; pascal: 19,946; sh: 18,133; perl: 7,482; ml: 4,937; javascript: 4,117; makefile: 3,840; awk: 3,535; xml: 914; fortran: 619; cs: 573; ruby: 573
file content (135 lines) | stat: -rw-r--r-- 5,391 bytes parent folder | download
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
/*
 This source file is part of the Swift.org open source project

 Copyright (c) 2021 Apple Inc. and the Swift project authors
 Licensed under Apache License v2.0 with Runtime Library Exception

 See https://swift.org/LICENSE.txt for license information
 See https://swift.org/CONTRIBUTORS.txt for Swift project authors
*/

#if canImport(NIOHTTP1)
import Foundation
import NIO
import NIOHTTP1

/// An HTTP request handler that serves file assets and the default app page.
///
/// A custom HTTP handler that's able to serve compiled documentation
/// via a web server bound to a port on the local machine or a socket.
///
/// ### Features
/// - Serving static files from pre-defined assets directory paths
/// - A catch-all default GET response handler for all non-asset requests
/// - Ignores unknown requests
///
/// ### Life cycle
/// The handler is a simple state machine alternating between 'idle' and 'requestInProgress' states
/// while going through the following cycle:
///
/// 1. ``state`` is `.idle`
/// 2. ``channelRead(context:data:)`` is called with HTTP request `head` context
/// 3. ``state`` is set to `.requestInProgress(head, handler)`
/// 4. ``channelRead(context:data:)`` is called with HTTP request `end` context
/// 5. ``state`` is set to `.idle`
/// 6. response data is flushed to the client (go to 1)
final class PreviewHTTPHandler: ChannelInboundHandler {
    /// The handler's expected input data format.
    public typealias InboundIn = HTTPServerRequestPart
    
    /// The handler's expected output data format.
    public typealias OutboundOut = HTTPServerResponsePart
    
    /// The current handler's request state.
    private enum State {
        case idle, requestInProgress(requestHead: HTTPRequestHead, handler: RequestHandler)
    }
    
    // MARK: - Properties
    private var state: State = .idle
    
    private var keepAlive = false
    private let rootURL: URL

    private var handlerFuture: EventLoopFuture<Void>?
    private let fileIO: NonBlockingFileIO

    /// - Parameters:
    ///   - fileIO: Async file I/O.
    ///   - rootURL: The root of the content directory to serve.
    ///   - credentials: Optional user credentials to authorize incoming requests.
    init(fileIO: NonBlockingFileIO, rootURL: URL) {
        self.rootURL = rootURL
        self.fileIO = fileIO
    }
    
    /// Handles incoming data on a channel.
    ///
    /// When receiving a request's head this method prepares the correct handler
    /// for the requested resource.
    ///
    /// - Parameters:
    ///   - context: A channel context.
    ///   - data: The current inbound request data.
    func channelRead(context: ChannelHandlerContext, data: NIOAny) {
        let requestPart = unwrapInboundIn(data)
        
        switch (requestPart, state) {
        case (.head(let head), _):
            let handler: RequestHandlerFactory
            if FileRequestHandler.isAssetPath(head.uri) {
                // Serve a static asset file.
                handler = FileRequestHandler(rootURL: rootURL, fileIO: fileIO)
            } else {
                // Serve the fallback index file.
                handler = DefaultRequestHandler(rootURL: rootURL)
            }
            state = .requestInProgress(requestHead: head, handler: handler.create(channelHandler: self))
            
        case (.end, .requestInProgress(let head, let handler)):
            defer {
                // Complete the response to the client, reset ``state``
                completeResponse(context, trailers: nil, promise: nil)
            }
            
            // Call the pre-defined during the `head` context handler.
            do {
                try handler(context, head)
            } catch {
                let errorHandler = ErrorRequestHandler(error: error as? RequestError)
                    .create(channelHandler: self)
                
                // The error handler will never throw.
                try! errorHandler(context, head)
            }
            
        // Ignore other parts of a request, e.g. POST data or others.
        default: break
        }
    }

    /// Complete the current response and flush the buffer to the client.
    private func completeResponse(_ context: ChannelHandlerContext, trailers: HTTPHeaders?, promise: EventLoopPromise<Void>?) {
        guard case State.requestInProgress = state else { return }
        state = .idle
        
        let promise = promise ?? context.eventLoop.makePromise()
        
        // If we don't need to keep the connection alive, close `context` after flushing the response
        if !self.keepAlive {
            promise.futureResult.whenComplete { _ in context.close(promise: nil) }
        }

        context.writeAndFlush(self.wrapOutboundOut(.end(trailers)), promise: promise)
    }
    
    /// Replaces the current in-progress response with an error response and flushes the output to the client.
    private func error(context: ChannelHandlerContext, requestPart: PreviewHTTPHandler.InboundIn, head: HTTPRequestHead, status: HTTPResponseStatus, headers: [(String, String)] = []) {
        let errorHandler = ErrorRequestHandler(error: RequestError(status: status), headers: headers)
            .create(channelHandler: self)
        
        try! errorHandler(context, head)
        completeResponse(context, trailers: nil, promise: nil)
    }
}
#endif