Skip to content

Commit be80606

Browse files
committed
Diffusion app.
0 parents  commit be80606

24 files changed

+1711
-0
lines changed

Diffusion.xcodeproj/project.pbxproj

+734
Large diffs are not rendered by default.

Diffusion.xcodeproj/project.xcworkspace/contents.xcworkspacedata

+7
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
<?xml version="1.0" encoding="UTF-8"?>
2+
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
3+
<plist version="1.0">
4+
<dict>
5+
<key>IDEDidComputeMac32BitWarning</key>
6+
<true/>
7+
</dict>
8+
</plist>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
{
2+
"colors" : [
3+
{
4+
"idiom" : "universal"
5+
}
6+
],
7+
"info" : {
8+
"author" : "xcode",
9+
"version" : 1
10+
}
11+
}
Binary file not shown.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,65 @@
1+
{
2+
"images" : [
3+
{
4+
"filename" : "labrador-x2.png",
5+
"idiom" : "universal",
6+
"platform" : "ios",
7+
"size" : "1024x1024"
8+
},
9+
{
10+
"idiom" : "mac",
11+
"scale" : "1x",
12+
"size" : "16x16"
13+
},
14+
{
15+
"idiom" : "mac",
16+
"scale" : "2x",
17+
"size" : "16x16"
18+
},
19+
{
20+
"idiom" : "mac",
21+
"scale" : "1x",
22+
"size" : "32x32"
23+
},
24+
{
25+
"idiom" : "mac",
26+
"scale" : "2x",
27+
"size" : "32x32"
28+
},
29+
{
30+
"idiom" : "mac",
31+
"scale" : "1x",
32+
"size" : "128x128"
33+
},
34+
{
35+
"idiom" : "mac",
36+
"scale" : "2x",
37+
"size" : "128x128"
38+
},
39+
{
40+
"idiom" : "mac",
41+
"scale" : "1x",
42+
"size" : "256x256"
43+
},
44+
{
45+
"idiom" : "mac",
46+
"scale" : "2x",
47+
"size" : "256x256"
48+
},
49+
{
50+
"idiom" : "mac",
51+
"scale" : "1x",
52+
"size" : "512x512"
53+
},
54+
{
55+
"filename" : "labrador-x2 1.png",
56+
"idiom" : "mac",
57+
"scale" : "2x",
58+
"size" : "512x512"
59+
}
60+
],
61+
"info" : {
62+
"author" : "xcode",
63+
"version" : 1
64+
}
65+
}
Loading
Loading
+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
{
2+
"info" : {
3+
"author" : "xcode",
4+
"version" : 1
5+
}
6+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
{
2+
"images" : [
3+
{
4+
"idiom" : "universal",
5+
"scale" : "1x"
6+
},
7+
{
8+
"filename" : "labrador.png",
9+
"idiom" : "universal",
10+
"scale" : "2x"
11+
},
12+
{
13+
"idiom" : "universal",
14+
"scale" : "3x"
15+
}
16+
],
17+
"info" : {
18+
"author" : "xcode",
19+
"version" : 1
20+
}
21+
}
Loading

Diffusion/Diffusion.entitlements

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
<?xml version="1.0" encoding="UTF-8"?>
2+
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
3+
<plist version="1.0">
4+
<dict>
5+
<key>com.apple.security.app-sandbox</key>
6+
<true/>
7+
<key>com.apple.security.files.user-selected.read-only</key>
8+
<true/>
9+
<key>com.apple.security.network.client</key>
10+
<true/>
11+
</dict>
12+
</plist>

Diffusion/DiffusionApp.swift

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
//
2+
// DiffusionApp.swift
3+
// Diffusion
4+
//
5+
// Created by Pedro Cuenca on December 2022.
6+
// See LICENSE at https://github.com/huggingface/swift-coreml-diffusers/LICENSE
7+
//
8+
9+
import SwiftUI
10+
11+
@main
12+
struct DiffusionApp: App {
13+
var body: some Scene {
14+
WindowGroup {
15+
LoadingView()
16+
}
17+
}
18+
}
19+
20+
extension String: Error {}

Diffusion/Downloader.swift

+91
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,91 @@
1+
//
2+
// Downloader.swift
3+
// Diffusion
4+
//
5+
// Created by Pedro Cuenca on December 2022.
6+
// See LICENSE at https://github.com/huggingface/swift-coreml-diffusers/LICENSE
7+
//
8+
9+
import Foundation
10+
import Combine
11+
import Path
12+
13+
class Downloader: NSObject, ObservableObject {
14+
private(set) var destination: URL
15+
16+
enum DownloadState {
17+
case notStarted
18+
case downloading(Double)
19+
case completed(URL)
20+
case failed(Error)
21+
}
22+
23+
private(set) lazy var downloadState: CurrentValueSubject<DownloadState, Never> = CurrentValueSubject(.notStarted)
24+
private var stateSubscriber: Cancellable?
25+
26+
init(from url: URL, to destination: URL) {
27+
self.destination = destination
28+
super.init()
29+
30+
// .background allows downloads to proceed in the background
31+
let config = URLSessionConfiguration.background(withIdentifier: "net.pcuenca.diffusion.download")
32+
let urlSession = URLSession(configuration: config, delegate: self, delegateQueue: OperationQueue())
33+
downloadState.value = .downloading(0)
34+
urlSession.getAllTasks { tasks in
35+
// If there's an existing pending background task, let it proceed, otherwise start a new one.
36+
// TODO: check URL when we support downloading more models.
37+
if tasks.first == nil {
38+
urlSession.downloadTask(with: url).resume()
39+
}
40+
}
41+
}
42+
43+
@discardableResult
44+
func waitUntilDone() throws -> URL {
45+
// It's either this, or stream the bytes ourselves (add to a buffer, save to disk, etc; boring and finicky)
46+
let semaphore = DispatchSemaphore(value: 0)
47+
stateSubscriber = downloadState.sink { state in
48+
switch state {
49+
case .completed: semaphore.signal()
50+
case .failed: semaphore.signal()
51+
default: break
52+
}
53+
}
54+
semaphore.wait()
55+
56+
switch downloadState.value {
57+
case .completed(let url): return url
58+
case .failed(let error): throw error
59+
default: throw("Should never happen, lol")
60+
}
61+
}
62+
}
63+
64+
extension Downloader: URLSessionDelegate, URLSessionDownloadDelegate {
65+
func urlSession(_: URLSession, downloadTask: URLSessionDownloadTask, didWriteData _: Int64, totalBytesWritten _: Int64, totalBytesExpectedToWrite _: Int64) {
66+
downloadState.value = .downloading(downloadTask.progress.fractionCompleted)
67+
}
68+
69+
func urlSession(_: URLSession, downloadTask _: URLSessionDownloadTask, didFinishDownloadingTo location: URL) {
70+
guard let path = Path(url: location) else {
71+
downloadState.value = .failed("Invalid download location received: \(location)")
72+
return
73+
}
74+
guard let toPath = Path(url: destination) else {
75+
downloadState.value = .failed("Invalid destination: \(destination)")
76+
return
77+
}
78+
do {
79+
try path.move(to: toPath, overwrite: true)
80+
downloadState.value = .completed(destination)
81+
} catch {
82+
downloadState.value = .failed(error)
83+
}
84+
}
85+
86+
func urlSession(_: URLSession, task: URLSessionTask, didCompleteWithError error: Error?) {
87+
if let error = error {
88+
downloadState.value = .failed(error)
89+
}
90+
}
91+
}

Diffusion/Pipeline/Pipeline.swift

+56
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,56 @@
1+
//
2+
// Pipeline.swift
3+
// Diffusion
4+
//
5+
// Created by Pedro Cuenca on December 2022.
6+
// See LICENSE at https://github.com/huggingface/swift-coreml-diffusers/LICENSE
7+
//
8+
9+
import Foundation
10+
import CoreML
11+
import Combine
12+
13+
import StableDiffusion
14+
15+
typealias StableDiffusionProgress = StableDiffusionPipeline.Progress
16+
17+
class Pipeline {
18+
let pipeline: StableDiffusionPipeline
19+
20+
var progress: StableDiffusionProgress? = nil {
21+
didSet {
22+
progressPublisher.value = progress
23+
}
24+
}
25+
lazy private(set) var progressPublisher: CurrentValueSubject<StableDiffusionProgress?, Never> = CurrentValueSubject(progress)
26+
27+
28+
init(_ pipeline: StableDiffusionPipeline) {
29+
self.pipeline = pipeline
30+
}
31+
32+
func generate(prompt: String, scheduler: StableDiffusionScheduler, numInferenceSteps stepCount: Int = 50, seed: UInt32? = nil) throws -> CGImage {
33+
let beginDate = Date()
34+
print("Generating...")
35+
let theSeed = seed ?? UInt32.random(in: 0..<UInt32.max)
36+
let images = try pipeline.generateImages(
37+
prompt: prompt,
38+
imageCount: 1,
39+
stepCount: stepCount,
40+
seed: theSeed,
41+
scheduler: scheduler
42+
) { progress in
43+
handleProgress(progress)
44+
return true
45+
}
46+
print("Got images: \(images) in \(Date().timeIntervalSince(beginDate))")
47+
48+
// unwrap the 1 image we asked for
49+
guard let image = images.compactMap({ $0 }).first else { throw "Generation failed" }
50+
return image
51+
}
52+
53+
func handleProgress(_ progress: StableDiffusionPipeline.Progress) {
54+
self.progress = progress
55+
}
56+
}

0 commit comments

Comments
 (0)