/*
Copyright (c) 2025 WuJingrun(吴京润)
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package f_log
import std.collection.ArrayList
import std.env
import std.time.DateTime
import stdx.log.{LogLevel, LogWriter}
import f_base.OS
import f_version.*
public class LogPattern <: LogPart & Hashable & Equatable<LogPattern> {
private static let appLogPart = AppLogPart()
private static let appVersionLogPart = AppVersionLogPart()
private static let levelLogPart = LevelLogPart()
private static let nameLogPart = NameLogPart()
private static let messageLogPart = MessageLogPart()
private static let threadIdLogPart = ThreadIdLogPart()
private static let threadNameLogPart = ThreadNameLogPart()
private static let processIdLogPart = ProcessIdLogPart()
private static let fountainVersionLogPart = FountainVersionLogPart()
public static let default = LogPattern("[%level-%name] %d{yyyy/MM/dd,HH:mm:ss.SSS}|%m") //pattern共44个字符
private let patterns: ArrayList<LogPart>
private let pattern: String
public init(pattern: String) {
this.pattern = pattern
patterns = compile(pattern)
}
public operator func ==(that: LogPattern): Bool {
this.pattern == that.pattern
}
public func hashCode(): Int64 {
pattern.hashCode()
}
private static func compile(pattern: String): ArrayList<LogPart> {
let patterns = ArrayList<LogPart>()
let origin = ArrayList<Byte>()
func appendOrigin() {
if (origin.size > 0) {
patterns.add(OriginLogPart(String.fromUtf8(origin.toArray())))
origin.clear()
}
}
var i = 0
let size = pattern.size
while (i < size) {
let c = pattern[i]
if (c == b'%') {
if (i + 5 < size && pattern[i + 1..=i + 5] == "level") {
appendOrigin()
patterns.add(levelLogPart)
i += 6
} else if (i + 3 < size && pattern[i + 1..=i + 3] == "app") {
appendOrigin()
patterns.add(appLogPart)
i += 4
} else if (i + 6 < size && pattern[i + 1..=i + 6] == "appver") {
appendOrigin()
patterns.add(appVersionLogPart)
i += 7
} else if (i + 4 < size && pattern[i + 1..=i + 4] == "name") {
appendOrigin()
patterns.add(nameLogPart)
i += 5
} else if (i + 4 < size && pattern[i + 1..=i + 4] == "fver") {
appendOrigin()
patterns.add(fountainVersionLogPart)
i += 5
} else if (i + 5 < size && pattern[i + 1..=i + 5] == "tname") {
appendOrigin()
patterns.add(threadNameLogPart)
i += 6
} else if (i + 3 < size && pattern[i + 1 ..= i + 3] == 'tid') {
appendOrigin()
patterns.add(threadIdLogPart)
i += 4
} else if (i + 3 < size && pattern[i + 1 ..= i + 3] == 'pid') {
appendOrigin()
patterns.add(processIdLogPart)
i += 4
} else if (i + 3 < size && pattern[i + 1] == b'd') {
if (pattern[i + 2] == b'{') {
let ds = i + 3
var di = ds
while (di < size && pattern[di] != b'}') {
di++
}
if (di < size) {
appendOrigin()
patterns.add(TimeLogPart(pattern[ds..di]))
i = di + 1
} else {
origin.add(c)
i++
}
} else {
appendOrigin()
patterns.add(TimeLogPart())
i += 2
}
} else if (i + 1 < size && pattern[i + 1] == b'm') {
appendOrigin()
patterns.add(messageLogPart)
i += 2
} else {
origin.add(c)
i++
}
} else {
origin.add(c)
i++
}
}
appendOrigin()
patterns
}
public func generate(name: String, level: LogLevel, message: ToString, current: DateTime, tid: Int64, writer: LogWriter): Unit {
for (p in patterns) {
p.generate(name, level, message, current, tid, writer)
}
writer.writeString(OS.current.nextLine)
}
}
public interface LogPart {
func generate(name: String, level: LogLevel, message: ToString, current: DateTime, tid: Int64, writer: LogWriter): Unit
}
class OriginLogPart <: LogPart {
OriginLogPart(private let txt: String) {}
public func generate(_: String, _: LogLevel, _: ToString, _: DateTime, _: Int64, writer: LogWriter): Unit {
writer.writeString(txt)
}
}
class LevelLogPart <: LogPart {
init() {}
public func generate(_: String, level: LogLevel, _: ToString, _: DateTime, _: Int64, writer: LogWriter): Unit {
writer.writeString(level.text)
}
}
class NameLogPart <: LogPart {
init() {}
public func generate(name: String, _: LogLevel, _: ToString, _: DateTime, _: Int64, writer: LogWriter): Unit {
writer.writeString(name)
}
}
class AppLogPart <: LogPart {
init() {}
private let app = AppVersion.name
public func generate(_: String, _: LogLevel, _: ToString, _: DateTime, _: Int64, writer: LogWriter): Unit {
writer.writeString(app)
}
}
class AppVersionLogPart <: LogPart {
init(){}
private let version = AppVersion.version
public func generate(_: String, _: LogLevel, _: ToString, _: DateTime, _: Int64, writer: LogWriter): Unit {
writer.writeString(version)
}
}
class TimeLogPart <: LogPart {
TimeLogPart(private let format: String) {}
init() {
this('yyyy-MM-dd,HH:mm:ss.SSS')
}
public func generate(_: String, _: LogLevel, _: ToString, datetime: DateTime, _: Int64, writer: LogWriter): Unit {
writer.writeString(datetime.format(format))
}
}
class MessageLogPart <: LogPart {
init() {}
public func generate(_: String, _: LogLevel, message: ToString, _: DateTime, _: Int64, writer: LogWriter): Unit {
writer.writeString(message.toString())
}
}
class ThreadIdLogPart <: LogPart {
init(){}
public func generate(_: String, _: LogLevel, _: ToString, _: DateTime, tid: Int64, writer: LogWriter): Unit {
writer.writeInt(tid)
}
}
class ThreadNameLogPart <: LogPart {
init(){}
public func generate(_: String, _: LogLevel, _: ToString, _: DateTime, _: Int64, writer: LogWriter): Unit {
writer.writeString(Thread.currentThread.name)
}
}
class ProcessIdLogPart <: LogPart {
init(){}
private let pid = env.getProcessId()
public func generate(_: String, _: LogLevel, _: ToString, _: DateTime, _: Int64, writer: LogWriter): Unit {
writer.writeInt(pid)
}
}
class FountainVersionLogPart <: LogPart {
init(){}
public func generate(_: String, _: LogLevel, _: ToString, _: DateTime, _: Int64, writer: LogWriter): Unit {
writer.writeString(FountainVersion)
}
}