Changed PGP backend from ObjectivePGP to GopenPGP
This commit is contained in:
parent
5439ad0f5b
commit
96f7c1960d
27 changed files with 647 additions and 193 deletions
1
Podfile
1
Podfile
|
|
@ -2,7 +2,6 @@ platform :ios, '10.2'
|
||||||
use_frameworks!
|
use_frameworks!
|
||||||
|
|
||||||
target 'passKit' do
|
target 'passKit' do
|
||||||
pod 'ObjectivePGP', :git => 'https://github.com/krzyzanowskim/ObjectivePGP.git', :tag => '0.14.0'
|
|
||||||
target 'pass' do
|
target 'pass' do
|
||||||
inherit! :search_paths
|
inherit! :search_paths
|
||||||
end
|
end
|
||||||
|
|
|
||||||
BIN
go/dist/Gopenpgpwrapper.framework/Gopenpgpwrapper
vendored
Normal file
BIN
go/dist/Gopenpgpwrapper.framework/Gopenpgpwrapper
vendored
Normal file
Binary file not shown.
13
go/dist/Gopenpgpwrapper.framework/Headers/Gopenpgpwrapper.h
vendored
Normal file
13
go/dist/Gopenpgpwrapper.framework/Headers/Gopenpgpwrapper.h
vendored
Normal file
|
|
@ -0,0 +1,13 @@
|
||||||
|
|
||||||
|
// Objective-C API for talking to the following Go packages
|
||||||
|
//
|
||||||
|
// github.com/ZortacDev/GopenPGPWrapper/go/gopenpgpwrapper
|
||||||
|
//
|
||||||
|
// File is generated by gomobile bind. Do not edit.
|
||||||
|
#ifndef __Gopenpgpwrapper_FRAMEWORK_H__
|
||||||
|
#define __Gopenpgpwrapper_FRAMEWORK_H__
|
||||||
|
|
||||||
|
#include "Gopenpgpwrapper.objc.h"
|
||||||
|
#include "Universe.objc.h"
|
||||||
|
|
||||||
|
#endif
|
||||||
29
go/dist/Gopenpgpwrapper.framework/Headers/Gopenpgpwrapper.objc.h
vendored
Normal file
29
go/dist/Gopenpgpwrapper.framework/Headers/Gopenpgpwrapper.objc.h
vendored
Normal file
|
|
@ -0,0 +1,29 @@
|
||||||
|
// Objective-C API for talking to github.com/ZortacDev/GopenPGPWrapper/go/gopenpgpwrapper Go package.
|
||||||
|
// gobind -lang=objc github.com/ZortacDev/GopenPGPWrapper/go/gopenpgpwrapper
|
||||||
|
//
|
||||||
|
// File is generated by gobind. Do not edit.
|
||||||
|
|
||||||
|
#ifndef __Gopenpgpwrapper_H__
|
||||||
|
#define __Gopenpgpwrapper_H__
|
||||||
|
|
||||||
|
@import Foundation;
|
||||||
|
#include "ref.h"
|
||||||
|
#include "Universe.objc.h"
|
||||||
|
|
||||||
|
|
||||||
|
@class GopenpgpwrapperKey;
|
||||||
|
|
||||||
|
@interface GopenpgpwrapperKey : NSObject <goSeqRefInterface> {
|
||||||
|
}
|
||||||
|
@property(strong, readonly) _Nonnull id _ref;
|
||||||
|
|
||||||
|
- (nonnull instancetype)initWithRef:(_Nonnull id)ref;
|
||||||
|
- (nonnull instancetype)init;
|
||||||
|
- (NSData* _Nullable)decrypt:(NSData* _Nullable)ciphertext passphrase:(NSString* _Nullable)passphrase;
|
||||||
|
- (NSData* _Nullable)encrypt:(NSData* _Nullable)plaintext armor:(BOOL)armor;
|
||||||
|
- (NSString* _Nonnull)getKeyID;
|
||||||
|
@end
|
||||||
|
|
||||||
|
FOUNDATION_EXPORT GopenpgpwrapperKey* _Nullable GopenpgpwrapperReadKey(NSData* _Nullable data);
|
||||||
|
|
||||||
|
#endif
|
||||||
29
go/dist/Gopenpgpwrapper.framework/Headers/Universe.objc.h
vendored
Normal file
29
go/dist/Gopenpgpwrapper.framework/Headers/Universe.objc.h
vendored
Normal file
|
|
@ -0,0 +1,29 @@
|
||||||
|
// Objective-C API for talking to Go package.
|
||||||
|
// gobind -lang=objc
|
||||||
|
//
|
||||||
|
// File is generated by gobind. Do not edit.
|
||||||
|
|
||||||
|
#ifndef __Universe_H__
|
||||||
|
#define __Universe_H__
|
||||||
|
|
||||||
|
@import Foundation;
|
||||||
|
#include "ref.h"
|
||||||
|
|
||||||
|
@protocol Universeerror;
|
||||||
|
@class Universeerror;
|
||||||
|
|
||||||
|
@protocol Universeerror <NSObject>
|
||||||
|
- (NSString* _Nonnull)error;
|
||||||
|
@end
|
||||||
|
|
||||||
|
@class Universeerror;
|
||||||
|
|
||||||
|
@interface Universeerror : NSError <goSeqRefInterface, Universeerror> {
|
||||||
|
}
|
||||||
|
@property(strong, readonly) _Nonnull id _ref;
|
||||||
|
|
||||||
|
- (nonnull instancetype)initWithRef:(_Nonnull id)ref;
|
||||||
|
- (NSString* _Nonnull)error;
|
||||||
|
@end
|
||||||
|
|
||||||
|
#endif
|
||||||
35
go/dist/Gopenpgpwrapper.framework/Headers/ref.h
vendored
Normal file
35
go/dist/Gopenpgpwrapper.framework/Headers/ref.h
vendored
Normal file
|
|
@ -0,0 +1,35 @@
|
||||||
|
// Copyright 2015 The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
#ifndef __GO_REF_HDR__
|
||||||
|
#define __GO_REF_HDR__
|
||||||
|
|
||||||
|
#include <Foundation/Foundation.h>
|
||||||
|
|
||||||
|
// GoSeqRef is an object tagged with an integer for passing back and
|
||||||
|
// forth across the language boundary. A GoSeqRef may represent either
|
||||||
|
// an instance of a Go object, or an Objective-C object passed to Go.
|
||||||
|
// The explicit allocation of a GoSeqRef is used to pin a Go object
|
||||||
|
// when it is passed to Objective-C. The Go seq package maintains a
|
||||||
|
// reference to the Go object in a map keyed by the refnum along with
|
||||||
|
// a reference count. When the reference count reaches zero, the Go
|
||||||
|
// seq package will clear the corresponding entry in the map.
|
||||||
|
@interface GoSeqRef : NSObject {
|
||||||
|
}
|
||||||
|
@property(readonly) int32_t refnum;
|
||||||
|
@property(strong) id obj; // NULL when representing a Go object.
|
||||||
|
|
||||||
|
// new GoSeqRef object to proxy a Go object. The refnum must be
|
||||||
|
// provided from Go side.
|
||||||
|
- (instancetype)initWithRefnum:(int32_t)refnum obj:(id)obj;
|
||||||
|
|
||||||
|
- (int32_t)incNum;
|
||||||
|
|
||||||
|
@end
|
||||||
|
|
||||||
|
@protocol goSeqRefInterface
|
||||||
|
-(GoSeqRef*) _ref;
|
||||||
|
@end
|
||||||
|
|
||||||
|
#endif
|
||||||
8
go/dist/Gopenpgpwrapper.framework/Modules/module.modulemap
vendored
Normal file
8
go/dist/Gopenpgpwrapper.framework/Modules/module.modulemap
vendored
Normal file
|
|
@ -0,0 +1,8 @@
|
||||||
|
framework module "Gopenpgpwrapper" {
|
||||||
|
header "ref.h"
|
||||||
|
header "Gopenpgpwrapper.objc.h"
|
||||||
|
header "Universe.objc.h"
|
||||||
|
header "Gopenpgpwrapper.h"
|
||||||
|
|
||||||
|
export *
|
||||||
|
}
|
||||||
6
go/dist/Gopenpgpwrapper.framework/Resources/Info.plist
vendored
Normal file
6
go/dist/Gopenpgpwrapper.framework/Resources/Info.plist
vendored
Normal file
|
|
@ -0,0 +1,6 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
||||||
BIN
go/dist/Gopenpgpwrapper.framework/Versions/A/Gopenpgpwrapper
vendored
Normal file
BIN
go/dist/Gopenpgpwrapper.framework/Versions/A/Gopenpgpwrapper
vendored
Normal file
Binary file not shown.
13
go/dist/Gopenpgpwrapper.framework/Versions/A/Headers/Gopenpgpwrapper.h
vendored
Normal file
13
go/dist/Gopenpgpwrapper.framework/Versions/A/Headers/Gopenpgpwrapper.h
vendored
Normal file
|
|
@ -0,0 +1,13 @@
|
||||||
|
|
||||||
|
// Objective-C API for talking to the following Go packages
|
||||||
|
//
|
||||||
|
// github.com/ZortacDev/GopenPGPWrapper/go/gopenpgpwrapper
|
||||||
|
//
|
||||||
|
// File is generated by gomobile bind. Do not edit.
|
||||||
|
#ifndef __Gopenpgpwrapper_FRAMEWORK_H__
|
||||||
|
#define __Gopenpgpwrapper_FRAMEWORK_H__
|
||||||
|
|
||||||
|
#include "Gopenpgpwrapper.objc.h"
|
||||||
|
#include "Universe.objc.h"
|
||||||
|
|
||||||
|
#endif
|
||||||
29
go/dist/Gopenpgpwrapper.framework/Versions/A/Headers/Gopenpgpwrapper.objc.h
vendored
Normal file
29
go/dist/Gopenpgpwrapper.framework/Versions/A/Headers/Gopenpgpwrapper.objc.h
vendored
Normal file
|
|
@ -0,0 +1,29 @@
|
||||||
|
// Objective-C API for talking to github.com/ZortacDev/GopenPGPWrapper/go/gopenpgpwrapper Go package.
|
||||||
|
// gobind -lang=objc github.com/ZortacDev/GopenPGPWrapper/go/gopenpgpwrapper
|
||||||
|
//
|
||||||
|
// File is generated by gobind. Do not edit.
|
||||||
|
|
||||||
|
#ifndef __Gopenpgpwrapper_H__
|
||||||
|
#define __Gopenpgpwrapper_H__
|
||||||
|
|
||||||
|
@import Foundation;
|
||||||
|
#include "ref.h"
|
||||||
|
#include "Universe.objc.h"
|
||||||
|
|
||||||
|
|
||||||
|
@class GopenpgpwrapperKey;
|
||||||
|
|
||||||
|
@interface GopenpgpwrapperKey : NSObject <goSeqRefInterface> {
|
||||||
|
}
|
||||||
|
@property(strong, readonly) _Nonnull id _ref;
|
||||||
|
|
||||||
|
- (nonnull instancetype)initWithRef:(_Nonnull id)ref;
|
||||||
|
- (nonnull instancetype)init;
|
||||||
|
- (NSData* _Nullable)decrypt:(NSData* _Nullable)ciphertext passphrase:(NSString* _Nullable)passphrase;
|
||||||
|
- (NSData* _Nullable)encrypt:(NSData* _Nullable)plaintext armor:(BOOL)armor;
|
||||||
|
- (NSString* _Nonnull)getKeyID;
|
||||||
|
@end
|
||||||
|
|
||||||
|
FOUNDATION_EXPORT GopenpgpwrapperKey* _Nullable GopenpgpwrapperReadKey(NSData* _Nullable data);
|
||||||
|
|
||||||
|
#endif
|
||||||
29
go/dist/Gopenpgpwrapper.framework/Versions/A/Headers/Universe.objc.h
vendored
Normal file
29
go/dist/Gopenpgpwrapper.framework/Versions/A/Headers/Universe.objc.h
vendored
Normal file
|
|
@ -0,0 +1,29 @@
|
||||||
|
// Objective-C API for talking to Go package.
|
||||||
|
// gobind -lang=objc
|
||||||
|
//
|
||||||
|
// File is generated by gobind. Do not edit.
|
||||||
|
|
||||||
|
#ifndef __Universe_H__
|
||||||
|
#define __Universe_H__
|
||||||
|
|
||||||
|
@import Foundation;
|
||||||
|
#include "ref.h"
|
||||||
|
|
||||||
|
@protocol Universeerror;
|
||||||
|
@class Universeerror;
|
||||||
|
|
||||||
|
@protocol Universeerror <NSObject>
|
||||||
|
- (NSString* _Nonnull)error;
|
||||||
|
@end
|
||||||
|
|
||||||
|
@class Universeerror;
|
||||||
|
|
||||||
|
@interface Universeerror : NSError <goSeqRefInterface, Universeerror> {
|
||||||
|
}
|
||||||
|
@property(strong, readonly) _Nonnull id _ref;
|
||||||
|
|
||||||
|
- (nonnull instancetype)initWithRef:(_Nonnull id)ref;
|
||||||
|
- (NSString* _Nonnull)error;
|
||||||
|
@end
|
||||||
|
|
||||||
|
#endif
|
||||||
35
go/dist/Gopenpgpwrapper.framework/Versions/A/Headers/ref.h
vendored
Normal file
35
go/dist/Gopenpgpwrapper.framework/Versions/A/Headers/ref.h
vendored
Normal file
|
|
@ -0,0 +1,35 @@
|
||||||
|
// Copyright 2015 The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
#ifndef __GO_REF_HDR__
|
||||||
|
#define __GO_REF_HDR__
|
||||||
|
|
||||||
|
#include <Foundation/Foundation.h>
|
||||||
|
|
||||||
|
// GoSeqRef is an object tagged with an integer for passing back and
|
||||||
|
// forth across the language boundary. A GoSeqRef may represent either
|
||||||
|
// an instance of a Go object, or an Objective-C object passed to Go.
|
||||||
|
// The explicit allocation of a GoSeqRef is used to pin a Go object
|
||||||
|
// when it is passed to Objective-C. The Go seq package maintains a
|
||||||
|
// reference to the Go object in a map keyed by the refnum along with
|
||||||
|
// a reference count. When the reference count reaches zero, the Go
|
||||||
|
// seq package will clear the corresponding entry in the map.
|
||||||
|
@interface GoSeqRef : NSObject {
|
||||||
|
}
|
||||||
|
@property(readonly) int32_t refnum;
|
||||||
|
@property(strong) id obj; // NULL when representing a Go object.
|
||||||
|
|
||||||
|
// new GoSeqRef object to proxy a Go object. The refnum must be
|
||||||
|
// provided from Go side.
|
||||||
|
- (instancetype)initWithRefnum:(int32_t)refnum obj:(id)obj;
|
||||||
|
|
||||||
|
- (int32_t)incNum;
|
||||||
|
|
||||||
|
@end
|
||||||
|
|
||||||
|
@protocol goSeqRefInterface
|
||||||
|
-(GoSeqRef*) _ref;
|
||||||
|
@end
|
||||||
|
|
||||||
|
#endif
|
||||||
8
go/dist/Gopenpgpwrapper.framework/Versions/A/Modules/module.modulemap
vendored
Normal file
8
go/dist/Gopenpgpwrapper.framework/Versions/A/Modules/module.modulemap
vendored
Normal file
|
|
@ -0,0 +1,8 @@
|
||||||
|
framework module "Gopenpgpwrapper" {
|
||||||
|
header "ref.h"
|
||||||
|
header "Gopenpgpwrapper.objc.h"
|
||||||
|
header "Universe.objc.h"
|
||||||
|
header "Gopenpgpwrapper.h"
|
||||||
|
|
||||||
|
export *
|
||||||
|
}
|
||||||
6
go/dist/Gopenpgpwrapper.framework/Versions/A/Resources/Info.plist
vendored
Normal file
6
go/dist/Gopenpgpwrapper.framework/Versions/A/Resources/Info.plist
vendored
Normal file
|
|
@ -0,0 +1,6 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
||||||
BIN
go/dist/Gopenpgpwrapper.framework/Versions/Current/Gopenpgpwrapper
vendored
Normal file
BIN
go/dist/Gopenpgpwrapper.framework/Versions/Current/Gopenpgpwrapper
vendored
Normal file
Binary file not shown.
13
go/dist/Gopenpgpwrapper.framework/Versions/Current/Headers/Gopenpgpwrapper.h
vendored
Normal file
13
go/dist/Gopenpgpwrapper.framework/Versions/Current/Headers/Gopenpgpwrapper.h
vendored
Normal file
|
|
@ -0,0 +1,13 @@
|
||||||
|
|
||||||
|
// Objective-C API for talking to the following Go packages
|
||||||
|
//
|
||||||
|
// github.com/ZortacDev/GopenPGPWrapper/go/gopenpgpwrapper
|
||||||
|
//
|
||||||
|
// File is generated by gomobile bind. Do not edit.
|
||||||
|
#ifndef __Gopenpgpwrapper_FRAMEWORK_H__
|
||||||
|
#define __Gopenpgpwrapper_FRAMEWORK_H__
|
||||||
|
|
||||||
|
#include "Gopenpgpwrapper.objc.h"
|
||||||
|
#include "Universe.objc.h"
|
||||||
|
|
||||||
|
#endif
|
||||||
29
go/dist/Gopenpgpwrapper.framework/Versions/Current/Headers/Gopenpgpwrapper.objc.h
vendored
Normal file
29
go/dist/Gopenpgpwrapper.framework/Versions/Current/Headers/Gopenpgpwrapper.objc.h
vendored
Normal file
|
|
@ -0,0 +1,29 @@
|
||||||
|
// Objective-C API for talking to github.com/ZortacDev/GopenPGPWrapper/go/gopenpgpwrapper Go package.
|
||||||
|
// gobind -lang=objc github.com/ZortacDev/GopenPGPWrapper/go/gopenpgpwrapper
|
||||||
|
//
|
||||||
|
// File is generated by gobind. Do not edit.
|
||||||
|
|
||||||
|
#ifndef __Gopenpgpwrapper_H__
|
||||||
|
#define __Gopenpgpwrapper_H__
|
||||||
|
|
||||||
|
@import Foundation;
|
||||||
|
#include "ref.h"
|
||||||
|
#include "Universe.objc.h"
|
||||||
|
|
||||||
|
|
||||||
|
@class GopenpgpwrapperKey;
|
||||||
|
|
||||||
|
@interface GopenpgpwrapperKey : NSObject <goSeqRefInterface> {
|
||||||
|
}
|
||||||
|
@property(strong, readonly) _Nonnull id _ref;
|
||||||
|
|
||||||
|
- (nonnull instancetype)initWithRef:(_Nonnull id)ref;
|
||||||
|
- (nonnull instancetype)init;
|
||||||
|
- (NSData* _Nullable)decrypt:(NSData* _Nullable)ciphertext passphrase:(NSString* _Nullable)passphrase;
|
||||||
|
- (NSData* _Nullable)encrypt:(NSData* _Nullable)plaintext armor:(BOOL)armor;
|
||||||
|
- (NSString* _Nonnull)getKeyID;
|
||||||
|
@end
|
||||||
|
|
||||||
|
FOUNDATION_EXPORT GopenpgpwrapperKey* _Nullable GopenpgpwrapperReadKey(NSData* _Nullable data);
|
||||||
|
|
||||||
|
#endif
|
||||||
29
go/dist/Gopenpgpwrapper.framework/Versions/Current/Headers/Universe.objc.h
vendored
Normal file
29
go/dist/Gopenpgpwrapper.framework/Versions/Current/Headers/Universe.objc.h
vendored
Normal file
|
|
@ -0,0 +1,29 @@
|
||||||
|
// Objective-C API for talking to Go package.
|
||||||
|
// gobind -lang=objc
|
||||||
|
//
|
||||||
|
// File is generated by gobind. Do not edit.
|
||||||
|
|
||||||
|
#ifndef __Universe_H__
|
||||||
|
#define __Universe_H__
|
||||||
|
|
||||||
|
@import Foundation;
|
||||||
|
#include "ref.h"
|
||||||
|
|
||||||
|
@protocol Universeerror;
|
||||||
|
@class Universeerror;
|
||||||
|
|
||||||
|
@protocol Universeerror <NSObject>
|
||||||
|
- (NSString* _Nonnull)error;
|
||||||
|
@end
|
||||||
|
|
||||||
|
@class Universeerror;
|
||||||
|
|
||||||
|
@interface Universeerror : NSError <goSeqRefInterface, Universeerror> {
|
||||||
|
}
|
||||||
|
@property(strong, readonly) _Nonnull id _ref;
|
||||||
|
|
||||||
|
- (nonnull instancetype)initWithRef:(_Nonnull id)ref;
|
||||||
|
- (NSString* _Nonnull)error;
|
||||||
|
@end
|
||||||
|
|
||||||
|
#endif
|
||||||
35
go/dist/Gopenpgpwrapper.framework/Versions/Current/Headers/ref.h
vendored
Normal file
35
go/dist/Gopenpgpwrapper.framework/Versions/Current/Headers/ref.h
vendored
Normal file
|
|
@ -0,0 +1,35 @@
|
||||||
|
// Copyright 2015 The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
#ifndef __GO_REF_HDR__
|
||||||
|
#define __GO_REF_HDR__
|
||||||
|
|
||||||
|
#include <Foundation/Foundation.h>
|
||||||
|
|
||||||
|
// GoSeqRef is an object tagged with an integer for passing back and
|
||||||
|
// forth across the language boundary. A GoSeqRef may represent either
|
||||||
|
// an instance of a Go object, or an Objective-C object passed to Go.
|
||||||
|
// The explicit allocation of a GoSeqRef is used to pin a Go object
|
||||||
|
// when it is passed to Objective-C. The Go seq package maintains a
|
||||||
|
// reference to the Go object in a map keyed by the refnum along with
|
||||||
|
// a reference count. When the reference count reaches zero, the Go
|
||||||
|
// seq package will clear the corresponding entry in the map.
|
||||||
|
@interface GoSeqRef : NSObject {
|
||||||
|
}
|
||||||
|
@property(readonly) int32_t refnum;
|
||||||
|
@property(strong) id obj; // NULL when representing a Go object.
|
||||||
|
|
||||||
|
// new GoSeqRef object to proxy a Go object. The refnum must be
|
||||||
|
// provided from Go side.
|
||||||
|
- (instancetype)initWithRefnum:(int32_t)refnum obj:(id)obj;
|
||||||
|
|
||||||
|
- (int32_t)incNum;
|
||||||
|
|
||||||
|
@end
|
||||||
|
|
||||||
|
@protocol goSeqRefInterface
|
||||||
|
-(GoSeqRef*) _ref;
|
||||||
|
@end
|
||||||
|
|
||||||
|
#endif
|
||||||
8
go/dist/Gopenpgpwrapper.framework/Versions/Current/Modules/module.modulemap
vendored
Normal file
8
go/dist/Gopenpgpwrapper.framework/Versions/Current/Modules/module.modulemap
vendored
Normal file
|
|
@ -0,0 +1,8 @@
|
||||||
|
framework module "Gopenpgpwrapper" {
|
||||||
|
header "ref.h"
|
||||||
|
header "Gopenpgpwrapper.objc.h"
|
||||||
|
header "Universe.objc.h"
|
||||||
|
header "Gopenpgpwrapper.h"
|
||||||
|
|
||||||
|
export *
|
||||||
|
}
|
||||||
6
go/dist/Gopenpgpwrapper.framework/Versions/Current/Resources/Info.plist
vendored
Normal file
6
go/dist/Gopenpgpwrapper.framework/Versions/Current/Resources/Info.plist
vendored
Normal file
|
|
@ -0,0 +1,6 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
||||||
84
go/src/gopenpgpwrapper.go
Normal file
84
go/src/gopenpgpwrapper.go
Normal file
|
|
@ -0,0 +1,84 @@
|
||||||
|
package gopenpgpwrapper // import "gopenpgpwrapper"
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"io"
|
||||||
|
"io/ioutil"
|
||||||
|
|
||||||
|
"github.com/ProtonMail/gopenpgp/crypto"
|
||||||
|
"github.com/ProtonMail/gopenpgp/armor"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Key struct {
|
||||||
|
kr crypto.KeyRing
|
||||||
|
}
|
||||||
|
|
||||||
|
func (k *Key) GetKeyID() string {
|
||||||
|
return k.kr.FirstKeyID
|
||||||
|
}
|
||||||
|
|
||||||
|
func (k *Key) Encrypt(plaintext []byte, armor bool) []byte {
|
||||||
|
var b bytes.Buffer
|
||||||
|
var w io.WriteCloser
|
||||||
|
if armor {
|
||||||
|
wr, err := k.kr.EncryptArmored(&b, nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
w = wr
|
||||||
|
} else {
|
||||||
|
wr, err := k.kr.Encrypt(&b, nil, "", false)
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
w = wr
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := w.Write(plaintext); err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := w.Close(); err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return b.Bytes()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (k Key) Decrypt(ciphertext []byte, passphrase string) []byte {
|
||||||
|
unarmored, err := armor.Unarmor(string(ciphertext))
|
||||||
|
if err != nil {
|
||||||
|
// Assume ciphertext is already in binary format
|
||||||
|
unarmored = ciphertext
|
||||||
|
}
|
||||||
|
|
||||||
|
err = k.kr.Unlock([]byte(passphrase))
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
r, _, err := k.kr.Decrypt(bytes.NewReader(unarmored))
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if b, err := ioutil.ReadAll(r); err != nil {
|
||||||
|
return nil
|
||||||
|
} else {
|
||||||
|
return b
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func ReadKey(data []byte) *Key {
|
||||||
|
kr, err := crypto.ReadArmoredKeyRing(bytes.NewReader(data))
|
||||||
|
if err != nil {
|
||||||
|
// Assume keyring is in binary form
|
||||||
|
kr, err = crypto.ReadKeyRing(bytes.NewReader(data))
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return &Key{kr: *kr}
|
||||||
|
}
|
||||||
|
|
@ -7,6 +7,7 @@
|
||||||
objects = {
|
objects = {
|
||||||
|
|
||||||
/* Begin PBXBuildFile section */
|
/* Begin PBXBuildFile section */
|
||||||
|
1154BEDD229AC00F00454075 /* Gopenpgpwrapper.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 1154BEDB229AC00F00454075 /* Gopenpgpwrapper.framework */; };
|
||||||
18F19A67B0C07F13C17169E0 /* Pods_pass.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 3A5620D17DF5E86B61761D0E /* Pods_pass.framework */; };
|
18F19A67B0C07F13C17169E0 /* Pods_pass.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 3A5620D17DF5E86B61761D0E /* Pods_pass.framework */; };
|
||||||
23B82F0228254275DBA609E7 /* Pods_passExtension.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = B975797E0F0B7476CADD6A7D /* Pods_passExtension.framework */; };
|
23B82F0228254275DBA609E7 /* Pods_passExtension.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = B975797E0F0B7476CADD6A7D /* Pods_passExtension.framework */; };
|
||||||
300713C52219D54100F553AC /* AutoCellHeightUITableViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 300713C42219D54100F553AC /* AutoCellHeightUITableViewController.swift */; };
|
300713C52219D54100F553AC /* AutoCellHeightUITableViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 300713C42219D54100F553AC /* AutoCellHeightUITableViewController.swift */; };
|
||||||
|
|
@ -207,6 +208,7 @@
|
||||||
/* End PBXCopyFilesBuildPhase section */
|
/* End PBXCopyFilesBuildPhase section */
|
||||||
|
|
||||||
/* Begin PBXFileReference section */
|
/* Begin PBXFileReference section */
|
||||||
|
1154BEDB229AC00F00454075 /* Gopenpgpwrapper.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = Gopenpgpwrapper.framework; path = go/dist/Gopenpgpwrapper.framework; sourceTree = "<group>"; };
|
||||||
300713C42219D54100F553AC /* AutoCellHeightUITableViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AutoCellHeightUITableViewController.swift; sourceTree = "<group>"; };
|
300713C42219D54100F553AC /* AutoCellHeightUITableViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AutoCellHeightUITableViewController.swift; sourceTree = "<group>"; };
|
||||||
301F6462216162550071A4CE /* AdditionField.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AdditionField.swift; sourceTree = "<group>"; };
|
301F6462216162550071A4CE /* AdditionField.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AdditionField.swift; sourceTree = "<group>"; };
|
||||||
301F6467216165290071A4CE /* ConstantsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ConstantsTest.swift; sourceTree = "<group>"; };
|
301F6467216165290071A4CE /* ConstantsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ConstantsTest.swift; sourceTree = "<group>"; };
|
||||||
|
|
@ -373,6 +375,7 @@
|
||||||
isa = PBXFrameworksBuildPhase;
|
isa = PBXFrameworksBuildPhase;
|
||||||
buildActionMask = 2147483647;
|
buildActionMask = 2147483647;
|
||||||
files = (
|
files = (
|
||||||
|
1154BEDD229AC00F00454075 /* Gopenpgpwrapper.framework in Frameworks */,
|
||||||
61326CDA7A73757FB68DCB04 /* Pods_passKit.framework in Frameworks */,
|
61326CDA7A73757FB68DCB04 /* Pods_passKit.framework in Frameworks */,
|
||||||
);
|
);
|
||||||
runOnlyForDeploymentPostprocessing = 0;
|
runOnlyForDeploymentPostprocessing = 0;
|
||||||
|
|
@ -413,6 +416,7 @@
|
||||||
files = (
|
files = (
|
||||||
A239F5902158C07D00576CBF /* AuthenticationServices.framework in Frameworks */,
|
A239F5902158C07D00576CBF /* AuthenticationServices.framework in Frameworks */,
|
||||||
A260758D1EEC6F34005DB03E /* passKit.framework in Frameworks */,
|
A260758D1EEC6F34005DB03E /* passKit.framework in Frameworks */,
|
||||||
|
1154BEDC229AC00F00454075 /* Gopenpgpwrapper.framework in Frameworks */,
|
||||||
DCC408C71E307DBB00F29B0E /* SVProgressHUD.framework in Frameworks */,
|
DCC408C71E307DBB00F29B0E /* SVProgressHUD.framework in Frameworks */,
|
||||||
18F19A67B0C07F13C17169E0 /* Pods_pass.framework in Frameworks */,
|
18F19A67B0C07F13C17169E0 /* Pods_pass.framework in Frameworks */,
|
||||||
);
|
);
|
||||||
|
|
@ -768,6 +772,7 @@
|
||||||
B975797E0F0B7476CADD6A7D /* Pods_passExtension.framework */,
|
B975797E0F0B7476CADD6A7D /* Pods_passExtension.framework */,
|
||||||
DAB3F5541E51ADC8C6B56642 /* Pods_passKit.framework */,
|
DAB3F5541E51ADC8C6B56642 /* Pods_passKit.framework */,
|
||||||
CF843B3CF7D55A4070CBA1E4 /* Pods_passKitTests.framework */,
|
CF843B3CF7D55A4070CBA1E4 /* Pods_passKitTests.framework */,
|
||||||
|
1154BEDB229AC00F00454075 /* Gopenpgpwrapper.framework */,
|
||||||
);
|
);
|
||||||
name = Frameworks;
|
name = Frameworks;
|
||||||
sourceTree = "<group>";
|
sourceTree = "<group>";
|
||||||
|
|
@ -830,7 +835,6 @@
|
||||||
A260757C1EEC6F34005DB03E /* Sources */,
|
A260757C1EEC6F34005DB03E /* Sources */,
|
||||||
A260757D1EEC6F34005DB03E /* Frameworks */,
|
A260757D1EEC6F34005DB03E /* Frameworks */,
|
||||||
A260757E1EEC6F34005DB03E /* Resources */,
|
A260757E1EEC6F34005DB03E /* Resources */,
|
||||||
6AEAED0AF4328940B21EAC44 /* [CP] Embed Pods Frameworks */,
|
|
||||||
);
|
);
|
||||||
buildRules = (
|
buildRules = (
|
||||||
);
|
);
|
||||||
|
|
@ -890,7 +894,6 @@
|
||||||
DC917BEC1E2F3659000FDF54 /* Run Script */,
|
DC917BEC1E2F3659000FDF54 /* Run Script */,
|
||||||
A26700191EEC450100176B8A /* Embed App Extensions */,
|
A26700191EEC450100176B8A /* Embed App Extensions */,
|
||||||
A26075921EEC6F34005DB03E /* Embed Frameworks */,
|
A26075921EEC6F34005DB03E /* Embed Frameworks */,
|
||||||
7F5ED3FD24ED627DC957D425 /* [CP] Embed Pods Frameworks */,
|
|
||||||
);
|
);
|
||||||
buildRules = (
|
buildRules = (
|
||||||
);
|
);
|
||||||
|
|
@ -1093,26 +1096,6 @@
|
||||||
shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n # print error to STDERR\n echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n";
|
shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n # print error to STDERR\n echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n";
|
||||||
showEnvVarsInLog = 0;
|
showEnvVarsInLog = 0;
|
||||||
};
|
};
|
||||||
6AEAED0AF4328940B21EAC44 /* [CP] Embed Pods Frameworks */ = {
|
|
||||||
isa = PBXShellScriptBuildPhase;
|
|
||||||
buildActionMask = 2147483647;
|
|
||||||
files = (
|
|
||||||
);
|
|
||||||
inputPaths = (
|
|
||||||
"${SRCROOT}/Pods/Target Support Files/Pods-passKitTests/Pods-passKitTests-frameworks.sh",
|
|
||||||
"${PODS_ROOT}/ObjectivePGP/Frameworks/ios/ObjectivePGP.framework",
|
|
||||||
"${PODS_ROOT}/ObjectivePGP/Frameworks/ios/ObjectivePGP.framework.dSYM",
|
|
||||||
);
|
|
||||||
name = "[CP] Embed Pods Frameworks";
|
|
||||||
outputPaths = (
|
|
||||||
"${TARGET_BUILD_DIR}/${FRAMEWORKS_FOLDER_PATH}/ObjectivePGP.framework",
|
|
||||||
"${DWARF_DSYM_FOLDER_PATH}/ObjectivePGP.framework.dSYM",
|
|
||||||
);
|
|
||||||
runOnlyForDeploymentPostprocessing = 0;
|
|
||||||
shellPath = /bin/sh;
|
|
||||||
shellScript = "\"${SRCROOT}/Pods/Target Support Files/Pods-passKitTests/Pods-passKitTests-frameworks.sh\"\n";
|
|
||||||
showEnvVarsInLog = 0;
|
|
||||||
};
|
|
||||||
736C6F64F90A20CB9A00B420 /* [CP] Check Pods Manifest.lock */ = {
|
736C6F64F90A20CB9A00B420 /* [CP] Check Pods Manifest.lock */ = {
|
||||||
isa = PBXShellScriptBuildPhase;
|
isa = PBXShellScriptBuildPhase;
|
||||||
buildActionMask = 2147483647;
|
buildActionMask = 2147483647;
|
||||||
|
|
@ -1131,26 +1114,6 @@
|
||||||
shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n # print error to STDERR\n echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n";
|
shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n # print error to STDERR\n echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n";
|
||||||
showEnvVarsInLog = 0;
|
showEnvVarsInLog = 0;
|
||||||
};
|
};
|
||||||
7F5ED3FD24ED627DC957D425 /* [CP] Embed Pods Frameworks */ = {
|
|
||||||
isa = PBXShellScriptBuildPhase;
|
|
||||||
buildActionMask = 2147483647;
|
|
||||||
files = (
|
|
||||||
);
|
|
||||||
inputPaths = (
|
|
||||||
"${SRCROOT}/Pods/Target Support Files/Pods-pass/Pods-pass-frameworks.sh",
|
|
||||||
"${PODS_ROOT}/ObjectivePGP/Frameworks/ios/ObjectivePGP.framework",
|
|
||||||
"${PODS_ROOT}/ObjectivePGP/Frameworks/ios/ObjectivePGP.framework.dSYM",
|
|
||||||
);
|
|
||||||
name = "[CP] Embed Pods Frameworks";
|
|
||||||
outputPaths = (
|
|
||||||
"${TARGET_BUILD_DIR}/${FRAMEWORKS_FOLDER_PATH}/ObjectivePGP.framework",
|
|
||||||
"${DWARF_DSYM_FOLDER_PATH}/ObjectivePGP.framework.dSYM",
|
|
||||||
);
|
|
||||||
runOnlyForDeploymentPostprocessing = 0;
|
|
||||||
shellPath = /bin/sh;
|
|
||||||
shellScript = "\"${SRCROOT}/Pods/Target Support Files/Pods-pass/Pods-pass-frameworks.sh\"\n";
|
|
||||||
showEnvVarsInLog = 0;
|
|
||||||
};
|
|
||||||
94AA4FCF7FF3474A970BE194 /* [CP] Check Pods Manifest.lock */ = {
|
94AA4FCF7FF3474A970BE194 /* [CP] Check Pods Manifest.lock */ = {
|
||||||
isa = PBXShellScriptBuildPhase;
|
isa = PBXShellScriptBuildPhase;
|
||||||
buildActionMask = 2147483647;
|
buildActionMask = 2147483647;
|
||||||
|
|
@ -1502,7 +1465,10 @@
|
||||||
DYLIB_CURRENT_VERSION = 1;
|
DYLIB_CURRENT_VERSION = 1;
|
||||||
DYLIB_INSTALL_NAME_BASE = "@rpath";
|
DYLIB_INSTALL_NAME_BASE = "@rpath";
|
||||||
ENABLE_BITCODE = NO;
|
ENABLE_BITCODE = NO;
|
||||||
FRAMEWORK_SEARCH_PATHS = "$(inherited)";
|
FRAMEWORK_SEARCH_PATHS = (
|
||||||
|
"$(inherited)",
|
||||||
|
"$(PROJECT_DIR)/go/dist",
|
||||||
|
);
|
||||||
HEADER_SEARCH_PATHS = "$(inherited)";
|
HEADER_SEARCH_PATHS = "$(inherited)";
|
||||||
INFOPLIST_FILE = passKit/Info.plist;
|
INFOPLIST_FILE = passKit/Info.plist;
|
||||||
INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks";
|
INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks";
|
||||||
|
|
@ -1541,7 +1507,10 @@
|
||||||
DYLIB_CURRENT_VERSION = 1;
|
DYLIB_CURRENT_VERSION = 1;
|
||||||
DYLIB_INSTALL_NAME_BASE = "@rpath";
|
DYLIB_INSTALL_NAME_BASE = "@rpath";
|
||||||
ENABLE_BITCODE = NO;
|
ENABLE_BITCODE = NO;
|
||||||
FRAMEWORK_SEARCH_PATHS = "$(inherited)";
|
FRAMEWORK_SEARCH_PATHS = (
|
||||||
|
"$(inherited)",
|
||||||
|
"$(PROJECT_DIR)/go/dist",
|
||||||
|
);
|
||||||
HEADER_SEARCH_PATHS = "$(inherited)";
|
HEADER_SEARCH_PATHS = "$(inherited)";
|
||||||
INFOPLIST_FILE = passKit/Info.plist;
|
INFOPLIST_FILE = passKit/Info.plist;
|
||||||
INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks";
|
INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks";
|
||||||
|
|
@ -1830,7 +1799,10 @@
|
||||||
DEFINES_MODULE = NO;
|
DEFINES_MODULE = NO;
|
||||||
DEVELOPMENT_TEAM = 4WDM8E95VU;
|
DEVELOPMENT_TEAM = 4WDM8E95VU;
|
||||||
ENABLE_BITCODE = NO;
|
ENABLE_BITCODE = NO;
|
||||||
FRAMEWORK_SEARCH_PATHS = "$(inherited)";
|
FRAMEWORK_SEARCH_PATHS = (
|
||||||
|
"$(inherited)",
|
||||||
|
"$(PROJECT_DIR)/go/dist",
|
||||||
|
);
|
||||||
HEADER_SEARCH_PATHS = "$(inherited)";
|
HEADER_SEARCH_PATHS = "$(inherited)";
|
||||||
INFOPLIST_FILE = pass/Info.plist;
|
INFOPLIST_FILE = pass/Info.plist;
|
||||||
LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks";
|
LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks";
|
||||||
|
|
@ -1862,7 +1834,10 @@
|
||||||
DEFINES_MODULE = NO;
|
DEFINES_MODULE = NO;
|
||||||
DEVELOPMENT_TEAM = 4WDM8E95VU;
|
DEVELOPMENT_TEAM = 4WDM8E95VU;
|
||||||
ENABLE_BITCODE = NO;
|
ENABLE_BITCODE = NO;
|
||||||
FRAMEWORK_SEARCH_PATHS = "$(inherited)";
|
FRAMEWORK_SEARCH_PATHS = (
|
||||||
|
"$(inherited)",
|
||||||
|
"$(PROJECT_DIR)/go/dist",
|
||||||
|
);
|
||||||
HEADER_SEARCH_PATHS = "$(inherited)";
|
HEADER_SEARCH_PATHS = "$(inherited)";
|
||||||
INFOPLIST_FILE = pass/Info.plist;
|
INFOPLIST_FILE = pass/Info.plist;
|
||||||
LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks";
|
LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks";
|
||||||
|
|
|
||||||
|
|
@ -38,8 +38,8 @@ class SettingsTableViewController: UITableViewController, UITabBarControllerDele
|
||||||
SVProgressHUD.show(withStatus: "FetchingPgpKey".localize())
|
SVProgressHUD.show(withStatus: "FetchingPgpKey".localize())
|
||||||
DispatchQueue.global(qos: .userInitiated).async { [unowned self] in
|
DispatchQueue.global(qos: .userInitiated).async { [unowned self] in
|
||||||
do {
|
do {
|
||||||
try self.passwordStore.initPGPKey(from: SharedDefaults[.pgpPublicKeyURL]!, keyType: .public)
|
try self.passwordStore.initPGPKey(from: SharedDefaults[.pgpPublicKeyURL]!, keyType: .PUBLIC)
|
||||||
try self.passwordStore.initPGPKey(from: SharedDefaults[.pgpPrivateKeyURL]!, keyType: .secret)
|
try self.passwordStore.initPGPKey(from: SharedDefaults[.pgpPrivateKeyURL]!, keyType: .PRIVATE)
|
||||||
DispatchQueue.main.async {
|
DispatchQueue.main.async {
|
||||||
self.pgpKeyTableViewCell.detailTextLabel?.text = self.passwordStore.pgpKeyID
|
self.pgpKeyTableViewCell.detailTextLabel?.text = self.passwordStore.pgpKeyID
|
||||||
SVProgressHUD.showSuccess(withStatus: "Success".localize())
|
SVProgressHUD.showSuccess(withStatus: "Success".localize())
|
||||||
|
|
@ -68,8 +68,8 @@ class SettingsTableViewController: UITableViewController, UITabBarControllerDele
|
||||||
SVProgressHUD.show(withStatus: "FetchingPgpKey".localize())
|
SVProgressHUD.show(withStatus: "FetchingPgpKey".localize())
|
||||||
DispatchQueue.global(qos: .userInitiated).async { [unowned self] in
|
DispatchQueue.global(qos: .userInitiated).async { [unowned self] in
|
||||||
do {
|
do {
|
||||||
try self.passwordStore.initPGPKey(with: SharedDefaults[.pgpPublicKeyArmor] ?? "", keyType: .public)
|
try self.passwordStore.initPGPKey(with: SharedDefaults[.pgpPublicKeyArmor] ?? "", keyType: .PUBLIC)
|
||||||
try self.passwordStore.initPGPKey(with: SharedDefaults[.pgpPrivateKeyArmor] ?? "", keyType: .secret)
|
try self.passwordStore.initPGPKey(with: SharedDefaults[.pgpPrivateKeyArmor] ?? "", keyType: .PRIVATE)
|
||||||
DispatchQueue.main.async {
|
DispatchQueue.main.async {
|
||||||
self.pgpKeyTableViewCell.detailTextLabel?.text = self.passwordStore.pgpKeyID
|
self.pgpKeyTableViewCell.detailTextLabel?.text = self.passwordStore.pgpKeyID
|
||||||
SVProgressHUD.showSuccess(withStatus: "Success".localize())
|
SVProgressHUD.showSuccess(withStatus: "Success".localize())
|
||||||
|
|
|
||||||
|
|
@ -9,7 +9,6 @@
|
||||||
#ifndef Objective_CBridgingHeader_h
|
#ifndef Objective_CBridgingHeader_h
|
||||||
#define Objective_CBridgingHeader_h
|
#define Objective_CBridgingHeader_h
|
||||||
|
|
||||||
#import "ObjectivePGP/ObjectivePGP.h"
|
|
||||||
#import "ObjectiveGit/ObjectiveGit.h"
|
#import "ObjectiveGit/ObjectiveGit.h"
|
||||||
|
|
||||||
#endif /* Objective_CBridgingHeader_h */
|
#endif /* Objective_CBridgingHeader_h */
|
||||||
|
|
|
||||||
|
|
@ -11,8 +11,8 @@ import CoreData
|
||||||
import UIKit
|
import UIKit
|
||||||
import SwiftyUserDefaults
|
import SwiftyUserDefaults
|
||||||
import ObjectiveGit
|
import ObjectiveGit
|
||||||
import ObjectivePGP
|
|
||||||
import KeychainAccess
|
import KeychainAccess
|
||||||
|
import Gopenpgpwrapper
|
||||||
|
|
||||||
public class PasswordStore {
|
public class PasswordStore {
|
||||||
public static let shared = PasswordStore()
|
public static let shared = PasswordStore()
|
||||||
|
|
@ -22,19 +22,26 @@ public class PasswordStore {
|
||||||
dateFormatter.timeStyle = .short
|
dateFormatter.timeStyle = .short
|
||||||
return dateFormatter
|
return dateFormatter
|
||||||
}()
|
}()
|
||||||
|
|
||||||
public let storeURL = URL(fileURLWithPath: "\(Globals.repositoryPath)")
|
public let storeURL = URL(fileURLWithPath: "\(Globals.repositoryPath)")
|
||||||
public let tempStoreURL = URL(fileURLWithPath: "\(Globals.repositoryPath)-temp")
|
public let tempStoreURL = URL(fileURLWithPath: "\(Globals.repositoryPath)-temp")
|
||||||
|
|
||||||
public var storeRepository: GTRepository?
|
public var storeRepository: GTRepository?
|
||||||
public var pgpKeyID: String?
|
public var pgpKeyID: String?
|
||||||
public var publicKey: Key? {
|
public var publicKey: GopenpgpwrapperKey? {
|
||||||
didSet {
|
didSet {
|
||||||
pgpKeyID = publicKey?.keyID.shortIdentifier
|
if publicKey != nil {
|
||||||
|
pgpKeyID = publicKey!.getID()
|
||||||
|
} else {
|
||||||
|
pgpKeyID = nil
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
public var privateKey: Key?
|
public var privateKey: GopenpgpwrapperKey?
|
||||||
|
public enum PGPKeyType {
|
||||||
|
case PUBLIC, PRIVATE
|
||||||
|
}
|
||||||
|
|
||||||
public var gitSignatureForNow: GTSignature {
|
public var gitSignatureForNow: GTSignature {
|
||||||
get {
|
get {
|
||||||
let gitSignatureName = SharedDefaults[.gitSignatureName] ?? Globals.gitSignatureDefaultName
|
let gitSignatureName = SharedDefaults[.gitSignatureName] ?? Globals.gitSignatureDefaultName
|
||||||
|
|
@ -42,9 +49,7 @@ public class PasswordStore {
|
||||||
return GTSignature(name: gitSignatureName, email: gitSignatureEmail, time: Date())!
|
return GTSignature(name: gitSignatureName, email: gitSignatureEmail, time: Date())!
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public let keyring = ObjectivePGP.defaultKeyring
|
|
||||||
|
|
||||||
public var pgpKeyPassphrase: String? {
|
public var pgpKeyPassphrase: String? {
|
||||||
set {
|
set {
|
||||||
Utils.addPasswordToKeychain(name: "pgpKeyPassphrase", password: newValue)
|
Utils.addPasswordToKeychain(name: "pgpKeyPassphrase", password: newValue)
|
||||||
|
|
@ -53,7 +58,7 @@ public class PasswordStore {
|
||||||
return Utils.getPasswordFromKeychain(name: "pgpKeyPassphrase")
|
return Utils.getPasswordFromKeychain(name: "pgpKeyPassphrase")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public var gitPassword: String? {
|
public var gitPassword: String? {
|
||||||
set {
|
set {
|
||||||
Utils.addPasswordToKeychain(name: "gitPassword", password: newValue)
|
Utils.addPasswordToKeychain(name: "gitPassword", password: newValue)
|
||||||
|
|
@ -62,7 +67,7 @@ public class PasswordStore {
|
||||||
return Utils.getPasswordFromKeychain(name: "gitPassword")
|
return Utils.getPasswordFromKeychain(name: "gitPassword")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public var gitSSHPrivateKeyPassphrase: String? {
|
public var gitSSHPrivateKeyPassphrase: String? {
|
||||||
set {
|
set {
|
||||||
Utils.addPasswordToKeychain(name: "gitSSHPrivateKeyPassphrase", password: newValue)
|
Utils.addPasswordToKeychain(name: "gitSSHPrivateKeyPassphrase", password: newValue)
|
||||||
|
|
@ -71,7 +76,7 @@ public class PasswordStore {
|
||||||
return Utils.getPasswordFromKeychain(name: "gitSSHPrivateKeyPassphrase")
|
return Utils.getPasswordFromKeychain(name: "gitSSHPrivateKeyPassphrase")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private let fm = FileManager.default
|
private let fm = FileManager.default
|
||||||
lazy private var context: NSManagedObjectContext = {
|
lazy private var context: NSManagedObjectContext = {
|
||||||
let modelURL = Bundle(identifier: Globals.passKitBundleIdentifier)!.url(forResource: "pass", withExtension: "momd")!
|
let modelURL = Bundle(identifier: Globals.passKitBundleIdentifier)!.url(forResource: "pass", withExtension: "momd")!
|
||||||
|
|
@ -85,7 +90,7 @@ public class PasswordStore {
|
||||||
if let error = error as NSError? {
|
if let error = error as NSError? {
|
||||||
// Replace this implementation with code to handle the error appropriately.
|
// Replace this implementation with code to handle the error appropriately.
|
||||||
// fatalError() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
|
// fatalError() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Typical reasons for an error here include:
|
Typical reasons for an error here include:
|
||||||
* The parent directory does not exist, cannot be created, or disallows writing.
|
* The parent directory does not exist, cannot be created, or disallows writing.
|
||||||
|
|
@ -99,28 +104,32 @@ public class PasswordStore {
|
||||||
})
|
})
|
||||||
return container.viewContext
|
return container.viewContext
|
||||||
}()
|
}()
|
||||||
|
|
||||||
public var numberOfPasswords : Int {
|
public var numberOfPasswords : Int {
|
||||||
return self.fetchPasswordEntityCoreData(withDir: false).count
|
return self.fetchPasswordEntityCoreData(withDir: false).count
|
||||||
}
|
}
|
||||||
|
|
||||||
public var sizeOfRepositoryByteCount : UInt64 {
|
public var sizeOfRepositoryByteCount : UInt64 {
|
||||||
return (try? fm.allocatedSizeOfDirectoryAtURL(directoryURL: self.storeURL)) ?? 0
|
return (try? fm.allocatedSizeOfDirectoryAtURL(directoryURL: self.storeURL)) ?? 0
|
||||||
}
|
}
|
||||||
|
|
||||||
public var numberOfLocalCommits: Int? {
|
public var numberOfLocalCommits: Int? {
|
||||||
return (try? getLocalCommits())?.count
|
return (try? getLocalCommits())?.flatMap { $0.count }
|
||||||
}
|
}
|
||||||
|
|
||||||
public var lastSyncedTime: Date? {
|
public var lastSyncedTime: Date? {
|
||||||
return SharedDefaults[.lastSyncedTime]
|
return SharedDefaults[.lastSyncedTime]
|
||||||
}
|
}
|
||||||
|
|
||||||
public var numberOfCommits: UInt? {
|
public var numberOfCommits: UInt? {
|
||||||
return storeRepository?.numberOfCommits(inCurrentBranch: nil)
|
return storeRepository?.numberOfCommits(inCurrentBranch: nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
private init() {
|
private init() {
|
||||||
|
// File migration to group
|
||||||
|
migrateIfNeeded()
|
||||||
|
backwardCompatibility()
|
||||||
|
|
||||||
do {
|
do {
|
||||||
if fm.fileExists(atPath: storeURL.path) {
|
if fm.fileExists(atPath: storeURL.path) {
|
||||||
try storeRepository = GTRepository.init(url: storeURL)
|
try storeRepository = GTRepository.init(url: storeURL)
|
||||||
|
|
@ -130,30 +139,76 @@ public class PasswordStore {
|
||||||
print(error)
|
print(error)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private func migrateIfNeeded() {
|
||||||
|
// migrate happens only if the repository was cloned and pgp keys were set up using earlier versions
|
||||||
|
let needMigration = !pgpKeyExists() && !gitSSHKeyExists() && !fm.fileExists(atPath: Globals.repositoryPath) && fm.fileExists(atPath: Globals.repositoryPathLegacy)
|
||||||
|
guard needMigration == true else {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
do {
|
||||||
|
// migrate Defaults
|
||||||
|
let userDefaults = UserDefaults()
|
||||||
|
for key in Defaults.dictionaryRepresentation().keys {
|
||||||
|
if SharedDefaults.value(forKey: key) == nil {
|
||||||
|
SharedDefaults.setValue(userDefaults.value(forKey: key), forKey: key)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// migrate files
|
||||||
|
try fm.createDirectory(atPath: Globals.documentPath, withIntermediateDirectories: true, attributes: nil)
|
||||||
|
try fm.createDirectory(atPath: Globals.libraryPath, withIntermediateDirectories: true, attributes: nil)
|
||||||
|
if fm.fileExists(atPath: Globals.pgpPublicKeyPathLegacy) {
|
||||||
|
try fm.moveItem(atPath: Globals.pgpPublicKeyPathLegacy, toPath: Globals.pgpPublicKeyPath)
|
||||||
|
}
|
||||||
|
if fm.fileExists(atPath: Globals.pgpPrivateKeyPathLegacy) {
|
||||||
|
try fm.moveItem(atPath: Globals.pgpPrivateKeyPathLegacy, toPath: Globals.pgpPrivateKeyPath)
|
||||||
|
}
|
||||||
|
if fm.fileExists(atPath: Globals.gitSSHPrivateKeyPathLegacy) {
|
||||||
|
try fm.moveItem(atPath: Globals.gitSSHPrivateKeyPathLegacy, toPath: Globals.gitSSHPrivateKeyPath)
|
||||||
|
}
|
||||||
|
try fm.moveItem(atPath: Globals.repositoryPathLegacy, toPath: Globals.repositoryPath)
|
||||||
|
} catch {
|
||||||
|
print("MigrationError".localize(error))
|
||||||
|
}
|
||||||
|
updatePasswordEntityCoreData()
|
||||||
|
}
|
||||||
|
|
||||||
|
private func backwardCompatibility() {
|
||||||
|
// For the newly-introduced isRememberGitCredentialPassphraseOn (20171008)
|
||||||
|
if (self.gitPassword != nil || self.gitSSHPrivateKeyPassphrase != nil) && SharedDefaults[.isRememberGitCredentialPassphraseOn] == false {
|
||||||
|
SharedDefaults[.isRememberGitCredentialPassphraseOn] = true
|
||||||
|
}
|
||||||
|
// For the renamed isRememberPGPPassphraseOn (20171008)
|
||||||
|
if self.pgpKeyPassphrase != nil && SharedDefaults[.isRememberPGPPassphraseOn] == false {
|
||||||
|
SharedDefaults[.isRememberPGPPassphraseOn] = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
enum SSHKeyType {
|
enum SSHKeyType {
|
||||||
case `public`, secret
|
case `public`, secret
|
||||||
}
|
}
|
||||||
|
|
||||||
public func initGitSSHKey(with armorKey: String) throws {
|
public func initGitSSHKey(with armorKey: String) throws {
|
||||||
let keyPath = Globals.gitSSHPrivateKeyPath
|
let keyPath = Globals.gitSSHPrivateKeyPath
|
||||||
try armorKey.write(toFile: keyPath, atomically: true, encoding: .ascii)
|
try armorKey.write(toFile: keyPath, atomically: true, encoding: .ascii)
|
||||||
}
|
}
|
||||||
|
|
||||||
public func initPGPKeys() throws {
|
public func initPGPKeys() throws {
|
||||||
try initPGPKey(.public)
|
try initPGPKey(.PUBLIC)
|
||||||
try initPGPKey(.secret)
|
try initPGPKey(.PRIVATE)
|
||||||
}
|
}
|
||||||
|
|
||||||
public func initPGPKey(_ keyType: PGPKeyType) throws {
|
public func initPGPKey(_ keyType: PGPKeyType) throws {
|
||||||
switch keyType {
|
switch keyType {
|
||||||
case .public:
|
case .PUBLIC:
|
||||||
let keyPath = Globals.pgpPublicKeyPath
|
let keyPath = Globals.pgpPublicKeyPath
|
||||||
self.publicKey = importKey(from: keyPath)
|
self.publicKey = importKey(from: keyPath)
|
||||||
if self.publicKey == nil {
|
if self.publicKey == nil {
|
||||||
throw AppError.KeyImport
|
throw AppError.KeyImport
|
||||||
}
|
}
|
||||||
case .secret:
|
case .PRIVATE:
|
||||||
let keyPath = Globals.pgpPrivateKeyPath
|
let keyPath = Globals.pgpPrivateKeyPath
|
||||||
self.privateKey = importKey(from: keyPath)
|
self.privateKey = importKey(from: keyPath)
|
||||||
if self.privateKey == nil {
|
if self.privateKey == nil {
|
||||||
|
|
@ -163,10 +218,10 @@ public class PasswordStore {
|
||||||
throw AppError.Unknown
|
throw AppError.Unknown
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public func initPGPKey(from url: URL, keyType: PGPKeyType) throws {
|
public func initPGPKey(from url: URL, keyType: PGPKeyType) throws {
|
||||||
var pgpKeyLocalPath = ""
|
var pgpKeyLocalPath = ""
|
||||||
if keyType == .public {
|
if keyType == .PUBLIC {
|
||||||
pgpKeyLocalPath = Globals.pgpPublicKeyPath
|
pgpKeyLocalPath = Globals.pgpPublicKeyPath
|
||||||
} else {
|
} else {
|
||||||
pgpKeyLocalPath = Globals.pgpPrivateKeyPath
|
pgpKeyLocalPath = Globals.pgpPrivateKeyPath
|
||||||
|
|
@ -175,10 +230,10 @@ public class PasswordStore {
|
||||||
try pgpKeyData.write(to: URL(fileURLWithPath: pgpKeyLocalPath), options: .atomic)
|
try pgpKeyData.write(to: URL(fileURLWithPath: pgpKeyLocalPath), options: .atomic)
|
||||||
try initPGPKey(keyType)
|
try initPGPKey(keyType)
|
||||||
}
|
}
|
||||||
|
|
||||||
public func initPGPKey(with armorKey: String, keyType: PGPKeyType) throws {
|
public func initPGPKey(with armorKey: String, keyType: PGPKeyType) throws {
|
||||||
var pgpKeyLocalPath = ""
|
var pgpKeyLocalPath = ""
|
||||||
if keyType == .public {
|
if keyType == .PUBLIC {
|
||||||
pgpKeyLocalPath = Globals.pgpPublicKeyPath
|
pgpKeyLocalPath = Globals.pgpPublicKeyPath
|
||||||
} else {
|
} else {
|
||||||
pgpKeyLocalPath = Globals.pgpPrivateKeyPath
|
pgpKeyLocalPath = Globals.pgpPrivateKeyPath
|
||||||
|
|
@ -186,28 +241,21 @@ public class PasswordStore {
|
||||||
try armorKey.write(toFile: pgpKeyLocalPath, atomically: true, encoding: .ascii)
|
try armorKey.write(toFile: pgpKeyLocalPath, atomically: true, encoding: .ascii)
|
||||||
try initPGPKey(keyType)
|
try initPGPKey(keyType)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private func importKey(from keyPath: String) -> Key? {
|
private func importKey(from keyPath: String) -> GopenpgpwrapperKey? {
|
||||||
if fm.fileExists(atPath: keyPath) {
|
if fm.fileExists(atPath: keyPath) {
|
||||||
let keys = try! ObjectivePGP.readKeys(fromPath: keyPath)
|
return GopenpgpwrapperReadKey(fm.contents(atPath: keyPath))
|
||||||
keyring.import(keys: keys)
|
|
||||||
if !keys.isEmpty {
|
|
||||||
return keys.first
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
public func getPgpPrivateKey() -> Key {
|
|
||||||
return keyring.keys.filter({$0.secretKey != nil})[0]
|
|
||||||
}
|
|
||||||
|
|
||||||
public func repositoryExisted() -> Bool {
|
public func repositoryExisted() -> Bool {
|
||||||
let fm = FileManager()
|
let fm = FileManager()
|
||||||
return fm.fileExists(atPath: Globals.repositoryPath)
|
return fm.fileExists(atPath: Globals.repositoryPath)
|
||||||
}
|
}
|
||||||
|
|
||||||
public func passwordExisted(password: Password) -> Bool {
|
public func passwordExisted(password: Password) -> Bool {
|
||||||
let passwordEntityFetchRequest = NSFetchRequest<NSFetchRequestResult>(entityName: "PasswordEntity")
|
let passwordEntityFetchRequest = NSFetchRequest<NSFetchRequestResult>(entityName: "PasswordEntity")
|
||||||
do {
|
do {
|
||||||
|
|
@ -221,8 +269,9 @@ public class PasswordStore {
|
||||||
} catch {
|
} catch {
|
||||||
fatalError("FailedToFetchPasswordEntities".localize(error))
|
fatalError("FailedToFetchPasswordEntities".localize(error))
|
||||||
}
|
}
|
||||||
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
public func passwordEntityExisted(path: String) -> Bool {
|
public func passwordEntityExisted(path: String) -> Bool {
|
||||||
let passwordEntityFetchRequest = NSFetchRequest<NSFetchRequestResult>(entityName: "PasswordEntity")
|
let passwordEntityFetchRequest = NSFetchRequest<NSFetchRequestResult>(entityName: "PasswordEntity")
|
||||||
do {
|
do {
|
||||||
|
|
@ -236,8 +285,9 @@ public class PasswordStore {
|
||||||
} catch {
|
} catch {
|
||||||
fatalError("FailedToFetchPasswordEntities".localize(error))
|
fatalError("FailedToFetchPasswordEntities".localize(error))
|
||||||
}
|
}
|
||||||
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
public func getPasswordEntity(by path: String, isDir: Bool) -> PasswordEntity? {
|
public func getPasswordEntity(by path: String, isDir: Bool) -> PasswordEntity? {
|
||||||
let passwordEntityFetchRequest = NSFetchRequest<NSFetchRequestResult>(entityName: "PasswordEntity")
|
let passwordEntityFetchRequest = NSFetchRequest<NSFetchRequestResult>(entityName: "PasswordEntity")
|
||||||
do {
|
do {
|
||||||
|
|
@ -247,13 +297,13 @@ public class PasswordStore {
|
||||||
fatalError("FailedToFetchPasswordEntities".localize(error))
|
fatalError("FailedToFetchPasswordEntities".localize(error))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public func cloneRepository(remoteRepoURL: URL,
|
public func cloneRepository(remoteRepoURL: URL,
|
||||||
credential: GitCredential,
|
credential: GitCredential,
|
||||||
branchName: String,
|
branchName: String,
|
||||||
requestGitPassword: @escaping (GitCredential.Credential, String?) -> String?,
|
requestGitPassword: @escaping (GitCredential.Credential, String?) -> String?,
|
||||||
transferProgressBlock: @escaping (UnsafePointer<git_transfer_progress>, UnsafeMutablePointer<ObjCBool>) -> Void,
|
transferProgressBlock: @escaping (UnsafePointer<git_transfer_progress>, UnsafeMutablePointer<ObjCBool>) -> Void,
|
||||||
checkoutProgressBlock: @escaping (String?, UInt, UInt) -> Void) throws {
|
checkoutProgressBlock: @escaping (String?, UInt, UInt) -> Void) throws {
|
||||||
try? fm.removeItem(at: storeURL)
|
try? fm.removeItem(at: storeURL)
|
||||||
try? fm.removeItem(at: tempStoreURL)
|
try? fm.removeItem(at: tempStoreURL)
|
||||||
self.gitPassword = nil
|
self.gitPassword = nil
|
||||||
|
|
@ -280,7 +330,7 @@ public class PasswordStore {
|
||||||
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
|
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private func checkoutAndChangeBranch(withName localBranchName: String) throws {
|
private func checkoutAndChangeBranch(withName localBranchName: String) throws {
|
||||||
if (localBranchName == "master") {
|
if (localBranchName == "master") {
|
||||||
return
|
return
|
||||||
|
|
@ -301,7 +351,7 @@ public class PasswordStore {
|
||||||
try storeRepository.checkoutReference(localBranch.reference, options: checkoutOptions)
|
try storeRepository.checkoutReference(localBranch.reference, options: checkoutOptions)
|
||||||
try storeRepository.moveHEAD(to: localBranch.reference)
|
try storeRepository.moveHEAD(to: localBranch.reference)
|
||||||
}
|
}
|
||||||
|
|
||||||
public func pullRepository(credential: GitCredential, requestGitPassword: @escaping (GitCredential.Credential, String?) -> String?, transferProgressBlock: @escaping (UnsafePointer<git_transfer_progress>, UnsafeMutablePointer<ObjCBool>) -> Void) throws {
|
public func pullRepository(credential: GitCredential, requestGitPassword: @escaping (GitCredential.Credential, String?) -> String?, transferProgressBlock: @escaping (UnsafePointer<git_transfer_progress>, UnsafeMutablePointer<ObjCBool>) -> Void) throws {
|
||||||
guard let storeRepository = storeRepository else {
|
guard let storeRepository = storeRepository else {
|
||||||
throw AppError.RepositoryNotSet
|
throw AppError.RepositoryNotSet
|
||||||
|
|
@ -317,22 +367,22 @@ public class PasswordStore {
|
||||||
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
|
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private func updatePasswordEntityCoreData() {
|
private func updatePasswordEntityCoreData() {
|
||||||
deleteCoreData(entityName: "PasswordEntity")
|
deleteCoreData(entityName: "PasswordEntity")
|
||||||
do {
|
do {
|
||||||
var q = try fm.contentsOfDirectory(atPath: self.storeURL.path).filter{
|
var q = try fm.contentsOfDirectory(atPath: self.storeURL.path).filter{
|
||||||
!$0.hasPrefix(".")
|
!$0.hasPrefix(".")
|
||||||
}.map { (filename) -> PasswordEntity in
|
}.map { (filename) -> PasswordEntity in
|
||||||
let passwordEntity = NSEntityDescription.insertNewObject(forEntityName: "PasswordEntity", into: context) as! PasswordEntity
|
let passwordEntity = NSEntityDescription.insertNewObject(forEntityName: "PasswordEntity", into: context) as! PasswordEntity
|
||||||
if filename.hasSuffix(".gpg") {
|
if filename.hasSuffix(".gpg") {
|
||||||
passwordEntity.name = String(filename.prefix(upTo: filename.index(filename.endIndex, offsetBy: -4)))
|
passwordEntity.name = String(filename.prefix(upTo: filename.index(filename.endIndex, offsetBy: -4)))
|
||||||
} else {
|
} else {
|
||||||
passwordEntity.name = filename
|
passwordEntity.name = filename
|
||||||
}
|
}
|
||||||
passwordEntity.path = filename
|
passwordEntity.path = filename
|
||||||
passwordEntity.parent = nil
|
passwordEntity.parent = nil
|
||||||
return passwordEntity
|
return passwordEntity
|
||||||
}
|
}
|
||||||
while q.count > 0 {
|
while q.count > 0 {
|
||||||
let e = q.first!
|
let e = q.first!
|
||||||
|
|
@ -371,7 +421,7 @@ public class PasswordStore {
|
||||||
print("ErrorSaving".localize(error))
|
print("ErrorSaving".localize(error))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public func getRecentCommits(count: Int) throws -> [GTCommit] {
|
public func getRecentCommits(count: Int) throws -> [GTCommit] {
|
||||||
guard let storeRepository = storeRepository else {
|
guard let storeRepository = storeRepository else {
|
||||||
return []
|
return []
|
||||||
|
|
@ -388,7 +438,7 @@ public class PasswordStore {
|
||||||
}
|
}
|
||||||
return commits
|
return commits
|
||||||
}
|
}
|
||||||
|
|
||||||
public func fetchPasswordEntityCoreData(parent: PasswordEntity?) -> [PasswordEntity] {
|
public func fetchPasswordEntityCoreData(parent: PasswordEntity?) -> [PasswordEntity] {
|
||||||
let passwordEntityFetch = NSFetchRequest<NSFetchRequestResult>(entityName: "PasswordEntity")
|
let passwordEntityFetch = NSFetchRequest<NSFetchRequestResult>(entityName: "PasswordEntity")
|
||||||
do {
|
do {
|
||||||
|
|
@ -399,7 +449,7 @@ public class PasswordStore {
|
||||||
fatalError("FailedToFetchPasswords".localize(error))
|
fatalError("FailedToFetchPasswords".localize(error))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public func fetchPasswordEntityCoreData(withDir: Bool) -> [PasswordEntity] {
|
public func fetchPasswordEntityCoreData(withDir: Bool) -> [PasswordEntity] {
|
||||||
let passwordEntityFetch = NSFetchRequest<NSFetchRequestResult>(entityName: "PasswordEntity")
|
let passwordEntityFetch = NSFetchRequest<NSFetchRequestResult>(entityName: "PasswordEntity")
|
||||||
do {
|
do {
|
||||||
|
|
@ -412,8 +462,8 @@ public class PasswordStore {
|
||||||
fatalError("FailedToFetchPasswords".localize(error))
|
fatalError("FailedToFetchPasswords".localize(error))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public func fetchUnsyncedPasswords() -> [PasswordEntity] {
|
public func fetchUnsyncedPasswords() -> [PasswordEntity] {
|
||||||
let passwordEntityFetchRequest = NSFetchRequest<NSFetchRequestResult>(entityName: "PasswordEntity")
|
let passwordEntityFetchRequest = NSFetchRequest<NSFetchRequestResult>(entityName: "PasswordEntity")
|
||||||
passwordEntityFetchRequest.predicate = NSPredicate(format: "synced = %i", 0)
|
passwordEntityFetchRequest.predicate = NSPredicate(format: "synced = %i", 0)
|
||||||
|
|
@ -424,7 +474,7 @@ public class PasswordStore {
|
||||||
fatalError("FailedToFetchPasswords".localize(error))
|
fatalError("FailedToFetchPasswords".localize(error))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public func setAllSynced() {
|
public func setAllSynced() {
|
||||||
let passwordEntities = fetchUnsyncedPasswords()
|
let passwordEntities = fetchUnsyncedPasswords()
|
||||||
for passwordEntity in passwordEntities {
|
for passwordEntity in passwordEntities {
|
||||||
|
|
@ -438,16 +488,16 @@ public class PasswordStore {
|
||||||
fatalError("ErrorSaving".localize(error))
|
fatalError("ErrorSaving".localize(error))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public func getLatestUpdateInfo(filename: String) -> String {
|
public func getLatestUpdateInfo(filename: String) -> String {
|
||||||
guard let storeRepository = storeRepository else {
|
guard let storeRepository = storeRepository else {
|
||||||
return "Unknown".localize()
|
return "Unknown".localize()
|
||||||
}
|
}
|
||||||
guard let blameHunks = try? storeRepository.blame(withFile: filename, options: nil).hunks,
|
guard let blameHunks = try? storeRepository.blame(withFile: filename, options: nil).hunks,
|
||||||
let latestCommitTime = blameHunks.map({
|
let latestCommitTime = blameHunks.map({
|
||||||
$0.finalSignature?.time?.timeIntervalSince1970 ?? 0
|
$0.finalSignature?.time?.timeIntervalSince1970 ?? 0
|
||||||
}).max() else {
|
}).max() else {
|
||||||
return "Unknown".localize()
|
return "Unknown".localize()
|
||||||
}
|
}
|
||||||
let lastCommitDate = Date(timeIntervalSince1970: latestCommitTime)
|
let lastCommitDate = Date(timeIntervalSince1970: latestCommitTime)
|
||||||
if Date().timeIntervalSince(lastCommitDate) <= 60 {
|
if Date().timeIntervalSince(lastCommitDate) <= 60 {
|
||||||
|
|
@ -455,10 +505,10 @@ public class PasswordStore {
|
||||||
}
|
}
|
||||||
return PasswordStore.dateFormatter.string(from: lastCommitDate)
|
return PasswordStore.dateFormatter.string(from: lastCommitDate)
|
||||||
}
|
}
|
||||||
|
|
||||||
public func updateRemoteRepo() {
|
public func updateRemoteRepo() {
|
||||||
}
|
}
|
||||||
|
|
||||||
private func gitAdd(path: String) throws {
|
private func gitAdd(path: String) throws {
|
||||||
guard let storeRepository = storeRepository else {
|
guard let storeRepository = storeRepository else {
|
||||||
throw AppError.RepositoryNotSet
|
throw AppError.RepositoryNotSet
|
||||||
|
|
@ -466,7 +516,7 @@ public class PasswordStore {
|
||||||
try storeRepository.index().addFile(path)
|
try storeRepository.index().addFile(path)
|
||||||
try storeRepository.index().write()
|
try storeRepository.index().write()
|
||||||
}
|
}
|
||||||
|
|
||||||
private func gitRm(path: String) throws {
|
private func gitRm(path: String) throws {
|
||||||
guard let storeRepository = storeRepository else {
|
guard let storeRepository = storeRepository else {
|
||||||
throw AppError.RepositoryNotSet
|
throw AppError.RepositoryNotSet
|
||||||
|
|
@ -478,7 +528,7 @@ public class PasswordStore {
|
||||||
try storeRepository.index().removeFile(path)
|
try storeRepository.index().removeFile(path)
|
||||||
try storeRepository.index().write()
|
try storeRepository.index().write()
|
||||||
}
|
}
|
||||||
|
|
||||||
private func deleteDirectoryTree(at url: URL) throws {
|
private func deleteDirectoryTree(at url: URL) throws {
|
||||||
var tempURL = storeURL.appendingPathComponent(url.deletingLastPathComponent().path)
|
var tempURL = storeURL.appendingPathComponent(url.deletingLastPathComponent().path)
|
||||||
var count = try fm.contentsOfDirectory(atPath: tempURL.path).count
|
var count = try fm.contentsOfDirectory(atPath: tempURL.path).count
|
||||||
|
|
@ -488,12 +538,12 @@ public class PasswordStore {
|
||||||
count = try fm.contentsOfDirectory(atPath: tempURL.path).count
|
count = try fm.contentsOfDirectory(atPath: tempURL.path).count
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private func createDirectoryTree(at url: URL) throws {
|
private func createDirectoryTree(at url: URL) throws {
|
||||||
let tempURL = storeURL.appendingPathComponent(url.deletingLastPathComponent().path)
|
let tempURL = storeURL.appendingPathComponent(url.deletingLastPathComponent().path)
|
||||||
try fm.createDirectory(at: tempURL, withIntermediateDirectories: true, attributes: nil)
|
try fm.createDirectory(at: tempURL, withIntermediateDirectories: true, attributes: nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
private func gitMv(from: String, to: String) throws {
|
private func gitMv(from: String, to: String) throws {
|
||||||
let fromURL = storeURL.appendingPathComponent(from)
|
let fromURL = storeURL.appendingPathComponent(from)
|
||||||
let toURL = storeURL.appendingPathComponent(to)
|
let toURL = storeURL.appendingPathComponent(to)
|
||||||
|
|
@ -501,7 +551,7 @@ public class PasswordStore {
|
||||||
try gitAdd(path: to)
|
try gitAdd(path: to)
|
||||||
try gitRm(path: from)
|
try gitRm(path: from)
|
||||||
}
|
}
|
||||||
|
|
||||||
private func gitCommit(message: String) throws -> GTCommit? {
|
private func gitCommit(message: String) throws -> GTCommit? {
|
||||||
guard let storeRepository = storeRepository else {
|
guard let storeRepository = storeRepository else {
|
||||||
throw AppError.RepositoryNotSet
|
throw AppError.RepositoryNotSet
|
||||||
|
|
@ -515,7 +565,7 @@ public class PasswordStore {
|
||||||
let commit = try storeRepository.createCommit(with: newTree, message: message, author: signature, committer: signature, parents: [parent], updatingReferenceNamed: headReference.name)
|
let commit = try storeRepository.createCommit(with: newTree, message: message, author: signature, committer: signature, parents: [parent], updatingReferenceNamed: headReference.name)
|
||||||
return commit
|
return commit
|
||||||
}
|
}
|
||||||
|
|
||||||
private func getLocalBranch(withName branchName: String) throws -> GTBranch? {
|
private func getLocalBranch(withName branchName: String) throws -> GTBranch? {
|
||||||
guard let storeRepository = storeRepository else {
|
guard let storeRepository = storeRepository else {
|
||||||
throw AppError.RepositoryNotSet
|
throw AppError.RepositoryNotSet
|
||||||
|
|
@ -524,7 +574,7 @@ public class PasswordStore {
|
||||||
let branches = try storeRepository.branches(withPrefix: reference)
|
let branches = try storeRepository.branches(withPrefix: reference)
|
||||||
return branches.first
|
return branches.first
|
||||||
}
|
}
|
||||||
|
|
||||||
public func pushRepository(credential: GitCredential, requestGitPassword: @escaping (GitCredential.Credential, String?) -> String?, transferProgressBlock: @escaping (UInt32, UInt32, Int, UnsafeMutablePointer<ObjCBool>) -> Void) throws {
|
public func pushRepository(credential: GitCredential, requestGitPassword: @escaping (GitCredential.Credential, String?) -> String?, transferProgressBlock: @escaping (UInt32, UInt32, Int, UnsafeMutablePointer<ObjCBool>) -> Void) throws {
|
||||||
guard let storeRepository = storeRepository else {
|
guard let storeRepository = storeRepository else {
|
||||||
throw AppError.RepositoryNotSet
|
throw AppError.RepositoryNotSet
|
||||||
|
|
@ -540,12 +590,12 @@ public class PasswordStore {
|
||||||
throw(error)
|
throw(error)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private func addPasswordEntities(password: Password) throws -> PasswordEntity? {
|
private func addPasswordEntities(password: Password) throws -> PasswordEntity? {
|
||||||
guard !passwordExisted(password: password) else {
|
guard !passwordExisted(password: password) else {
|
||||||
throw AppError.PasswordDuplicated
|
throw AppError.PasswordDuplicated
|
||||||
}
|
}
|
||||||
|
|
||||||
var passwordURL = password.url
|
var passwordURL = password.url
|
||||||
var previousPathLength = Int.max
|
var previousPathLength = Int.max
|
||||||
var paths: [String] = []
|
var paths: [String] = []
|
||||||
|
|
@ -575,7 +625,7 @@ public class PasswordStore {
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
private func insertPasswordEntity(name: String, path: String, parent: PasswordEntity?, synced: Bool = false, isDir: Bool = false) -> PasswordEntity? {
|
private func insertPasswordEntity(name: String, path: String, parent: PasswordEntity?, synced: Bool = false, isDir: Bool = false) -> PasswordEntity? {
|
||||||
var ret: PasswordEntity? = nil
|
var ret: PasswordEntity? = nil
|
||||||
if let passwordEntity = NSEntityDescription.insertNewObject(forEntityName: "PasswordEntity", into: self.context) as? PasswordEntity {
|
if let passwordEntity = NSEntityDescription.insertNewObject(forEntityName: "PasswordEntity", into: self.context) as? PasswordEntity {
|
||||||
|
|
@ -593,7 +643,7 @@ public class PasswordStore {
|
||||||
}
|
}
|
||||||
return ret
|
return ret
|
||||||
}
|
}
|
||||||
|
|
||||||
public func add(password: Password) throws -> PasswordEntity? {
|
public func add(password: Password) throws -> PasswordEntity? {
|
||||||
try createDirectoryTree(at: password.url)
|
try createDirectoryTree(at: password.url)
|
||||||
let newPasswordEntity = try addPasswordEntities(password: password)
|
let newPasswordEntity = try addPasswordEntities(password: password)
|
||||||
|
|
@ -604,7 +654,7 @@ public class PasswordStore {
|
||||||
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
|
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
|
||||||
return newPasswordEntity
|
return newPasswordEntity
|
||||||
}
|
}
|
||||||
|
|
||||||
public func delete(passwordEntity: PasswordEntity) throws {
|
public func delete(passwordEntity: PasswordEntity) throws {
|
||||||
let deletedFileURL = passwordEntity.getURL()!
|
let deletedFileURL = passwordEntity.getURL()!
|
||||||
try gitRm(path: deletedFileURL.path)
|
try gitRm(path: deletedFileURL.path)
|
||||||
|
|
@ -613,10 +663,10 @@ public class PasswordStore {
|
||||||
let _ = try gitCommit(message: "RemovePassword.".localize(deletedFileURL.deletingPathExtension().path.removingPercentEncoding!))
|
let _ = try gitCommit(message: "RemovePassword.".localize(deletedFileURL.deletingPathExtension().path.removingPercentEncoding!))
|
||||||
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
|
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
public func edit(passwordEntity: PasswordEntity, password: Password) throws -> PasswordEntity? {
|
public func edit(passwordEntity: PasswordEntity, password: Password) throws -> PasswordEntity? {
|
||||||
var newPasswordEntity: PasswordEntity? = passwordEntity
|
var newPasswordEntity: PasswordEntity? = passwordEntity
|
||||||
|
|
||||||
if password.changed&PasswordChange.content.rawValue != 0 {
|
if password.changed&PasswordChange.content.rawValue != 0 {
|
||||||
let saveURL = storeURL.appendingPathComponent(passwordEntity.getURL()!.path)
|
let saveURL = storeURL.appendingPathComponent(passwordEntity.getURL()!.path)
|
||||||
try self.encrypt(password: password).write(to: saveURL)
|
try self.encrypt(password: password).write(to: saveURL)
|
||||||
|
|
@ -625,27 +675,25 @@ public class PasswordStore {
|
||||||
newPasswordEntity = passwordEntity
|
newPasswordEntity = passwordEntity
|
||||||
newPasswordEntity?.synced = false
|
newPasswordEntity?.synced = false
|
||||||
}
|
}
|
||||||
|
|
||||||
if password.changed&PasswordChange.path.rawValue != 0 {
|
if password.changed&PasswordChange.path.rawValue != 0 {
|
||||||
let deletedFileURL = passwordEntity.getURL()!
|
let deletedFileURL = passwordEntity.getURL()!
|
||||||
// add
|
// add
|
||||||
try createDirectoryTree(at: password.url)
|
try createDirectoryTree(at: password.url)
|
||||||
newPasswordEntity = try addPasswordEntities(password: password)
|
newPasswordEntity = try addPasswordEntities(password: password)
|
||||||
|
|
||||||
// mv
|
// mv
|
||||||
try gitMv(from: deletedFileURL.path, to: password.url.path)
|
try gitMv(from: deletedFileURL.path, to: password.url.path)
|
||||||
|
|
||||||
// delete
|
// delete
|
||||||
try deleteDirectoryTree(at: deletedFileURL)
|
try deleteDirectoryTree(at: deletedFileURL)
|
||||||
try deletePasswordEntities(passwordEntity: passwordEntity)
|
try deletePasswordEntities(passwordEntity: passwordEntity)
|
||||||
let _ = try gitCommit(message: "RenamePassword.".localize(deletedFileURL.deletingPathExtension().path.removingPercentEncoding!, password.url.deletingPathExtension().path.removingPercentEncoding!))
|
let _ = try gitCommit(message: "RenamePassword.".localize(deletedFileURL.deletingPathExtension().path.removingPercentEncoding!, password.url.deletingPathExtension().path.removingPercentEncoding!))
|
||||||
}
|
}
|
||||||
|
|
||||||
self.saveUpdated(passwordEntity: newPasswordEntity!)
|
|
||||||
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
|
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
|
||||||
return newPasswordEntity
|
return newPasswordEntity
|
||||||
}
|
}
|
||||||
|
|
||||||
private func deletePasswordEntities(passwordEntity: PasswordEntity) throws {
|
private func deletePasswordEntities(passwordEntity: PasswordEntity) throws {
|
||||||
var current: PasswordEntity? = passwordEntity
|
var current: PasswordEntity? = passwordEntity
|
||||||
while current != nil && (current!.children!.count == 0 || !current!.isDir) {
|
while current != nil && (current!.children!.count == 0 || !current!.isDir) {
|
||||||
|
|
@ -659,7 +707,7 @@ public class PasswordStore {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public func saveUpdated(passwordEntity: PasswordEntity) {
|
public func saveUpdated(passwordEntity: PasswordEntity) {
|
||||||
do {
|
do {
|
||||||
try context.save()
|
try context.save()
|
||||||
|
|
@ -667,11 +715,11 @@ public class PasswordStore {
|
||||||
fatalError("FailedToSavePasswordEntity".localize(error))
|
fatalError("FailedToSavePasswordEntity".localize(error))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public func deleteCoreData(entityName: String) {
|
public func deleteCoreData(entityName: String) {
|
||||||
let deleteFetchRequest = NSFetchRequest<NSFetchRequestResult>(entityName: entityName)
|
let deleteFetchRequest = NSFetchRequest<NSFetchRequestResult>(entityName: entityName)
|
||||||
let deleteRequest = NSBatchDeleteRequest(fetchRequest: deleteFetchRequest)
|
let deleteRequest = NSBatchDeleteRequest(fetchRequest: deleteFetchRequest)
|
||||||
|
|
||||||
do {
|
do {
|
||||||
try context.execute(deleteRequest)
|
try context.execute(deleteRequest)
|
||||||
try context.save()
|
try context.save()
|
||||||
|
|
@ -680,7 +728,7 @@ public class PasswordStore {
|
||||||
print(error)
|
print(error)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public func updateImage(passwordEntity: PasswordEntity, image: Data?) {
|
public func updateImage(passwordEntity: PasswordEntity, image: Data?) {
|
||||||
guard let image = image else {
|
guard let image = image else {
|
||||||
return
|
return
|
||||||
|
|
@ -703,28 +751,28 @@ public class PasswordStore {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public func erase() {
|
public func erase() {
|
||||||
publicKey = nil
|
publicKey = nil
|
||||||
privateKey = nil
|
privateKey = nil
|
||||||
try? fm.removeItem(at: storeURL)
|
try? fm.removeItem(at: storeURL)
|
||||||
try? fm.removeItem(at: tempStoreURL)
|
try? fm.removeItem(at: tempStoreURL)
|
||||||
|
|
||||||
try? fm.removeItem(atPath: Globals.pgpPublicKeyPath)
|
try? fm.removeItem(atPath: Globals.pgpPublicKeyPath)
|
||||||
try? fm.removeItem(atPath: Globals.pgpPrivateKeyPath)
|
try? fm.removeItem(atPath: Globals.pgpPrivateKeyPath)
|
||||||
try? fm.removeItem(atPath: Globals.gitSSHPrivateKeyPath)
|
try? fm.removeItem(atPath: Globals.gitSSHPrivateKeyPath)
|
||||||
|
|
||||||
Utils.removeAllKeychain()
|
Utils.removeAllKeychain()
|
||||||
|
|
||||||
deleteCoreData(entityName: "PasswordEntity")
|
deleteCoreData(entityName: "PasswordEntity")
|
||||||
|
|
||||||
SharedDefaults.removeAll()
|
SharedDefaults.removeAll()
|
||||||
storeRepository = nil
|
storeRepository = nil
|
||||||
|
|
||||||
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
|
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
|
||||||
NotificationCenter.default.post(name: .passwordStoreErased, object: nil)
|
NotificationCenter.default.post(name: .passwordStoreErased, object: nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
// return the number of discarded commits
|
// return the number of discarded commits
|
||||||
public func reset() throws -> Int {
|
public func reset() throws -> Int {
|
||||||
guard let storeRepository = storeRepository else {
|
guard let storeRepository = storeRepository else {
|
||||||
|
|
@ -742,7 +790,7 @@ public class PasswordStore {
|
||||||
try storeRepository.reset(to: newHead, resetType: .hard)
|
try storeRepository.reset(to: newHead, resetType: .hard)
|
||||||
self.setAllSynced()
|
self.setAllSynced()
|
||||||
self.updatePasswordEntityCoreData()
|
self.updatePasswordEntityCoreData()
|
||||||
|
|
||||||
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
|
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
|
||||||
NotificationCenter.default.post(name: .passwordStoreChangeDiscarded, object: nil)
|
NotificationCenter.default.post(name: .passwordStoreChangeDiscarded, object: nil)
|
||||||
return localCommits.count
|
return localCommits.count
|
||||||
|
|
@ -750,8 +798,8 @@ public class PasswordStore {
|
||||||
return 0 // no new commit
|
return 0 // no new commit
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private func getLocalCommits() throws -> [GTCommit]? {
|
private func getLocalCommits() throws -> [GTCommit]? {
|
||||||
guard let storeRepository = storeRepository else {
|
guard let storeRepository = storeRepository else {
|
||||||
throw AppError.RepositoryNotSet
|
throw AppError.RepositoryNotSet
|
||||||
|
|
@ -765,13 +813,13 @@ public class PasswordStore {
|
||||||
guard remoteBranch.oid != nil else {
|
guard remoteBranch.oid != nil else {
|
||||||
throw AppError.RepositoryRemoteBranchNotFound(remoteBranchName)
|
throw AppError.RepositoryRemoteBranchNotFound(remoteBranchName)
|
||||||
}
|
}
|
||||||
|
|
||||||
// get a list of local commits
|
// get a list of local commits
|
||||||
return try storeRepository.localCommitsRelative(toRemoteBranch: remoteBranch)
|
return try storeRepository.localCommitsRelative(toRemoteBranch: remoteBranch)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
public func decrypt(passwordEntity: PasswordEntity, requestPGPKeyPassphrase: () -> String) throws -> Password? {
|
public func decrypt(passwordEntity: PasswordEntity, requestPGPKeyPassphrase: () -> String) throws -> Password? {
|
||||||
let encryptedDataPath = storeURL.appendingPathComponent(passwordEntity.getPath())
|
let encryptedDataPath = storeURL.appendingPathComponent(passwordEntity.getPath())
|
||||||
let encryptedData = try Data(contentsOf: encryptedDataPath)
|
let encryptedData = try Data(contentsOf: encryptedDataPath)
|
||||||
|
|
@ -779,27 +827,24 @@ public class PasswordStore {
|
||||||
if passphrase == nil {
|
if passphrase == nil {
|
||||||
passphrase = requestPGPKeyPassphrase()
|
passphrase = requestPGPKeyPassphrase()
|
||||||
}
|
}
|
||||||
let decryptedData = try ObjectivePGP.decrypt(encryptedData, andVerifySignature: false, using: keyring.keys, passphraseForKey: {(_) in passphrase})
|
let decryptedData = privateKey?.decrypt(encryptedData, passphrase: passphrase)
|
||||||
let plainText = String(data: decryptedData, encoding: .utf8) ?? ""
|
let plainText = String(data: decryptedData!, encoding: .utf8) ?? ""
|
||||||
guard let url = passwordEntity.getURL() else {
|
guard let url = passwordEntity.getURL() else {
|
||||||
throw AppError.Decryption
|
throw AppError.Decryption
|
||||||
}
|
}
|
||||||
return Password(name: passwordEntity.getName(), url: url, plainText: plainText)
|
return Password(name: passwordEntity.getName(), url: url, plainText: plainText)
|
||||||
}
|
}
|
||||||
|
|
||||||
public func encrypt(password: Password) throws -> Data {
|
public func encrypt(password: Password) throws -> Data {
|
||||||
guard keyring.keys.count > 0 else {
|
guard publicKey != nil else {
|
||||||
throw AppError.PgpPublicKeyNotExist
|
throw AppError.PgpPublicKeyNotExist
|
||||||
}
|
}
|
||||||
let plainData = password.plainData
|
let plainData = password.plainData
|
||||||
let encryptedData = try ObjectivePGP.encrypt(plainData, addSignature: false, using: keyring.keys, passphraseForKey: nil)
|
let encryptedData = publicKey?.encrypt(plainData, armor: SharedDefaults[.encryptInArmored])
|
||||||
if SharedDefaults[.encryptInArmored] {
|
|
||||||
return Armor.armored(encryptedData, as: .message).data(using: .utf8)!
|
return encryptedData!
|
||||||
} else {
|
|
||||||
return encryptedData
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public func removePGPKeys() {
|
public func removePGPKeys() {
|
||||||
try? fm.removeItem(atPath: Globals.pgpPublicKeyPath)
|
try? fm.removeItem(atPath: Globals.pgpPublicKeyPath)
|
||||||
try? fm.removeItem(atPath: Globals.pgpPrivateKeyPath)
|
try? fm.removeItem(atPath: Globals.pgpPrivateKeyPath)
|
||||||
|
|
@ -809,18 +854,17 @@ public class PasswordStore {
|
||||||
SharedDefaults.remove(.pgpPrivateKeyURL)
|
SharedDefaults.remove(.pgpPrivateKeyURL)
|
||||||
SharedDefaults.remove(.pgpPublicKeyURL)
|
SharedDefaults.remove(.pgpPublicKeyURL)
|
||||||
Utils.removeKeychain(name: ".pgpKeyPassphrase")
|
Utils.removeKeychain(name: ".pgpKeyPassphrase")
|
||||||
keyring.deleteAll()
|
|
||||||
publicKey = nil
|
publicKey = nil
|
||||||
privateKey = nil
|
privateKey = nil
|
||||||
}
|
}
|
||||||
|
|
||||||
public func removeGitSSHKeys() {
|
public func removeGitSSHKeys() {
|
||||||
try? fm.removeItem(atPath: Globals.gitSSHPrivateKeyPath)
|
try? fm.removeItem(atPath: Globals.gitSSHPrivateKeyPath)
|
||||||
Defaults.remove(.gitSSHPrivateKeyArmor)
|
Defaults.remove(.gitSSHPrivateKeyArmor)
|
||||||
Defaults.remove(.gitSSHPrivateKeyURL)
|
Defaults.remove(.gitSSHPrivateKeyURL)
|
||||||
self.gitSSHPrivateKeyPassphrase = nil
|
self.gitSSHPrivateKeyPassphrase = nil
|
||||||
}
|
}
|
||||||
|
|
||||||
public func gitSSHKeyExists(inFileSharing: Bool = false) -> Bool {
|
public func gitSSHKeyExists(inFileSharing: Bool = false) -> Bool {
|
||||||
if inFileSharing == false {
|
if inFileSharing == false {
|
||||||
return fm.fileExists(atPath: Globals.gitSSHPrivateKeyPath)
|
return fm.fileExists(atPath: Globals.gitSSHPrivateKeyPath)
|
||||||
|
|
@ -828,7 +872,7 @@ public class PasswordStore {
|
||||||
return fm.fileExists(atPath: Globals.iTunesFileSharingSSHPrivate)
|
return fm.fileExists(atPath: Globals.iTunesFileSharingSSHPrivate)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public func pgpKeyExists(inFileSharing: Bool = false) -> Bool {
|
public func pgpKeyExists(inFileSharing: Bool = false) -> Bool {
|
||||||
if inFileSharing == false {
|
if inFileSharing == false {
|
||||||
return fm.fileExists(atPath: Globals.pgpPublicKeyPath) && fm.fileExists(atPath: Globals.pgpPrivateKeyPath)
|
return fm.fileExists(atPath: Globals.pgpPublicKeyPath) && fm.fileExists(atPath: Globals.pgpPrivateKeyPath)
|
||||||
|
|
@ -836,19 +880,12 @@ public class PasswordStore {
|
||||||
return fm.fileExists(atPath: Globals.iTunesFileSharingPGPPublic) && fm.fileExists(atPath: Globals.iTunesFileSharingPGPPrivate)
|
return fm.fileExists(atPath: Globals.iTunesFileSharingPGPPublic) && fm.fileExists(atPath: Globals.iTunesFileSharingPGPPrivate)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public func gitSSHKeyImportFromFileSharing() throws {
|
public func gitSSHKeyImportFromFileSharing() throws {
|
||||||
if gitSSHKeyExists() {
|
|
||||||
try fm.removeItem(atPath: Globals.gitSSHPrivateKeyPath)
|
|
||||||
}
|
|
||||||
try fm.moveItem(atPath: Globals.iTunesFileSharingSSHPrivate, toPath: Globals.gitSSHPrivateKeyPath)
|
try fm.moveItem(atPath: Globals.iTunesFileSharingSSHPrivate, toPath: Globals.gitSSHPrivateKeyPath)
|
||||||
}
|
}
|
||||||
|
|
||||||
public func pgpKeyImportFromFileSharing() throws {
|
public func pgpKeyImportFromFileSharing() throws {
|
||||||
if pgpKeyExists() {
|
|
||||||
try fm.removeItem(atPath: Globals.pgpPublicKeyPath)
|
|
||||||
try fm.removeItem(atPath: Globals.pgpPrivateKeyPath)
|
|
||||||
}
|
|
||||||
try fm.moveItem(atPath: Globals.iTunesFileSharingPGPPublic, toPath: Globals.pgpPublicKeyPath)
|
try fm.moveItem(atPath: Globals.iTunesFileSharingPGPPublic, toPath: Globals.pgpPublicKeyPath)
|
||||||
try fm.moveItem(atPath: Globals.iTunesFileSharingPGPPrivate, toPath: Globals.pgpPrivateKeyPath)
|
try fm.moveItem(atPath: Globals.iTunesFileSharingPGPPrivate, toPath: Globals.pgpPrivateKeyPath)
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue