mirror of
https://github.com/formbricks/formbricks.git
synced 2026-05-04 03:16:15 -05:00
Compare commits
10 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 413e9575b6 | |||
| 712c3a0aaf | |||
| 55eb43f94e | |||
| 0982813fa7 | |||
| 403e337793 | |||
| a8a299e4b8 | |||
| e1a39d4ed9 | |||
| 959e46d5a9 | |||
| 7b0efc21b9 | |||
| 74d24f80d4 |
@@ -72,3 +72,4 @@ infra/terraform/.terraform/
|
|||||||
# IntelliJ IDEA
|
# IntelliJ IDEA
|
||||||
/.idea/
|
/.idea/
|
||||||
/*.iml
|
/*.iml
|
||||||
|
packages/ios/FormbricksSDK/FormbricksSDK.xcodeproj/project.xcworkspace/xcuserdata
|
||||||
|
|||||||
@@ -127,7 +127,7 @@ const nextConfig = {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
// matching all API routes
|
// matching all API routes
|
||||||
source: "/api/v1/client/:path*",
|
source: "/api/(v1|v2)/client/:path*",
|
||||||
headers: [
|
headers: [
|
||||||
{ key: "Access-Control-Allow-Credentials", value: "true" },
|
{ key: "Access-Control-Allow-Credentials", value: "true" },
|
||||||
{ key: "Access-Control-Allow-Origin", value: "*" },
|
{ key: "Access-Control-Allow-Origin", value: "*" },
|
||||||
|
|||||||
@@ -1,12 +1,14 @@
|
|||||||
package com.formbricks.demo
|
package com.formbricks.demo
|
||||||
|
|
||||||
import android.os.Bundle
|
import android.os.Bundle
|
||||||
|
import android.util.Log
|
||||||
import android.widget.Button
|
import android.widget.Button
|
||||||
import androidx.activity.enableEdgeToEdge
|
import androidx.activity.enableEdgeToEdge
|
||||||
import androidx.appcompat.app.AppCompatActivity
|
import androidx.appcompat.app.AppCompatActivity
|
||||||
import androidx.core.view.ViewCompat
|
import androidx.core.view.ViewCompat
|
||||||
import androidx.core.view.WindowInsetsCompat
|
import androidx.core.view.WindowInsetsCompat
|
||||||
import com.formbricks.formbrickssdk.Formbricks
|
import com.formbricks.formbrickssdk.Formbricks
|
||||||
|
import com.formbricks.formbrickssdk.FormbricksCallback
|
||||||
import com.formbricks.formbrickssdk.helper.FormbricksConfig
|
import com.formbricks.formbrickssdk.helper.FormbricksConfig
|
||||||
import java.util.UUID
|
import java.util.UUID
|
||||||
|
|
||||||
@@ -15,6 +17,25 @@ class MainActivity : AppCompatActivity() {
|
|||||||
super.onCreate(savedInstanceState)
|
super.onCreate(savedInstanceState)
|
||||||
enableEdgeToEdge()
|
enableEdgeToEdge()
|
||||||
|
|
||||||
|
Formbricks.callback = object: FormbricksCallback {
|
||||||
|
override fun onSurveyStarted() {
|
||||||
|
Log.d("FormbricksCallback", "onSurveyStarted")
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onSurveyFinished() {
|
||||||
|
Log.d("FormbricksCallback", "onSurveyFinished")
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onSurveyClosed() {
|
||||||
|
Log.d("FormbricksCallback", "onSurveyClosed")
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onError(error: Exception) {
|
||||||
|
Log.d("FormbricksCallback", "onError: ${error.localizedMessage}")
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
val config = FormbricksConfig.Builder("[appUrl]","[environmentId]")
|
val config = FormbricksConfig.Builder("[appUrl]","[environmentId]")
|
||||||
.setLoggingEnabled(true)
|
.setLoggingEnabled(true)
|
||||||
.setFragmentManager(supportFragmentManager)
|
.setFragmentManager(supportFragmentManager)
|
||||||
@@ -35,4 +56,4 @@ class MainActivity : AppCompatActivity() {
|
|||||||
Formbricks.track("click_demo_button")
|
Formbricks.track("click_demo_button")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,15 @@
|
|||||||
|
{
|
||||||
|
"originHash" : "92c0230fb0adc404299bb05aba6c51a76f86c388fdfb9f4e9bed3a757f80fc07",
|
||||||
|
"pins" : [
|
||||||
|
{
|
||||||
|
"identity" : "anycodable",
|
||||||
|
"kind" : "remoteSourceControl",
|
||||||
|
"location" : "https://github.com/Flight-School/AnyCodable",
|
||||||
|
"state" : {
|
||||||
|
"revision" : "862808b2070cd908cb04f9aafe7de83d35f81b05",
|
||||||
|
"version" : "0.6.7"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"version" : 3
|
||||||
|
}
|
||||||
@@ -7,6 +7,7 @@
|
|||||||
objects = {
|
objects = {
|
||||||
|
|
||||||
/* Begin PBXBuildFile section */
|
/* Begin PBXBuildFile section */
|
||||||
|
4D7D8DD62DB14F18002C453E /* AnyCodable in Frameworks */ = {isa = PBXBuildFile; productRef = 4D7D8DD52DB14F18002C453E /* AnyCodable */; };
|
||||||
4DDAED692D50D49B00A19B1F /* FormbricksSDK.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 4DDAED602D50D49A00A19B1F /* FormbricksSDK.framework */; };
|
4DDAED692D50D49B00A19B1F /* FormbricksSDK.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 4DDAED602D50D49A00A19B1F /* FormbricksSDK.framework */; };
|
||||||
/* End PBXBuildFile section */
|
/* End PBXBuildFile section */
|
||||||
|
|
||||||
@@ -108,6 +109,7 @@
|
|||||||
isa = PBXFrameworksBuildPhase;
|
isa = PBXFrameworksBuildPhase;
|
||||||
buildActionMask = 2147483647;
|
buildActionMask = 2147483647;
|
||||||
files = (
|
files = (
|
||||||
|
4D7D8DD62DB14F18002C453E /* AnyCodable in Frameworks */,
|
||||||
);
|
);
|
||||||
runOnlyForDeploymentPostprocessing = 0;
|
runOnlyForDeploymentPostprocessing = 0;
|
||||||
};
|
};
|
||||||
@@ -122,11 +124,19 @@
|
|||||||
/* End PBXFrameworksBuildPhase section */
|
/* End PBXFrameworksBuildPhase section */
|
||||||
|
|
||||||
/* Begin PBXGroup section */
|
/* Begin PBXGroup section */
|
||||||
|
4D7D8DD42DB14F18002C453E /* Frameworks */ = {
|
||||||
|
isa = PBXGroup;
|
||||||
|
children = (
|
||||||
|
);
|
||||||
|
name = Frameworks;
|
||||||
|
sourceTree = "<group>";
|
||||||
|
};
|
||||||
4DDAED562D50D49A00A19B1F = {
|
4DDAED562D50D49A00A19B1F = {
|
||||||
isa = PBXGroup;
|
isa = PBXGroup;
|
||||||
children = (
|
children = (
|
||||||
4DDAED622D50D49A00A19B1F /* FormbricksSDK */,
|
4DDAED622D50D49A00A19B1F /* FormbricksSDK */,
|
||||||
4DDAED6C2D50D49B00A19B1F /* FormbricksSDKTests */,
|
4DDAED6C2D50D49B00A19B1F /* FormbricksSDKTests */,
|
||||||
|
4D7D8DD42DB14F18002C453E /* Frameworks */,
|
||||||
4DDAED612D50D49A00A19B1F /* Products */,
|
4DDAED612D50D49A00A19B1F /* Products */,
|
||||||
);
|
);
|
||||||
sourceTree = "<group>";
|
sourceTree = "<group>";
|
||||||
@@ -171,6 +181,7 @@
|
|||||||
);
|
);
|
||||||
name = FormbricksSDK;
|
name = FormbricksSDK;
|
||||||
packageProductDependencies = (
|
packageProductDependencies = (
|
||||||
|
4D7D8DD52DB14F18002C453E /* AnyCodable */,
|
||||||
);
|
);
|
||||||
productName = FormbricksSDK;
|
productName = FormbricksSDK;
|
||||||
productReference = 4DDAED602D50D49A00A19B1F /* FormbricksSDK.framework */;
|
productReference = 4DDAED602D50D49A00A19B1F /* FormbricksSDK.framework */;
|
||||||
@@ -227,6 +238,9 @@
|
|||||||
);
|
);
|
||||||
mainGroup = 4DDAED562D50D49A00A19B1F;
|
mainGroup = 4DDAED562D50D49A00A19B1F;
|
||||||
minimizedProjectReferenceProxies = 1;
|
minimizedProjectReferenceProxies = 1;
|
||||||
|
packageReferences = (
|
||||||
|
4DA4A0952DB14E67007299C0 /* XCRemoteSwiftPackageReference "AnyCodable" */,
|
||||||
|
);
|
||||||
preferredProjectObjectVersion = 77;
|
preferredProjectObjectVersion = 77;
|
||||||
productRefGroup = 4DDAED612D50D49A00A19B1F /* Products */;
|
productRefGroup = 4DDAED612D50D49A00A19B1F /* Products */;
|
||||||
projectDirPath = "";
|
projectDirPath = "";
|
||||||
@@ -536,6 +550,25 @@
|
|||||||
defaultConfigurationName = Release;
|
defaultConfigurationName = Release;
|
||||||
};
|
};
|
||||||
/* End XCConfigurationList section */
|
/* End XCConfigurationList section */
|
||||||
|
|
||||||
|
/* Begin XCRemoteSwiftPackageReference section */
|
||||||
|
4DA4A0952DB14E67007299C0 /* XCRemoteSwiftPackageReference "AnyCodable" */ = {
|
||||||
|
isa = XCRemoteSwiftPackageReference;
|
||||||
|
repositoryURL = "https://github.com/Flight-School/AnyCodable";
|
||||||
|
requirement = {
|
||||||
|
kind = exactVersion;
|
||||||
|
version = 0.6.7;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
/* End XCRemoteSwiftPackageReference section */
|
||||||
|
|
||||||
|
/* Begin XCSwiftPackageProductDependency section */
|
||||||
|
4D7D8DD52DB14F18002C453E /* AnyCodable */ = {
|
||||||
|
isa = XCSwiftPackageProductDependency;
|
||||||
|
package = 4DA4A0952DB14E67007299C0 /* XCRemoteSwiftPackageReference "AnyCodable" */;
|
||||||
|
productName = AnyCodable;
|
||||||
|
};
|
||||||
|
/* End XCSwiftPackageProductDependency section */
|
||||||
};
|
};
|
||||||
rootObject = 4DDAED572D50D49A00A19B1F /* Project object */;
|
rootObject = 4DDAED572D50D49A00A19B1F /* Project object */;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -91,6 +91,11 @@ import Network
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let existing = userManager?.userId, !existing.isEmpty {
|
||||||
|
logger?.error("A userId is already set (\"\(existing)\") – please call Formbricks.logout() before setting a new one.")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
userManager?.set(userId: userId)
|
userManager?.set(userId: userId)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,147 +0,0 @@
|
|||||||
// https://github.com/Flight-School/AnyCodable/blob/master/Sources/AnyCodable/AnyCodable.swift
|
|
||||||
|
|
||||||
import Foundation
|
|
||||||
/**
|
|
||||||
A type-erased `Codable` value.
|
|
||||||
|
|
||||||
The `AnyCodable` type forwards encoding and decoding responsibilities
|
|
||||||
to an underlying value, hiding its specific underlying type.
|
|
||||||
|
|
||||||
You can encode or decode mixed-type values in dictionaries
|
|
||||||
and other collections that require `Encodable` or `Decodable` conformance
|
|
||||||
by declaring their contained type to be `AnyCodable`.
|
|
||||||
|
|
||||||
- SeeAlso: `AnyEncodable`
|
|
||||||
- SeeAlso: `AnyDecodable`
|
|
||||||
*/
|
|
||||||
struct AnyCodable: Codable {
|
|
||||||
public let value: Any
|
|
||||||
|
|
||||||
public init<T>(_ value: T?) {
|
|
||||||
self.value = value ?? ()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
extension AnyCodable: AnyEncodableProtocol, AnyDecodableProtocol {}
|
|
||||||
|
|
||||||
extension AnyCodable: Equatable {
|
|
||||||
public static func == (lhs: AnyCodable, rhs: AnyCodable) -> Bool {
|
|
||||||
switch (lhs.value, rhs.value) {
|
|
||||||
case is (Void, Void):
|
|
||||||
return true
|
|
||||||
case let (lhs as Bool, rhs as Bool):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Int, rhs as Int):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Int8, rhs as Int8):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Int16, rhs as Int16):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Int32, rhs as Int32):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Int64, rhs as Int64):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as UInt, rhs as UInt):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as UInt8, rhs as UInt8):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as UInt16, rhs as UInt16):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as UInt32, rhs as UInt32):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as UInt64, rhs as UInt64):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Float, rhs as Float):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Double, rhs as Double):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as String, rhs as String):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as [String: AnyCodable], rhs as [String: AnyCodable]):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as [AnyCodable], rhs as [AnyCodable]):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as [String: Any], rhs as [String: Any]):
|
|
||||||
return NSDictionary(dictionary: lhs) == NSDictionary(dictionary: rhs)
|
|
||||||
case let (lhs as [Any], rhs as [Any]):
|
|
||||||
return NSArray(array: lhs) == NSArray(array: rhs)
|
|
||||||
case is (NSNull, NSNull):
|
|
||||||
return true
|
|
||||||
default:
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
extension AnyCodable: CustomStringConvertible {
|
|
||||||
public var description: String {
|
|
||||||
switch value {
|
|
||||||
case is Void:
|
|
||||||
return String(describing: nil as Any?)
|
|
||||||
case let value as CustomStringConvertible:
|
|
||||||
return value.description
|
|
||||||
default:
|
|
||||||
return String(describing: value)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
extension AnyCodable: CustomDebugStringConvertible {
|
|
||||||
public var debugDescription: String {
|
|
||||||
if let value = value as? CustomDebugStringConvertible {
|
|
||||||
return "AnyCodable(\(value.debugDescription))"
|
|
||||||
}
|
|
||||||
return "AnyCodable(\(description))"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
extension AnyCodable: ExpressibleByNilLiteral {}
|
|
||||||
extension AnyCodable: ExpressibleByBooleanLiteral {}
|
|
||||||
extension AnyCodable: ExpressibleByIntegerLiteral {}
|
|
||||||
extension AnyCodable: ExpressibleByFloatLiteral {}
|
|
||||||
extension AnyCodable: ExpressibleByStringLiteral {}
|
|
||||||
extension AnyCodable: ExpressibleByStringInterpolation {}
|
|
||||||
extension AnyCodable: ExpressibleByArrayLiteral {}
|
|
||||||
extension AnyCodable: ExpressibleByDictionaryLiteral {}
|
|
||||||
|
|
||||||
|
|
||||||
extension AnyCodable: Hashable {
|
|
||||||
public func hash(into hasher: inout Hasher) {
|
|
||||||
switch value {
|
|
||||||
case let value as Bool:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Int:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Int8:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Int16:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Int32:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Int64:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as UInt:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as UInt8:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as UInt16:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as UInt32:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as UInt64:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Float:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Double:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as String:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as [String: AnyCodable]:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as [AnyCodable]:
|
|
||||||
hasher.combine(value)
|
|
||||||
default:
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,189 +0,0 @@
|
|||||||
// https://github.com/Flight-School/AnyCodable/blob/master/Sources/AnyCodable/AnyCodable.swift
|
|
||||||
|
|
||||||
#if canImport(Foundation)
|
|
||||||
import Foundation
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/**
|
|
||||||
A type-erased `Decodable` value.
|
|
||||||
|
|
||||||
The `AnyDecodable` type forwards decoding responsibilities
|
|
||||||
to an underlying value, hiding its specific underlying type.
|
|
||||||
|
|
||||||
You can decode mixed-type values in dictionaries
|
|
||||||
and other collections that require `Decodable` conformance
|
|
||||||
by declaring their contained type to be `AnyDecodable`:
|
|
||||||
|
|
||||||
let json = """
|
|
||||||
{
|
|
||||||
"boolean": true,
|
|
||||||
"integer": 42,
|
|
||||||
"double": 3.141592653589793,
|
|
||||||
"string": "string",
|
|
||||||
"array": [1, 2, 3],
|
|
||||||
"nested": {
|
|
||||||
"a": "alpha",
|
|
||||||
"b": "bravo",
|
|
||||||
"c": "charlie"
|
|
||||||
},
|
|
||||||
"null": null
|
|
||||||
}
|
|
||||||
""".data(using: .utf8)!
|
|
||||||
|
|
||||||
let decoder = JSONDecoder()
|
|
||||||
let dictionary = try! decoder.decode([String: AnyDecodable].self, from: json)
|
|
||||||
*/
|
|
||||||
struct AnyDecodable: Decodable {
|
|
||||||
public let value: Any
|
|
||||||
|
|
||||||
public init<T>(_ value: T?) {
|
|
||||||
self.value = value ?? ()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@usableFromInline
|
|
||||||
protocol AnyDecodableProtocol {
|
|
||||||
var value: Any { get }
|
|
||||||
init<T>(_ value: T?)
|
|
||||||
}
|
|
||||||
|
|
||||||
extension AnyDecodable: AnyDecodableProtocol {}
|
|
||||||
|
|
||||||
extension AnyDecodableProtocol {
|
|
||||||
public init(from decoder: Decoder) throws {
|
|
||||||
let container = try decoder.singleValueContainer()
|
|
||||||
|
|
||||||
if container.decodeNil() {
|
|
||||||
#if canImport(Foundation)
|
|
||||||
self.init(NSNull())
|
|
||||||
#else
|
|
||||||
self.init(Optional<Self>.none)
|
|
||||||
#endif
|
|
||||||
} else if let bool = try? container.decode(Bool.self) {
|
|
||||||
self.init(bool)
|
|
||||||
} else if let int = try? container.decode(Int.self) {
|
|
||||||
self.init(int)
|
|
||||||
} else if let uint = try? container.decode(UInt.self) {
|
|
||||||
self.init(uint)
|
|
||||||
} else if let double = try? container.decode(Double.self) {
|
|
||||||
self.init(double)
|
|
||||||
} else if let string = try? container.decode(String.self) {
|
|
||||||
self.init(string)
|
|
||||||
} else if let array = try? container.decode([AnyDecodable].self) {
|
|
||||||
self.init(array.map { $0.value })
|
|
||||||
} else if let dictionary = try? container.decode([String: AnyDecodable].self) {
|
|
||||||
self.init(dictionary.mapValues { $0.value })
|
|
||||||
} else {
|
|
||||||
throw DecodingError.dataCorruptedError(in: container, debugDescription: "AnyDecodable value cannot be decoded")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
extension AnyDecodable: Equatable {
|
|
||||||
public static func == (lhs: AnyDecodable, rhs: AnyDecodable) -> Bool {
|
|
||||||
switch (lhs.value, rhs.value) {
|
|
||||||
#if canImport(Foundation)
|
|
||||||
case is (NSNull, NSNull), is (Void, Void):
|
|
||||||
return true
|
|
||||||
#endif
|
|
||||||
case let (lhs as Bool, rhs as Bool):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Int, rhs as Int):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Int8, rhs as Int8):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Int16, rhs as Int16):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Int32, rhs as Int32):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Int64, rhs as Int64):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as UInt, rhs as UInt):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as UInt8, rhs as UInt8):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as UInt16, rhs as UInt16):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as UInt32, rhs as UInt32):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as UInt64, rhs as UInt64):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Float, rhs as Float):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Double, rhs as Double):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as String, rhs as String):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as [String: AnyDecodable], rhs as [String: AnyDecodable]):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as [AnyDecodable], rhs as [AnyDecodable]):
|
|
||||||
return lhs == rhs
|
|
||||||
default:
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
extension AnyDecodable: CustomStringConvertible {
|
|
||||||
public var description: String {
|
|
||||||
switch value {
|
|
||||||
case is Void:
|
|
||||||
return String(describing: nil as Any?)
|
|
||||||
case let value as CustomStringConvertible:
|
|
||||||
return value.description
|
|
||||||
default:
|
|
||||||
return String(describing: value)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
extension AnyDecodable: CustomDebugStringConvertible {
|
|
||||||
public var debugDescription: String {
|
|
||||||
if let value = value as? CustomDebugStringConvertible {
|
|
||||||
return "AnyDecodable(\(value.debugDescription))"
|
|
||||||
} else {
|
|
||||||
return "AnyDecodable(\(description))"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
extension AnyDecodable: Hashable {
|
|
||||||
public func hash(into hasher: inout Hasher) {
|
|
||||||
switch value {
|
|
||||||
case let value as Bool:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Int:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Int8:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Int16:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Int32:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Int64:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as UInt:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as UInt8:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as UInt16:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as UInt32:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as UInt64:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Float:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Double:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as String:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as [String: AnyDecodable]:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as [AnyDecodable]:
|
|
||||||
hasher.combine(value)
|
|
||||||
default:
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,292 +0,0 @@
|
|||||||
// https://github.com/Flight-School/AnyCodable/blob/master/Sources/AnyCodable/AnyCodable.swift
|
|
||||||
|
|
||||||
#if canImport(Foundation)
|
|
||||||
import Foundation
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/**
|
|
||||||
A type-erased `Encodable` value.
|
|
||||||
|
|
||||||
The `AnyEncodable` type forwards encoding responsibilities
|
|
||||||
to an underlying value, hiding its specific underlying type.
|
|
||||||
|
|
||||||
You can encode mixed-type values in dictionaries
|
|
||||||
and other collections that require `Encodable` conformance
|
|
||||||
by declaring their contained type to be `AnyEncodable`:
|
|
||||||
|
|
||||||
let dictionary: [String: AnyEncodable] = [
|
|
||||||
"boolean": true,
|
|
||||||
"integer": 42,
|
|
||||||
"double": 3.141592653589793,
|
|
||||||
"string": "string",
|
|
||||||
"array": [1, 2, 3],
|
|
||||||
"nested": [
|
|
||||||
"a": "alpha",
|
|
||||||
"b": "bravo",
|
|
||||||
"c": "charlie"
|
|
||||||
],
|
|
||||||
"null": nil
|
|
||||||
]
|
|
||||||
|
|
||||||
let encoder = JSONEncoder()
|
|
||||||
let json = try! encoder.encode(dictionary)
|
|
||||||
*/
|
|
||||||
struct AnyEncodable: Encodable {
|
|
||||||
public let value: Any
|
|
||||||
|
|
||||||
public init<T>(_ value: T?) {
|
|
||||||
self.value = value ?? ()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@usableFromInline
|
|
||||||
protocol AnyEncodableProtocol {
|
|
||||||
var value: Any { get }
|
|
||||||
init<T>(_ value: T?)
|
|
||||||
}
|
|
||||||
|
|
||||||
extension AnyEncodable: AnyEncodableProtocol {}
|
|
||||||
|
|
||||||
// MARK: - Encodable
|
|
||||||
|
|
||||||
extension AnyEncodableProtocol {
|
|
||||||
public func encode(to encoder: Encoder) throws {
|
|
||||||
var container = encoder.singleValueContainer()
|
|
||||||
|
|
||||||
switch value {
|
|
||||||
#if canImport(Foundation)
|
|
||||||
case is NSNull:
|
|
||||||
try container.encodeNil()
|
|
||||||
#endif
|
|
||||||
case is Void:
|
|
||||||
try container.encodeNil()
|
|
||||||
case let bool as Bool:
|
|
||||||
try container.encode(bool)
|
|
||||||
case let int as Int:
|
|
||||||
try container.encode(int)
|
|
||||||
case let int8 as Int8:
|
|
||||||
try container.encode(int8)
|
|
||||||
case let int16 as Int16:
|
|
||||||
try container.encode(int16)
|
|
||||||
case let int32 as Int32:
|
|
||||||
try container.encode(int32)
|
|
||||||
case let int64 as Int64:
|
|
||||||
try container.encode(int64)
|
|
||||||
case let uint as UInt:
|
|
||||||
try container.encode(uint)
|
|
||||||
case let uint8 as UInt8:
|
|
||||||
try container.encode(uint8)
|
|
||||||
case let uint16 as UInt16:
|
|
||||||
try container.encode(uint16)
|
|
||||||
case let uint32 as UInt32:
|
|
||||||
try container.encode(uint32)
|
|
||||||
case let uint64 as UInt64:
|
|
||||||
try container.encode(uint64)
|
|
||||||
case let float as Float:
|
|
||||||
try container.encode(float)
|
|
||||||
case let double as Double:
|
|
||||||
try container.encode(double)
|
|
||||||
case let string as String:
|
|
||||||
try container.encode(string)
|
|
||||||
#if canImport(Foundation)
|
|
||||||
case let number as NSNumber:
|
|
||||||
try encode(nsnumber: number, into: &container)
|
|
||||||
case let date as Date:
|
|
||||||
try container.encode(date)
|
|
||||||
case let url as URL:
|
|
||||||
try container.encode(url)
|
|
||||||
#endif
|
|
||||||
case let array as [Any?]:
|
|
||||||
try container.encode(array.map { AnyEncodable($0) })
|
|
||||||
case let dictionary as [String: Any?]:
|
|
||||||
try container.encode(dictionary.mapValues { AnyEncodable($0) })
|
|
||||||
case let encodable as Encodable:
|
|
||||||
try encodable.encode(to: encoder)
|
|
||||||
default:
|
|
||||||
let context = EncodingError.Context(codingPath: container.codingPath, debugDescription: "AnyEncodable value cannot be encoded")
|
|
||||||
throw EncodingError.invalidValue(value, context)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#if canImport(Foundation)
|
|
||||||
private func encode(nsnumber: NSNumber, into container: inout SingleValueEncodingContainer) throws {
|
|
||||||
switch Character(Unicode.Scalar(UInt8(nsnumber.objCType.pointee))) {
|
|
||||||
case "B":
|
|
||||||
try container.encode(nsnumber.boolValue)
|
|
||||||
case "c":
|
|
||||||
try container.encode(nsnumber.int8Value)
|
|
||||||
case "s":
|
|
||||||
try container.encode(nsnumber.int16Value)
|
|
||||||
case "i", "l":
|
|
||||||
try container.encode(nsnumber.int32Value)
|
|
||||||
case "q":
|
|
||||||
try container.encode(nsnumber.int64Value)
|
|
||||||
case "C":
|
|
||||||
try container.encode(nsnumber.uint8Value)
|
|
||||||
case "S":
|
|
||||||
try container.encode(nsnumber.uint16Value)
|
|
||||||
case "I", "L":
|
|
||||||
try container.encode(nsnumber.uint32Value)
|
|
||||||
case "Q":
|
|
||||||
try container.encode(nsnumber.uint64Value)
|
|
||||||
case "f":
|
|
||||||
try container.encode(nsnumber.floatValue)
|
|
||||||
case "d":
|
|
||||||
try container.encode(nsnumber.doubleValue)
|
|
||||||
default:
|
|
||||||
let context = EncodingError.Context(codingPath: container.codingPath, debugDescription: "NSNumber cannot be encoded because its type is not handled")
|
|
||||||
throw EncodingError.invalidValue(nsnumber, context)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
extension AnyEncodable: Equatable {
|
|
||||||
public static func == (lhs: AnyEncodable, rhs: AnyEncodable) -> Bool {
|
|
||||||
switch (lhs.value, rhs.value) {
|
|
||||||
case is (Void, Void):
|
|
||||||
return true
|
|
||||||
case let (lhs as Bool, rhs as Bool):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Int, rhs as Int):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Int8, rhs as Int8):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Int16, rhs as Int16):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Int32, rhs as Int32):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Int64, rhs as Int64):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as UInt, rhs as UInt):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as UInt8, rhs as UInt8):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as UInt16, rhs as UInt16):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as UInt32, rhs as UInt32):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as UInt64, rhs as UInt64):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Float, rhs as Float):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as Double, rhs as Double):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as String, rhs as String):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as [String: AnyEncodable], rhs as [String: AnyEncodable]):
|
|
||||||
return lhs == rhs
|
|
||||||
case let (lhs as [AnyEncodable], rhs as [AnyEncodable]):
|
|
||||||
return lhs == rhs
|
|
||||||
default:
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
extension AnyEncodable: CustomStringConvertible {
|
|
||||||
public var description: String {
|
|
||||||
switch value {
|
|
||||||
case is Void:
|
|
||||||
return String(describing: nil as Any?)
|
|
||||||
case let value as CustomStringConvertible:
|
|
||||||
return value.description
|
|
||||||
default:
|
|
||||||
return String(describing: value)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
extension AnyEncodable: CustomDebugStringConvertible {
|
|
||||||
public var debugDescription: String {
|
|
||||||
if let value = value as? CustomDebugStringConvertible {
|
|
||||||
return "AnyEncodable(\(value.debugDescription))"
|
|
||||||
} else {
|
|
||||||
return "AnyEncodable(\(description))"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
extension AnyEncodable: ExpressibleByNilLiteral {}
|
|
||||||
extension AnyEncodable: ExpressibleByBooleanLiteral {}
|
|
||||||
extension AnyEncodable: ExpressibleByIntegerLiteral {}
|
|
||||||
extension AnyEncodable: ExpressibleByFloatLiteral {}
|
|
||||||
extension AnyEncodable: ExpressibleByStringLiteral {}
|
|
||||||
extension AnyEncodable: ExpressibleByStringInterpolation {}
|
|
||||||
extension AnyEncodable: ExpressibleByArrayLiteral {}
|
|
||||||
extension AnyEncodable: ExpressibleByDictionaryLiteral {}
|
|
||||||
|
|
||||||
extension AnyEncodableProtocol {
|
|
||||||
public init(nilLiteral _: ()) {
|
|
||||||
self.init(nil as Any?)
|
|
||||||
}
|
|
||||||
|
|
||||||
public init(booleanLiteral value: Bool) {
|
|
||||||
self.init(value)
|
|
||||||
}
|
|
||||||
|
|
||||||
public init(integerLiteral value: Int) {
|
|
||||||
self.init(value)
|
|
||||||
}
|
|
||||||
|
|
||||||
public init(floatLiteral value: Double) {
|
|
||||||
self.init(value)
|
|
||||||
}
|
|
||||||
|
|
||||||
public init(extendedGraphemeClusterLiteral value: String) {
|
|
||||||
self.init(value)
|
|
||||||
}
|
|
||||||
|
|
||||||
public init(stringLiteral value: String) {
|
|
||||||
self.init(value)
|
|
||||||
}
|
|
||||||
|
|
||||||
public init(arrayLiteral elements: Any...) {
|
|
||||||
self.init(elements)
|
|
||||||
}
|
|
||||||
|
|
||||||
public init(dictionaryLiteral elements: (AnyHashable, Any)...) {
|
|
||||||
self.init([AnyHashable: Any](elements, uniquingKeysWith: { first, _ in first }))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
extension AnyEncodable: Hashable {
|
|
||||||
public func hash(into hasher: inout Hasher) {
|
|
||||||
switch value {
|
|
||||||
case let value as Bool:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Int:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Int8:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Int16:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Int32:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Int64:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as UInt:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as UInt8:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as UInt16:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as UInt32:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as UInt64:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Float:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as Double:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as String:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as [String: AnyEncodable]:
|
|
||||||
hasher.combine(value)
|
|
||||||
case let value as [AnyEncodable]:
|
|
||||||
hasher.combine(value)
|
|
||||||
default:
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -95,6 +95,9 @@ final class UserManager: UserManagerSyncable {
|
|||||||
self?.lastDisplayedAt = userResponse.data.state?.data?.lastDisplayAt
|
self?.lastDisplayedAt = userResponse.data.state?.data?.lastDisplayAt
|
||||||
self?.expiresAt = userResponse.data.state?.expiresAt
|
self?.expiresAt = userResponse.data.state?.expiresAt
|
||||||
|
|
||||||
|
let serverLanguage = userResponse.data.state?.data?.language
|
||||||
|
Formbricks.language = serverLanguage ?? "default"
|
||||||
|
|
||||||
self?.updateQueue?.reset()
|
self?.updateQueue?.reset()
|
||||||
self?.surveyManager?.filterSurveys()
|
self?.surveyManager?.filterSurveys()
|
||||||
self?.startSyncTimer()
|
self?.startSyncTimer()
|
||||||
@@ -128,6 +131,7 @@ final class UserManager: UserManagerSyncable {
|
|||||||
backingResponses = nil
|
backingResponses = nil
|
||||||
backingLastDisplayedAt = nil
|
backingLastDisplayedAt = nil
|
||||||
backingExpiresAt = nil
|
backingExpiresAt = nil
|
||||||
|
Formbricks.language = "default"
|
||||||
updateQueue?.reset()
|
updateQueue?.reset()
|
||||||
|
|
||||||
if isUserIdDefined {
|
if isUserIdDefined {
|
||||||
|
|||||||
@@ -7,4 +7,5 @@ struct UserStateDetails: Codable {
|
|||||||
let displays: [Display]?
|
let displays: [Display]?
|
||||||
let responses: [String]?
|
let responses: [String]?
|
||||||
let lastDisplayAt: Date?
|
let lastDisplayAt: Date?
|
||||||
|
let language: String?
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,131 +12,124 @@ class APIClient<Request: CodableRequest>: Operation, @unchecked Sendable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
override func main() {
|
override func main() {
|
||||||
guard let apiURL = request.baseURL, var baseUrlComponents = URLComponents(string: apiURL) else {
|
guard let finalURL = buildFinalURL() else {
|
||||||
completion?(.failure(FormbricksSDKError(type: .sdkIsNotInitialized)))
|
completion?(.failure(FormbricksSDKError(type: .sdkIsNotInitialized)))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
baseUrlComponents.queryItems = request.queryParams?.map { URLQueryItem(name: $0.key, value: $0.value) }
|
|
||||||
|
|
||||||
guard var finalURL = baseUrlComponents.url else {
|
|
||||||
completion?(.failure(FormbricksSDKError(type: .invalidAppUrl)))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
guard let requestEndPoint = setPathParams(request.requestEndPoint) else {
|
|
||||||
completion?(.failure(FormbricksSDKError(type: .sdkIsNotInitialized)))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
finalURL.appendPathComponent(requestEndPoint)
|
|
||||||
|
|
||||||
let urlRequest = createURLRequest(forURL: finalURL)
|
let urlRequest = createURLRequest(forURL: finalURL)
|
||||||
|
logRequest(urlRequest)
|
||||||
|
|
||||||
// LOG
|
session.dataTask(with: urlRequest) { data, response, error in
|
||||||
var requestLogMessage = "\(request.requestType.rawValue) >>> "
|
self.processResponse(data: data, response: response, error: error)
|
||||||
if let urlString = urlRequest.url?.absoluteString {
|
|
||||||
requestLogMessage.append(urlString)
|
|
||||||
}
|
|
||||||
if let headers = urlRequest.allHTTPHeaderFields {
|
|
||||||
requestLogMessage.append("\nHeaders: \(headers)")
|
|
||||||
}
|
|
||||||
if let body = urlRequest.httpBody {
|
|
||||||
requestLogMessage.append("\nBody: \(String(data: body, encoding: .utf8) ?? "")")
|
|
||||||
}
|
|
||||||
|
|
||||||
Formbricks.logger?.info(requestLogMessage)
|
|
||||||
|
|
||||||
session.dataTask(with: urlRequest) { (data, response, error) in
|
|
||||||
if let httpStatus = (response as? HTTPURLResponse)?.status {
|
|
||||||
var responseLogMessage = "\(httpStatus.rawValue) <<< "
|
|
||||||
if let urlString = response?.url?.absoluteString {
|
|
||||||
responseLogMessage.append(urlString)
|
|
||||||
}
|
|
||||||
|
|
||||||
if httpStatus.responseType == .success {
|
|
||||||
guard let data = data else {
|
|
||||||
self.completion?(.failure(FormbricksAPIClientError(type: .invalidResponse, statusCode: httpStatus.rawValue)))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if let responseString = String(data: data, encoding: .utf8) {
|
|
||||||
responseLogMessage.append("\n\(responseString)\n")
|
|
||||||
}
|
|
||||||
|
|
||||||
do {
|
|
||||||
if Request.Response.self == VoidResponse.self {
|
|
||||||
Formbricks.logger?.info(responseLogMessage)
|
|
||||||
|
|
||||||
if let response = VoidResponse() as? Request.Response {
|
|
||||||
self.completion?(.success(response))
|
|
||||||
} else {
|
|
||||||
self.completion?(.failure(FormbricksAPIClientError(type: .invalidResponse)))
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
var body = try self.request.decoder.decode(Request.Response.self, from: data)
|
|
||||||
Formbricks.logger?.info(responseLogMessage)
|
|
||||||
|
|
||||||
// We want to save the entire response dictionary for the environment response
|
|
||||||
if var environmentResponse = body as? EnvironmentResponse,
|
|
||||||
let jsonString = String(data: data, encoding: .utf8) {
|
|
||||||
environmentResponse.responseString = jsonString
|
|
||||||
body = environmentResponse as! Request.Response
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
self.completion?(.success(body))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch let DecodingError.dataCorrupted(context) {
|
|
||||||
responseLogMessage.append("Data corrupted \(context)\n")
|
|
||||||
Formbricks.logger?.error(responseLogMessage)
|
|
||||||
self.completion?(.failure(FormbricksAPIClientError(type: .invalidResponse, statusCode: httpStatus.rawValue)))
|
|
||||||
}
|
|
||||||
catch let DecodingError.keyNotFound(key, context) {
|
|
||||||
responseLogMessage.append("Key '\(key)' not found: \(context.debugDescription)\n")
|
|
||||||
responseLogMessage.append("codingPath: \(context.codingPath)")
|
|
||||||
Formbricks.logger?.error(responseLogMessage)
|
|
||||||
self.completion?(.failure(FormbricksAPIClientError(type: .invalidResponse, statusCode: httpStatus.rawValue)))
|
|
||||||
}
|
|
||||||
catch let DecodingError.valueNotFound(value, context) {
|
|
||||||
responseLogMessage.append("Value '\(value)' not found: \(context.debugDescription)\n")
|
|
||||||
responseLogMessage.append("codingPath: \(context.codingPath)")
|
|
||||||
Formbricks.logger?.error(responseLogMessage)
|
|
||||||
self.completion?(.failure(FormbricksAPIClientError(type: .invalidResponse, statusCode: httpStatus.rawValue)))
|
|
||||||
}
|
|
||||||
catch let DecodingError.typeMismatch(type, context) {
|
|
||||||
responseLogMessage.append("Type '\(type)' mismatch: \(context.debugDescription)\n")
|
|
||||||
responseLogMessage.append("codingPath: \(context.codingPath)")
|
|
||||||
Formbricks.logger?.error(responseLogMessage)
|
|
||||||
self.completion?(.failure(FormbricksAPIClientError(type: .invalidResponse, statusCode: httpStatus.rawValue)))
|
|
||||||
}
|
|
||||||
catch {
|
|
||||||
responseLogMessage.append("error: \(error.message)")
|
|
||||||
Formbricks.logger?.error(responseLogMessage)
|
|
||||||
self.completion?(.failure(FormbricksAPIClientError(type: .invalidResponse, statusCode: httpStatus.rawValue)))
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if let error = error {
|
|
||||||
responseLogMessage.append("\nError: \(error.localizedDescription)")
|
|
||||||
Formbricks.logger?.error(responseLogMessage)
|
|
||||||
self.completion?(.failure(error))
|
|
||||||
} else if let data = data, let apiError = try? self.request.decoder.decode(FormbricksAPIError.self, from: data) {
|
|
||||||
Formbricks.logger?.error("\(responseLogMessage)\n\(apiError.getDetailedErrorMessage())")
|
|
||||||
self.completion?(.failure(apiError))
|
|
||||||
} else {
|
|
||||||
let error = FormbricksAPIClientError(type: .responseError, statusCode: httpStatus.rawValue)
|
|
||||||
Formbricks.logger?.error("\(responseLogMessage)\n\(error.message)")
|
|
||||||
self.completion?(.failure(error))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
let error = FormbricksAPIClientError(type: .invalidResponse)
|
|
||||||
Formbricks.logger?.error("ERROR \(error.message)")
|
|
||||||
self.completion?(.failure(error))
|
|
||||||
}
|
|
||||||
}.resume()
|
}.resume()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private func buildFinalURL() -> URL? {
|
||||||
|
guard let apiURL = request.baseURL, var components = URLComponents(string: apiURL) else { return nil }
|
||||||
|
|
||||||
|
components.queryItems = request.queryParams?.map { URLQueryItem(name: $0.key, value: $0.value) }
|
||||||
|
|
||||||
|
guard var url = components.url, let path = setPathParams(request.requestEndPoint) else { return nil }
|
||||||
|
|
||||||
|
url.appendPathComponent(path)
|
||||||
|
return url
|
||||||
|
}
|
||||||
|
|
||||||
|
private func processResponse(data: Data?, response: URLResponse?, error: Error?) {
|
||||||
|
guard let httpStatus = (response as? HTTPURLResponse)?.status else {
|
||||||
|
let error = FormbricksAPIClientError(type: .invalidResponse)
|
||||||
|
Formbricks.logger?.error("ERROR \(error.message)")
|
||||||
|
completion?(.failure(error))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var message = "\(httpStatus.rawValue) <<< \(response?.url?.absoluteString ?? "")"
|
||||||
|
|
||||||
|
if httpStatus.responseType == .success {
|
||||||
|
handleSuccessResponse(data: data, statusCode: httpStatus.rawValue, message: &message)
|
||||||
|
} else {
|
||||||
|
handleFailureResponse(data: data, error: error, statusCode: httpStatus.rawValue, message: message)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private func handleSuccessResponse(data: Data?, statusCode: Int, message: inout String) {
|
||||||
|
guard let data = data else {
|
||||||
|
completion?(.failure(FormbricksAPIClientError(type: .invalidResponse, statusCode: statusCode)))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if let responseString = String(data: data, encoding: .utf8) {
|
||||||
|
message.append("\n\(responseString)\n")
|
||||||
|
}
|
||||||
|
|
||||||
|
do {
|
||||||
|
if Request.Response.self == VoidResponse.self {
|
||||||
|
Formbricks.logger?.info(message)
|
||||||
|
completion?(.success(VoidResponse() as! Request.Response))
|
||||||
|
} else {
|
||||||
|
var body = try request.decoder.decode(Request.Response.self, from: data)
|
||||||
|
if var env = body as? EnvironmentResponse, let jsonString = String(data: data, encoding: .utf8) {
|
||||||
|
env.responseString = jsonString
|
||||||
|
body = env as! Request.Response
|
||||||
|
}
|
||||||
|
Formbricks.logger?.info(message)
|
||||||
|
completion?(.success(body))
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
handleDecodingError(error, message: &message, statusCode: statusCode)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private func handleFailureResponse(data: Data?, error: Error?, statusCode: Int, message: String) {
|
||||||
|
var log = message
|
||||||
|
|
||||||
|
if let error = error {
|
||||||
|
log.append("\nError: \(error.localizedDescription)")
|
||||||
|
Formbricks.logger?.error(log)
|
||||||
|
completion?(.failure(error))
|
||||||
|
} else if let data = data, let apiError = try? request.decoder.decode(FormbricksAPIError.self, from: data) {
|
||||||
|
Formbricks.logger?.error("\(log)\n\(apiError.getDetailedErrorMessage())")
|
||||||
|
completion?(.failure(apiError))
|
||||||
|
} else {
|
||||||
|
let error = FormbricksAPIClientError(type: .responseError, statusCode: statusCode)
|
||||||
|
Formbricks.logger?.error("\(log)\n\(error.message)")
|
||||||
|
completion?(.failure(error))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private func handleDecodingError(_ error: Error, message: inout String, statusCode: Int) {
|
||||||
|
switch error {
|
||||||
|
case let DecodingError.dataCorrupted(context):
|
||||||
|
message.append("Data corrupted: \(context)")
|
||||||
|
case let DecodingError.keyNotFound(key, context):
|
||||||
|
message.append("Key '\(key)' not found: \(context.debugDescription)\ncodingPath: \(context.codingPath)")
|
||||||
|
case let DecodingError.valueNotFound(value, context):
|
||||||
|
message.append("Value '\(value)' not found: \(context.debugDescription)\ncodingPath: \(context.codingPath)")
|
||||||
|
case let DecodingError.typeMismatch(type, context):
|
||||||
|
message.append("Type '\(type)' mismatch: \(context.debugDescription)\ncodingPath: \(context.codingPath)")
|
||||||
|
default:
|
||||||
|
message.append("Error: \(error.localizedDescription)")
|
||||||
|
}
|
||||||
|
|
||||||
|
Formbricks.logger?.error(message)
|
||||||
|
completion?(.failure(FormbricksAPIClientError(type: .invalidResponse, statusCode: statusCode)))
|
||||||
|
}
|
||||||
|
|
||||||
|
private func logRequest(_ request: URLRequest) {
|
||||||
|
var message = "\(request.httpMethod ?? "") >>> \(request.url?.absoluteString ?? "")"
|
||||||
|
|
||||||
|
if let headers = request.allHTTPHeaderFields {
|
||||||
|
message.append("\nHeaders: \(headers)")
|
||||||
|
}
|
||||||
|
|
||||||
|
if let body = request.httpBody, let bodyString = String(data: body, encoding: .utf8) {
|
||||||
|
message.append("\nBody: \(bodyString)")
|
||||||
|
}
|
||||||
|
|
||||||
|
Formbricks.logger?.info(message)
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private extension APIClient {
|
private extension APIClient {
|
||||||
|
|||||||
@@ -61,6 +61,7 @@ final class UpdateQueue {
|
|||||||
} else {
|
} else {
|
||||||
// If no userId, just update locally without API call
|
// If no userId, just update locally without API call
|
||||||
Formbricks.logger?.debug("UpdateQueue - updating language locally: \(language)")
|
Formbricks.logger?.debug("UpdateQueue - updating language locally: \(language)")
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
startDebounceTimer()
|
startDebounceTimer()
|
||||||
|
|||||||
@@ -54,15 +54,19 @@ struct SurveyWebView: UIViewRepresentable {
|
|||||||
func clean() {
|
func clean() {
|
||||||
HTTPCookieStorage.shared.removeCookies(since: Date.distantPast)
|
HTTPCookieStorage.shared.removeCookies(since: Date.distantPast)
|
||||||
WKWebsiteDataStore.default().fetchDataRecords(ofTypes: WKWebsiteDataStore.allWebsiteDataTypes()) { records in
|
WKWebsiteDataStore.default().fetchDataRecords(ofTypes: WKWebsiteDataStore.allWebsiteDataTypes()) { records in
|
||||||
records.forEach { record in
|
self.remove(records)
|
||||||
WKWebsiteDataStore.default().removeData(ofTypes: record.dataTypes, for: [record], completionHandler: {
|
}
|
||||||
/*
|
}
|
||||||
This completion handler is intentionally empty since we only need to
|
|
||||||
ensure the data is removed. No additional actions are required after
|
private func remove(_ records: [WKWebsiteDataRecord]) {
|
||||||
the website data has been cleared.
|
records.forEach { record in
|
||||||
*/
|
WKWebsiteDataStore.default().removeData(ofTypes: record.dataTypes, for: [record], completionHandler: {
|
||||||
})
|
/*
|
||||||
}
|
This completion handler is intentionally empty since we only need to
|
||||||
|
ensure the data is removed. No additional actions are required after
|
||||||
|
the website data has been cleared.
|
||||||
|
*/
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user