feat: update

This commit is contained in:
zhangmin 2026-02-11 17:00:58 +08:00
parent c4dbf83e7d
commit 500502c206
4 changed files with 122 additions and 220 deletions

View File

@ -6,12 +6,14 @@
//
import UIKit
import Network
/// SDK
/// 1SDK
/// 2/20600
/// 3
/// 4
/// 5
public class YXTracking: NSObject {
static let shared = YXTracking()
@ -28,6 +30,8 @@ public class YXTracking: NSObject {
private var systemAllDimInfoResponseResult: SystemAllDimInfoResponseResult = SystemAllDimInfoResponseResult() //
private var isPushingEventList: Bool = false //
/// SDK
/// - Parameters:
/// - host:
@ -63,6 +67,9 @@ public class YXTracking: NSObject {
//
YXTracking.shared.log(message: "SDK config success!");
//
YXTracking.shared.addNetworkReachabilityObserver()
//
YXTracking.shared.addEnterBackgroundNotificationObserver()
YXTracking.shared.addWillTerminateNotificationObserver()
@ -87,7 +94,22 @@ public class YXTracking: NSObject {
/// - customTags:
/// - pushNow
/// - Returns:
public class func addEvent(eventType: String, page: String, buttonId: String, url: String, customTags: Array<String>?, pushNow: Bool) {
public class func addEvent(eventType: String, page: String, buttonId: String, url: String, customTags: Dictionary<String, String>?, pushNow: Bool) {
//
YXTracking.shared.eventRequestResult.eventType = eventType
//
let eventParams = EventParams(page: page, buttonId: buttonId, url: url)
YXTracking.shared.eventRequestResult.eventParams = eventParams
//
YXTracking.shared.eventRequestResult.customTags = customTags
//
YXTracking.shared.getTimestamp()
YXTracking.shared.log(message: "SDK add event:\n\(YXTracking.shared.eventRequestResult)");
//
if let systemEventTypes = YXTracking.shared.systemAllDimInfoResponseResult.systemEventTypes,
systemEventTypes.count > 0 {
@ -102,22 +124,10 @@ public class YXTracking: NSObject {
}
} else {
YXTracking.shared.error(message: "SDK push event error, systemEventTypes is empty!");
YXTrackingSQLiteManager.addTrackingData(data: YXTracking.shared.eventRequestResult)
return
}
//
YXTracking.shared.eventRequestResult.eventType = eventType
//
let eventParams = EventParams(page: page, buttonId: buttonId, url: url)
YXTracking.shared.eventRequestResult.eventParams = eventParams
//
YXTracking.shared.eventRequestResult.customTags = customTags
//
YXTracking.shared.getTimestamp()
//
YXTracking.shared.log(message: "SDK add event success\nEvent Request Parameter:\n\(YXTracking.shared.eventRequestResult)");
@ -141,7 +151,7 @@ public class YXTracking: NSObject {
YXTracking.shared.log(message: "SDK config user info success\nUser info:\n\(userInfo)");
}
///
///
public class func clearUserInfo() {
YXTracking.shared.eventRequestResult.userInfo = nil
@ -185,7 +195,7 @@ public class YXTracking: NSObject {
YXTracking.shared.eventRequestResult.timestamp = isoStringWithMilliseconds
}
///
///
func checkEventCount() {
let count = YXTrackingSQLiteManager.getTrackingDataCount()
if count >= cacheCount {
@ -196,10 +206,12 @@ public class YXTracking: NSObject {
///
func startThreadPool() {
let queue = DispatchQueue(label: "com.yx.tracking", qos: .userInitiated)
queue.async { [self] in
queue.async { [weak self] in
guard let self = self else { return }
while true {
YXTracking.shared.log(message: "SDK push event list on thread pool!")
YXTracking.pushEventList()
sleep(UInt32(pushInterval))
Thread.sleep(forTimeInterval: TimeInterval(pushInterval))
}
}
}
@ -272,6 +284,8 @@ public class YXTracking: NSObject {
// eventJSON
let jsonEncoder = JSONEncoder()
do {
YXTracking.shared.log(message: "SDK push event:\n\(event)");
let jsonData = try jsonEncoder.encode(event)
request.httpBody = jsonData
} catch {
@ -306,44 +320,56 @@ public class YXTracking: NSObject {
///
public class func pushEventList() {
/// pushEventList
if YXTracking.shared.isPushingEventList {
return
}
YXTracking.shared.isPushingEventList = true
///
let count = YXTrackingSQLiteManager.getTrackingDataCount()
guard count > 0 else { return }
/// URL
let urlString = "\(YXTracking.shared.host)/api/ExternalEventlogs/AddEventListLog"
guard let url = URL(string: urlString) else {
return
}
guard let url = URL(string: urlString) else { return }
// URL
/// URL
var request = URLRequest(url: url)
request.httpMethod = "POST"
request.setValue("application/json", forHTTPHeaderField: "Content-Type")
//
///
let eventList = YXTrackingSQLiteManager.readTrackingData()
var event: [EventRequestResult] = []
var events: [EventRequestResult] = []
for item in eventList {
event.append(item)
events.append(item)
}
// event listJSON
///
guard !events.isEmpty else { return }
/// event listJSON
let jsonEncoder = JSONEncoder()
do {
let jsonData = try jsonEncoder.encode(event)
let jsonData = try jsonEncoder.encode(events)
request.httpBody = jsonData
} catch {
YXTracking.shared.error(message: "SDK push event list error encoding\n\(error)");
YXTracking.shared.isPushingEventList = false
return
}
//
///
let task = URLSession.shared.dataTask(with: request) { data, response, error in
if let error = error {
YXTracking.shared.error(message: "SDK push event list error sending\n\(error)");
YXTracking.shared.isPushingEventList = false
return
}
guard let data = data else {
YXTracking.shared.isPushingEventList = false
return
}
@ -351,27 +377,47 @@ public class YXTracking: NSObject {
if let dictionary = json as? [String: Any] {
if let code = dictionary["code"] as? Int, code == 200 {
YXTrackingSQLiteManager.deleteTrackingData()
YXTracking.shared.success(message: "SDK push event list success Response:\n\(String(data: data, encoding: .utf8) ?? "")");
YXTracking.shared.success(message: "SDK push event list success Response:\n\(String(data: data, encoding: .utf8) ?? "")")
} else {
YXTracking.shared.error(message: "SDK push event fail Response:\n\(String(data: data, encoding: .utf8) ?? "")");
YXTracking.shared.error(message: "SDK push event list fail Response:\n\(String(data: data, encoding: .utf8) ?? "")")
}
}
YXTracking.shared.isPushingEventList = false
}
task.resume()
}
//
///
func addNetworkReachabilityObserver() {
let monitor = NWPathMonitor()
let queue = DispatchQueue(label: "NetworkMonitor")
monitor.pathUpdateHandler = { path in
let systemEventTypes = YXTracking.shared.systemAllDimInfoResponseResult.systemEventTypes ?? []
guard systemEventTypes.isEmpty else { return }
if path.status == .satisfied {
YXTracking.shared.log(message: "SDK network connected!");
YXTracking.shared.getSystemAllDimInfo()
} else {
YXTracking.shared.log(message: "SDK network not connected!");
}
}
monitor.start(queue: queue)
}
///
func addEnterBackgroundNotificationObserver() {
NotificationCenter.default.addObserver(self, selector: #selector(onPushEventList), name: UIApplication.didEnterBackgroundNotification, object: nil)
}
//
///
func addWillTerminateNotificationObserver() {
NotificationCenter.default.addObserver(self, selector: #selector(onPushEventList), name: UIApplication.willTerminateNotification, object: nil)
}
//
///
@objc func onPushEventList() {
YXTracking.pushEventList()
}

View File

@ -38,7 +38,7 @@ public struct EventRequestResult: Codable {
var timestamp: String? // ISO8601
var deviceInfo: DeviceInfo? //
var eventParams: EventParams? //
var customTags: Array<String>? //
var customTags: Dictionary<String, String>? //
}
///

View File

@ -10,210 +10,66 @@ import SQLite3
class YXTrackingSQLiteManager: NSObject {
//
class func addTrackingData(data: EventRequestResult) {
//
let dbPath = NSSearchPathForDirectoriesInDomains(.documentDirectory, .userDomainMask, true).first!
let path = URL(fileURLWithPath: dbPath).appendingPathComponent("tracking.sqlite").path
var db: OpaquePointer? = nil
if sqlite3_open(path, &db) != SQLITE_OK {
YXTracking.shared.error(message: "SDK sqlite open failed!");
return
}
static let shared = YXTrackingSQLiteManager()
YXTracking.shared.success(message: "SDK sqlite open success!");
//
let createTableSQL = """
CREATE TABLE IF NOT EXISTS tracking_data (
id INTEGER PRIMARY KEY AUTOINCREMENT,
system_code TEXT,
event_type TEXT,
user_info BLOB,
client_type INT,
client_timestamp INT64,
timestamp TEXT,
device_info BLOB,
event_params BLOB,
custom_tags BLOB
)
"""
if sqlite3_exec(db, createTableSQL, nil, nil, nil) != SQLITE_OK {
YXTracking.shared.error(message: "SDK sqlite create table failed!");
return
}
YXTracking.shared.success(message: "SDK sqlite create table success!");
//
let encoder = JSONEncoder()
let insertSQL = "INSERT INTO tracking_data (system_code, event_type, user_info, client_type, client_timestamp, timestamp, device_info, event_params, custom_tags) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)"
var statement: OpaquePointer? = nil
if sqlite3_prepare_v2(db, insertSQL, -1, &statement, nil) == SQLITE_OK {
let system_code = data.system_code ?? ""
let event_type = data.eventType ?? ""
let user_info = (data.userInfo != nil) ? try! encoder.encode(data.userInfo) : nil
let client_type = data.clientType ?? 0
let client_timestamp = data.clientTimestamp ?? 0
let timestamp = data.timestamp ?? ""
let device_info = try! encoder.encode(data.deviceInfo)
let event_params = try! encoder.encode(data.eventParams)
let custom_tags = try! JSONSerialization.data(withJSONObject: data.customTags ?? [], options: [])
if sqlite3_bind_text(statement, 1, system_code, -1, nil) != SQLITE_OK || sqlite3_bind_text(statement, 2, event_type, -1, nil) != SQLITE_OK || ((data.userInfo != nil) ? (sqlite3_bind_blob(statement, 3, user_info! as CFTypeRef as? UnsafeRawPointer, Int32(user_info!.count), nil) != SQLITE_OK) : (sqlite3_bind_null(statement, 3) != SQLITE_OK)) || sqlite3_bind_int(statement, 4, client_type) != SQLITE_OK || sqlite3_bind_int64(statement, 5, client_timestamp) != SQLITE_OK || sqlite3_bind_text(statement, 6, timestamp, -1, nil) != SQLITE_OK || sqlite3_bind_blob(statement, 7, device_info as CFTypeRef as? UnsafeRawPointer, Int32(device_info.count), nil) != SQLITE_OK || sqlite3_bind_blob(statement, 8, event_params as CFTypeRef as? UnsafeRawPointer, Int32(event_params.count), nil) != SQLITE_OK || sqlite3_bind_blob(statement, 9, custom_tags as CFTypeRef as? UnsafeRawPointer, Int32(custom_tags.count), nil) != SQLITE_OK {
YXTracking.shared.error(message: "SDK sqlite bind parameter failed!");
return
} else if sqlite3_step(statement) != SQLITE_DONE {
YXTracking.shared.error(message: "SDK sqlite insert data failed!");
return
}
YXTracking.shared.success(message: "SDK sqlite bind parameter success!");
} else {
YXTracking.shared.error(message: "SDK sqlite preparing to insert data failed!");
return
}
YXTracking.shared.success(message: "SDK sqlite insert data success!");
//
do {
sqlite3_close(db)
}
///
private let fileManager = FileManager.default
///
var documentsDirectory: URL {
fileManager.urls(for: .documentDirectory, in: .userDomainMask).first!
}
//
///
var fileURL: URL {
documentsDirectory.appendingPathComponent("tracking.json")
}
/// JSON
let encoder = JSONEncoder()
/// JSON
let decoder = JSONDecoder()
///
class func addTrackingData(data: EventRequestResult) {
var trackingData: [EventRequestResult] = []
trackingData.append(contentsOf: readTrackingData())
trackingData.append(data)
let data = try! YXTrackingSQLiteManager.shared.encoder.encode(trackingData)
try! data.write(to: YXTrackingSQLiteManager.shared.fileURL, options: .atomic)
}
///
class func readTrackingData() -> [EventRequestResult] {
var trackingData: [EventRequestResult] = []
//
let dbPath = NSSearchPathForDirectoriesInDomains(.documentDirectory, .userDomainMask, true).first!
let path = URL(fileURLWithPath: dbPath).appendingPathComponent("tracking.sqlite").path
var db: OpaquePointer? = nil
if sqlite3_open(path, &db) != SQLITE_OK {
YXTracking.shared.error(message: "SDK sqlite open failed!");
/// fileURL
guard YXTrackingSQLiteManager.shared.fileManager.fileExists(atPath: YXTrackingSQLiteManager.shared.fileURL.path) else {
return trackingData
}
//
let decoder = JSONDecoder()
let querySQL = "SELECT system_code, event_type, user_info, client_type, client_timestamp, timestamp, device_info, event_params, custom_tags FROM tracking_data"
var statement: OpaquePointer? = nil
if sqlite3_prepare_v2(db, querySQL, -1, &statement, nil) == SQLITE_OK {
while (sqlite3_step(statement) == SQLITE_ROW) {
// EventRequestResult
let system_code = String(cString: sqlite3_column_text(statement, 1))
let event_type = String(cString: sqlite3_column_text(statement, 2))
var user_info = UserInfo(userId: nil, userName: nil, account: nil)
if sqlite3_column_type(statement, 3) != SQLITE_NULL {
if let user_info_blob = sqlite3_column_blob(statement, 3) {
let user_info_blob_length = sqlite3_column_bytes(statement, 3)
let data = Data(bytes: user_info_blob, count: Int(user_info_blob_length))
user_info = try! decoder.decode(UserInfo.self, from: data)
}
}
let client_type = sqlite3_column_int(statement, 4)
let client_timestamp = sqlite3_column_int64(statement, 5)
let timestamp = String(cString: sqlite3_column_text(statement, 6))
var device_info = DeviceInfo(os: nil, model: nil, screenResolution: nil)
if sqlite3_column_type(statement, 7) != SQLITE_NULL {
if let device_info_blob = sqlite3_column_blob(statement, 7) {
let device_info_blob_length = sqlite3_column_bytes(statement, 7)
let data = Data(bytes: device_info_blob, count: Int(device_info_blob_length))
device_info = try! decoder.decode(DeviceInfo.self, from: data)
}
}
var event_params = EventParams(page: nil, buttonId: nil, url: nil)
if sqlite3_column_type(statement, 8) != SQLITE_NULL {
if let event_params_blob = sqlite3_column_blob(statement, 8) {
let event_params_blob_length = sqlite3_column_bytes(statement, 8)
let data = Data(bytes: event_params_blob, count: Int(event_params_blob_length))
event_params = try! decoder.decode(EventParams.self, from: data)
}
}
var custom_tags = Array<String>()
if sqlite3_column_type(statement, 9) != SQLITE_NULL {
if let custom_tags_blob = sqlite3_column_blob(statement, 9) {
let custom_tags_blob_length = sqlite3_column_bytes(statement, 9)
let data = Data(bytes: custom_tags_blob, count: Int(custom_tags_blob_length))
custom_tags = try! decoder.decode([String].self, from: data)
}
}
let event = EventRequestResult(system_code: system_code,
eventType: event_type,
userInfo: user_info,
clientType: client_type,
clientTimestamp: client_timestamp,
timestamp: timestamp,
deviceInfo: device_info,
eventParams: event_params,
customTags: custom_tags)
trackingData.append(event)
}
sqlite3_finalize(statement)
} else {
YXTracking.shared.error(message: "SDK sqlite read data failed!");
}
//
do {
sqlite3_close(db)
}
let data = try! Data(contentsOf: YXTrackingSQLiteManager.shared.fileURL)
let array = try! YXTrackingSQLiteManager.shared.decoder.decode([EventRequestResult].self, from: data)
trackingData.append(contentsOf: array)
return trackingData
}
//
///
class func deleteTrackingData() {
//
let dbPath = NSSearchPathForDirectoriesInDomains(.documentDirectory, .userDomainMask, true).first!
let path = URL(fileURLWithPath: dbPath).appendingPathComponent("tracking.sqlite").path
var db: OpaquePointer? = nil
if sqlite3_open(path, &db) != SQLITE_OK {
YXTracking.shared.error(message: "SDK sqlite open failed!");
guard YXTrackingSQLiteManager.shared.fileManager.fileExists(atPath: YXTrackingSQLiteManager.shared.fileURL.path) else {
YXTracking.shared.debug(message: "SDK tracking data file not exist!")
return
}
//
let deleteSQL = "DELETE FROM tracking_data"
if sqlite3_exec(db, deleteSQL, nil, nil, nil) != SQLITE_OK {
YXTracking.shared.error(message: "SDK sqlite delete data failed!");
}
//
do {
sqlite3_close(db)
}
try! YXTrackingSQLiteManager.shared.fileManager.removeItem(at: YXTrackingSQLiteManager.shared.fileURL)
YXTracking.shared.debug(message: "SDK tracking data file deleted!")
}
//
///
class func getTrackingDataCount() -> Int {
//
let dbPath = NSSearchPathForDirectoriesInDomains(.documentDirectory, .userDomainMask, true).first!
let path = URL(fileURLWithPath: dbPath).appendingPathComponent("tracking.sqlite").path
var db: OpaquePointer? = nil
if sqlite3_open(path, &db) != SQLITE_OK {
YXTracking.shared.error(message: "SDK sqlite open failed!");
return -1
}
//
var dataCount = -1
let querySQL = "SELECT COUNT(*) FROM tracking_data"
var statement: OpaquePointer? = nil
if sqlite3_prepare_v2(db, querySQL, -1, &statement, nil) == SQLITE_OK {
sqlite3_step(statement)
dataCount = Int(sqlite3_column_int64(statement, 0))
sqlite3_finalize(statement)
} else {
YXTracking.shared.error(message: "SDK sqlite read data failed!");
}
//
do {
sqlite3_close(db)
}
return dataCount
var trackingData: [EventRequestResult] = []
trackingData.append(contentsOf: readTrackingData())
return trackingData.count
}
}