This repository was archived by the owner on Oct 8, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 72
/
Copy pathRsyncAsync.swift
92 lines (87 loc) · 3.31 KB
/
RsyncAsync.swift
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
//
// RsyncAsync.swift
// RsyncUI
//
// Created by Thomas Evensen on 22/09/2022.
//
import Combine
import Foundation
@MainActor
final class RsyncAsync: Errors {
// Combine subscribers
var subscriptons = Set<AnyCancellable>()
// Verify network connection
// Arguments to command
var arguments: [String]?
// Process termination closure
var processtermination: ([String]?) -> Void
// Output
var outputprocess: OutputfromProcess?
// Enable and disable select profile
weak var profilepopupDelegate: DisableEnablePopupSelectProfile?
func executeProcess() async {
// Must check valid rsync exists
guard SharedReference.shared.norsync == false else { return }
// Process
let task = Process()
// Getting version of rsync
task.launchPath = Getrsyncpath().rsyncpath
task.arguments = arguments
// If there are any Environmentvariables like
// SSH_AUTH_SOCK": "/Users/user/.gnupg/S.gpg-agent.ssh"
if let environment = Environment() {
task.environment = environment.environment
}
// Pipe for reading output from Process
let pipe = Pipe()
task.standardOutput = pipe
task.standardError = pipe
let outHandle = pipe.fileHandleForReading
outHandle.waitForDataInBackgroundAndNotify()
// Combine, subscribe to NSNotification.Name.NSFileHandleDataAvailable
NotificationCenter.default.publisher(
for: NSNotification.Name.NSFileHandleDataAvailable)
.sink { _ in
let data = outHandle.availableData
if data.count > 0 {
if let str = NSString(data: data, encoding: String.Encoding.utf8.rawValue) {
self.outputprocess?.addlinefromoutput(str: str as String)
}
outHandle.waitForDataInBackgroundAndNotify()
}
}.store(in: &subscriptons)
// Combine, subscribe to Process.didTerminateNotification
NotificationCenter.default.publisher(
for: Process.didTerminateNotification)
.debounce(for: .milliseconds(500), scheduler: globalMainQueue)
.sink { _ in
// Logg to file
self.processtermination(self.outputprocess?.getOutput())
// Release Combine subscribers
// print("process termination")
self.subscriptons.removeAll()
}.store(in: &subscriptons)
profilepopupDelegate?.disableselectpopupprofile()
SharedReference.shared.process = task
do {
try task.run()
} catch let e {
let error = e
self.error(errordescription: error.localizedDescription, errortype: .task)
}
}
init(arguments: [String]?,
processtermination: @escaping ([String]?) -> Void)
{
self.arguments = arguments
self.processtermination = processtermination
outputprocess = OutputfromProcess()
profilepopupDelegate = SharedReference.shared.getvcref(viewcontroller: .vctabmain) as? ViewControllerMain
}
deinit {
SharedReference.shared.process = nil
// Enable select profile
self.profilepopupDelegate?.enableselectpopupprofile()
// print("deinit RsyncAsync")
}
}