File: LogMessageNotification%2BrepresentingStructureUsingEmojiPrefixIfNecessary.swift

package info (click to toggle)
swiftlang 6.2.3-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 2,856,264 kB
  • sloc: cpp: 9,995,718; ansic: 2,234,019; asm: 1,092,167; python: 313,940; objc: 82,726; f90: 80,126; lisp: 38,373; pascal: 25,580; sh: 20,378; ml: 5,058; perl: 4,751; makefile: 4,725; awk: 3,535; javascript: 3,018; xml: 918; fortran: 664; cs: 573; ruby: 396
file content (73 lines) | stat: -rw-r--r-- 2,807 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
//===----------------------------------------------------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2020 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 the list of Swift project authors
//
//===----------------------------------------------------------------------===//

import LanguageServerProtocol
import SKOptions

import struct TSCBasic.ProcessResult

private func numberToEmojis(_ number: Int, numEmojis: Int) -> String {
  let emojis = ["🟥", "🟩", "🟦", "⬜️", "🟪", "⬛️", "🟨", "🟫"]
  var number = abs(number)
  var result = ""
  for _ in 0..<numEmojis {
    let (quotient, remainder) = number.quotientAndRemainder(dividingBy: emojis.count)
    result += emojis[remainder]
    number = quotient
  }
  return result
}

fileprivate extension String {
  /// Returns a two-character emoji string that allows easy differentiation between different task IDs.
  ///
  /// This marker is prepended to every line in the index log.
  var emojiRepresentation: String {
    // Run .hashValue to make sure we semi-randomly pick new emoji markers for new tasks
    return numberToEmojis(self.hashValue, numEmojis: 3)
  }
}

/// Add an emoji hash of the given `taskID` to the start of every line in `message`.
private func prefixMessageWithTaskEmoji(taskID: String, message: String) -> String {
  var message: Substring = message[...]
  while message.last?.isNewline ?? false {
    message = message.dropLast(1)
  }
  let messageWithEmojiLinePrefixes = message.split(separator: "\n", omittingEmptySubsequences: false).map {
    "\(taskID.emojiRepresentation) \($0)"
  }.joined(separator: "\n")
  return messageWithEmojiLinePrefixes
}

extension LogMessageNotification {
  /// If the client does not support the experimental structured logs capability and the `LogMessageNotification`
  /// contains structure, clear it and instead prefix the message with a an emoji sequence that indicates the task it
  /// belongs to.
  func representingTaskIDUsingEmojiPrefixIfNecessary(
    options: SourceKitLSPOptions
  ) -> LogMessageNotification {
    if options.hasExperimentalFeature(.structuredLogs) {
      // Client supports structured logs, nothing to do.
      return self
    }
    var transformed = self
    if let structure = transformed.structure {
      if case .begin(let begin) = structure {
        transformed.message = "\(begin.title)\n\(message)"
      }
      transformed.message = prefixMessageWithTaskEmoji(taskID: structure.taskID, message: transformed.message)
      transformed.structure = nil
    }
    return transformed
  }
}