diff --git a/Sources/ElixxirDAppsSDK/Callbacks/SingleUseCallback.swift b/Sources/ElixxirDAppsSDK/Callbacks/SingleUseCallback.swift
new file mode 100644
index 0000000000000000000000000000000000000000..99a4cf1f95be9328f7325aecaf852eb785cab414
--- /dev/null
+++ b/Sources/ElixxirDAppsSDK/Callbacks/SingleUseCallback.swift
@@ -0,0 +1,44 @@
+import Bindings
+import XCTestDynamicOverlay
+
+public struct SingleUseCallback {
+  public init(handle: @escaping (Result<SingleUseCallbackReport, NSError>) -> Void) {
+    self.handle = handle
+  }
+
+  public var handle: (Result<SingleUseCallbackReport, NSError>) -> Void
+}
+
+extension SingleUseCallback {
+  public static let unimplemented = SingleUseCallback(
+    handle: XCTUnimplemented("\(Self.self)")
+  )
+}
+
+extension SingleUseCallback {
+  func makeBindingsSingleUseCallback() -> BindingsSingleUseCallbackProtocol {
+    class CallbackObject: NSObject, BindingsSingleUseCallbackProtocol {
+      init(_ callback: SingleUseCallback) {
+        self.callback = callback
+      }
+
+      let callback: SingleUseCallback
+
+      func callback(_ callbackReport: Data?, err: Error?) {
+        if let error = err {
+          callback.handle(.failure(error as NSError))
+        } else if let callbackReport = callbackReport {
+          do {
+            callback.handle(.success(try SingleUseCallbackReport.decode(callbackReport)))
+          } catch {
+            callback.handle(.failure(error as NSError))
+          }
+        } else {
+          fatalError("BindingsSingleUseCallback received `nil` callbackReport and `nil` error")
+        }
+      }
+    }
+
+    return CallbackObject(self)
+  }
+}