Skip to content

Commit 030fc07

Browse files
authored
Merge pull request #29 from k-kohey/feature/auto_trackings
Add modifier that hooks onAppear and sends logs
2 parents 59699b9 + af82ebc commit 030fc07

File tree

8 files changed

+189
-41
lines changed

8 files changed

+189
-41
lines changed

Demo.swiftpm/Sources/AppModule/Demo.swift

+26-13
Original file line numberDiff line numberDiff line change
@@ -16,10 +16,6 @@ struct MyLogger: LoggerComponent {
1616
}
1717

1818
extension TrackingEvent {
19-
static func impletion(_ screen: String) -> Self {
20-
TrackingEvent(eventName: "impletion", parameters: ["screen": screen])
21-
}
22-
2319
static var tapIncrement: Self {
2420
.init(eventName: "tap increment", parameters: [:])
2521
}
@@ -30,7 +26,7 @@ extension TrackingEvent {
3026
}
3127

3228
struct TimestampMutation: Mutation {
33-
func transform(_ e: Parchment.Loggable, id: Parchment.LoggerComponentID) -> Parchment.AnyLoggable {
29+
func transform(_ e: Loggable, id: LoggerComponentID) -> AnyLoggable {
3430
var e = AnyLoggable(e)
3531
e.parameters["createdAt"] = Date()
3632
return e
@@ -40,16 +36,34 @@ struct TimestampMutation: Mutation {
4036
struct UserIDMutation: Mutation {
4137
let userID = 1
4238

43-
func transform(_ e: Parchment.Loggable, id: Parchment.LoggerComponentID) -> Parchment.AnyLoggable {
39+
func transform(_ e: Loggable, id: LoggerComponentID) -> AnyLoggable {
4440
var e = AnyLoggable(e)
4541
e.parameters["userID"] = userID
4642
return e
4743
}
4844
}
4945

46+
struct ImpletionMutation: Mutation {
47+
func transform(_ e: Loggable, id: LoggerComponentID) -> AnyLoggable {
48+
var e = AnyLoggable(e)
49+
if e.isBased(ImpletionEvent.self) {
50+
e.eventName = (e.parameters["screen"] as! String) + "ScreenEvent"
51+
e.parameters = ["event": "onAppear"]
52+
}
53+
return e
54+
}
55+
}
56+
5057
let logger = LoggerBundler.make(
5158
components: [MyLogger(), DebugLogger()],
52-
bufferFlowController: DefaultBufferFlowController(pollingInterval: 5, delayInputLimit: 5)
59+
bufferFlowController: DefaultBufferFlowController(
60+
pollingInterval: 5, delayInputLimit: 5
61+
),
62+
mutations: [
63+
TimestampMutation(),
64+
ImpletionMutation(),
65+
UserIDMutation()
66+
]
5367
)
5468

5569
@main
@@ -84,13 +98,12 @@ struct ExampleAppApp: App {
8498
.background(Color.gray)
8599
.task {
86100
await logger.startLogging()
87-
88-
await logger.add(
89-
mutations: [TimestampMutation(), UserIDMutation()]
90-
)
91-
92-
await logger.send(event: .impletion("home"))
93101
}
102+
.track(
103+
screen: "Top",
104+
with: logger,
105+
option: .init(policy: .immediately)
106+
)
94107
}
95108
}
96109
}

Sources/Parchment/AnyLoggable.swift

+63
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,63 @@
1+
//
2+
// File.swift
3+
//
4+
//
5+
// Created by Kohei Kawaguchi on 2023/05/23.
6+
//
7+
8+
import Foundation
9+
10+
@globalActor
11+
final actor AnyLoggableActor {
12+
static var shared: AnyLoggableActor = .init()
13+
}
14+
15+
@AnyLoggableActor
16+
private var findCache: [UUID: [String: Bool]] = [:]
17+
18+
public struct AnyLoggable: Loggable {
19+
public var eventName: String
20+
public var parameters: [String : Sendable]
21+
22+
public let base: (any Loggable)?
23+
private let id: UUID = .init()
24+
25+
public init(_ base: any Loggable) {
26+
self.base = base
27+
self.eventName = base.eventName
28+
self.parameters = base.parameters
29+
}
30+
31+
public init(
32+
eventName: String, parameters: [String : Sendable]
33+
) {
34+
self.base = nil
35+
self.eventName = eventName
36+
self.parameters = parameters
37+
}
38+
39+
@AnyLoggableActor
40+
public func isBased<T: Loggable>(_ type: T.Type) -> Bool {
41+
if findCache[id]?["\(T.self)"] == true {
42+
return true
43+
}
44+
let result = find(T.self, from: self)
45+
if findCache[id] == nil {
46+
findCache[id] = [:]
47+
}
48+
findCache[id]?["\(T.self)"] = result
49+
return result
50+
}
51+
52+
private func find<T: Loggable>(
53+
_ type: T.Type, from anyLoggable: AnyLoggable
54+
) -> Bool {
55+
if let base = anyLoggable.base, base is T {
56+
return true
57+
} else if let innerAnyLoggable = anyLoggable.base as? AnyLoggable {
58+
return find(T.self, from: innerAnyLoggable)
59+
} else {
60+
return false
61+
}
62+
}
63+
}

Sources/Parchment/Mutation.swift

+2-22
Original file line numberDiff line numberDiff line change
@@ -7,30 +7,10 @@
77

88
import Foundation
99

10-
public struct AnyLoggable: Loggable {
11-
public var eventName: String
12-
public var parameters: [String : Sendable]
13-
14-
public let base: (any Loggable)?
15-
16-
public init(_ base: any Loggable) {
17-
self.base = base
18-
self.eventName = base.eventName
19-
self.parameters = base.parameters
20-
}
21-
22-
public init(
23-
eventName: String, parameters: [String : Sendable]
24-
) {
25-
self.base = nil
26-
self.eventName = eventName
27-
self.parameters = parameters
28-
}
29-
}
30-
31-
typealias Transform = @Sendable (Loggable, LoggerComponentID) -> AnyLoggable
10+
typealias Transform = @Sendable @AnyLoggableActor (Loggable, LoggerComponentID) -> AnyLoggable
3211

3312
public protocol Mutation: Sendable {
13+
@AnyLoggableActor
3414
func transform(_: any Loggable, id: LoggerComponentID) -> AnyLoggable
3515
}
3616

Sources/Parchment/View+track.swift

+52
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,52 @@
1+
//
2+
// File.swift
3+
//
4+
//
5+
// Created by Kohei Kawaguchi on 2023/05/22.
6+
//
7+
8+
import SwiftUI
9+
10+
public struct ImpletionEvent: Loggable {
11+
public var eventName = "ImpletionEvent"
12+
public var parameters: [String : Sendable]
13+
}
14+
15+
private struct Impletion: ViewModifier {
16+
let screenName: String
17+
let logger: LoggerBundler
18+
let option: LoggerBundler.LoggingOption?
19+
20+
func body(content: Content) -> some View {
21+
content.onAppear {
22+
Task {
23+
let e = ImpletionEvent(
24+
parameters: [
25+
"screen": screenName
26+
]
27+
)
28+
if let option {
29+
await logger.send(e, with: option)
30+
} else {
31+
await logger.send(e)
32+
}
33+
}
34+
}
35+
}
36+
}
37+
38+
public extension View {
39+
func track(
40+
screen name: String,
41+
with logger: LoggerBundler,
42+
option: LoggerBundler.LoggingOption? = nil
43+
) -> some View {
44+
modifier(
45+
Impletion(
46+
screenName: name,
47+
logger: logger,
48+
option: option
49+
)
50+
)
51+
}
52+
}

Sources/ParchmentDefault/LoggerBundler+.swift

+8-2
Original file line numberDiff line numberDiff line change
@@ -11,8 +11,14 @@ public extension LoggerBundler {
1111
static func make(
1212
components: [any LoggerComponent],
1313
buffer: some LogBuffer = try! SQLiteBuffer(),
14-
bufferFlowController: some BufferFlowController = DefaultBufferFlowController(pollingInterval: 60)
14+
bufferFlowController: some BufferFlowController = DefaultBufferFlowController(pollingInterval: 60),
15+
mutations: [Mutation] = []
1516
) -> LoggerBundler {
16-
Self(components: components, buffer: buffer, bufferFlowController: bufferFlowController)
17+
Self(
18+
components: components,
19+
buffer: buffer,
20+
bufferFlowController: bufferFlowController,
21+
mutations: mutations
22+
)
1723
}
1824
}

Tests/ParchmentDefaultTests/MutationTest.swift

+2-2
Original file line numberDiff line numberDiff line change
@@ -30,11 +30,11 @@ private enum Event: Loggable {
3030
class MutationTests: XCTestCase {
3131
#if canImport(UIKit)
3232
// 的確なテストに直す
33-
@MainActor func testTransform() throws {
33+
@MainActor func testTransform() async throws {
3434
let event = Event.hoge
3535
let mutation: [Mutation] = [DeviceDataMutation(device: .current)]
3636

37-
let newEvent = mutation.composed()(event, .init("hoge"))
37+
let newEvent = await mutation.composed()(event, .init("hoge"))
3838

3939
XCTAssertTrue(newEvent.parameters.count > 1)
4040
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
//
2+
// AnyLoggableTests.swift
3+
//
4+
//
5+
// Created by Kohei Kawaguchi on 2023/05/23.
6+
//
7+
8+
@testable import Parchment
9+
import XCTest
10+
11+
private struct HogeEvent: Loggable {
12+
var eventName: String = ""
13+
var parameters: [String : Sendable] = [:]
14+
}
15+
16+
private struct FugaEvent: Loggable {
17+
var eventName: String = ""
18+
var parameters: [String : Sendable] = [:]
19+
}
20+
21+
class AnyLoggableTests: XCTestCase {
22+
@AnyLoggableActor
23+
func test_isBased() async throws {
24+
let e = AnyLoggable(
25+
AnyLoggable(
26+
HogeEvent()
27+
)
28+
)
29+
30+
XCTAssertTrue(e.isBased(HogeEvent.self))
31+
XCTAssertTrue(e.isBased(AnyLoggable.self))
32+
XCTAssertFalse(e.isBased(FugaEvent.self))
33+
}
34+
}

Tests/ParchmentTests/TransformTests.swift

+2-2
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@ private struct MutationMock: Mutation {
2020
}
2121

2222
final class TransformTests: XCTestCase {
23-
func testComposed() {
23+
func testComposed() async {
2424
let mutationA = MutationMock { l, id in
2525
AnyLoggable(
2626
eventName: l.eventName,
@@ -40,7 +40,7 @@ final class TransformTests: XCTestCase {
4040

4141

4242
let mutations: [any Mutation] = [mutationA, mutationB]
43-
let r = mutations.composed()(TrackingEvent(eventName: "", parameters: ["foo": 2]), .init(""))
43+
let r = await mutations.composed()(TrackingEvent(eventName: "", parameters: ["foo": 2]), .init(""))
4444

4545
XCTAssertEqual(
4646
r.parameters as NSDictionary,

0 commit comments

Comments
 (0)