Skip to content

Commit 5ed9576

Browse files
authored
Adding functionalities: 'update' and 'remove' item from ios keychain. (#584)
1 parent 6c55d7e commit 5ed9576

File tree

5 files changed

+94
-0
lines changed

5 files changed

+94
-0
lines changed

agent/src/ios/keychain.ts

+33
Original file line numberDiff line numberDiff line change
@@ -170,6 +170,39 @@ export const empty = (): void => {
170170
});
171171
};
172172

173+
// remove matching itemms from the keychain
174+
export const remove = (account: string, service: string): void => {
175+
const searchDictionary: NSMutableDictionaryType = ObjC.classes.NSMutableDictionary.alloc().init();
176+
searchDictionary.setObject_forKey_(kSec.kSecAttrSynchronizableAny, kSec.kSecAttrSynchronizable);
177+
itemClasses.forEach((clazz) => {
178+
// set the class-type we are querying for now & delete
179+
searchDictionary.setObject_forKey_(clazz, kSec.kSecClass);
180+
searchDictionary.setObject_forKey_(account, kSec.kSecAttrAccount);
181+
searchDictionary.setObject_forKey_(service, kSec.kSecAttrService);
182+
libObjc.SecItemDelete(searchDictionary);
183+
});
184+
};
185+
186+
// update matching item from the keychain
187+
export const update = (account: string, service: string, newData: string): void => {
188+
189+
const searchDictionary: NSMutableDictionaryType = ObjC.classes.NSMutableDictionary.alloc().init();
190+
searchDictionary.setObject_forKey_(kSec.kSecAttrSynchronizableAny, kSec.kSecAttrSynchronizable);
191+
192+
// set the class-type we are querying for now & update
193+
searchDictionary.setObject_forKey_(kSec.kSecClassGenericPassword, kSec.kSecClass);
194+
searchDictionary.setObject_forKey_(account, kSec.kSecAttrAccount);
195+
searchDictionary.setObject_forKey_(service, kSec.kSecAttrService);
196+
197+
// set the dictionary with new value
198+
const itemDict: NSMutableDictionaryType = ObjC.classes.NSMutableDictionary.alloc().init();
199+
const v: NSStringType = ObjC.classes.NSString.stringWithString_(newData).dataUsingEncoding_(NSUTF8StringEncoding);
200+
itemDict.setObject_forKey_(account, kSec.kSecAttrAccount);
201+
itemDict.setObject_forKey_(v, kSec.kSecValueData);
202+
libObjc.SecItemUpdate(searchDictionary, itemDict);
203+
204+
};
205+
173206
// add a string entry to the keychain
174207
export const add = (account: string, service: string, data: string): boolean => {
175208

agent/src/ios/lib/libobjc.ts

+7
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,12 @@ const nativeExports: any = {
2424
moduleName: "Security",
2525
retType: "pointer",
2626
},
27+
SecItemUpdate: {
28+
argTypes: ["pointer", "pointer"],
29+
exportName: "SecItemUpdate",
30+
moduleName: "Security",
31+
retType: "pointer",
32+
},
2733

2834
// SSL pinning methods
2935
SSLCreateContext: {
@@ -86,6 +92,7 @@ const api: any = {
8692
SecAccessControlGetConstraints: null,
8793
SecItemAdd: null,
8894
SecItemCopyMatching: null,
95+
SecItemUpdate: null,
8996
SecItemDelete: null,
9097

9198
SSLCreateContext: null,

agent/src/rpc/ios.ts

+3
Original file line numberDiff line numberDiff line change
@@ -96,6 +96,9 @@ export const ios = {
9696
// ios keychain
9797
iosKeychainAdd: (account: string, service: string, data: string): boolean =>
9898
ioskeychain.add(account, service, data),
99+
iosKeychainRemove: (account: string, service: string): void => ioskeychain.remove(account, service),
100+
iosKeychainUpdate: (account: string, service: string, newData: string): void =>
101+
ioskeychain.update(account, service, newData),
99102
iosKeychainEmpty: (): void => ioskeychain.empty(),
100103
iosKeychainList: (smartDecode): IKeychainItem[] => ioskeychain.list(smartDecode),
101104
iosKeychainListRaw: (): void => ioskeychain.listRaw(),

objection/commands/ios/keychain.py

+41
Original file line numberDiff line numberDiff line change
@@ -138,6 +138,47 @@ def clear(args: list = None) -> None:
138138
click.secho('Keychain cleared', fg='green')
139139

140140

141+
def remove(args: list) -> None:
142+
"""
143+
Remove matching keychain entries from the keychain
144+
145+
:param args:
146+
:return:
147+
"""
148+
149+
account = _get_flag_value(args, '--account')
150+
service = _get_flag_value(args, '--service')
151+
152+
click.secho('Removing entry from the iOS keychain...', dim=True)
153+
click.secho('Account: {0}'.format(account), dim=True)
154+
click.secho('Service: {0}'.format(service), dim=True)
155+
156+
api = state_connection.get_api()
157+
api.ios_keychain_remove(account, service);
158+
click.secho('Successfully removed matching keychain items', fg='green')
159+
160+
161+
def update(args: list) -> None:
162+
"""
163+
Update matching keychain entry from the keychain
164+
165+
:param args:
166+
:return:
167+
"""
168+
169+
account = _get_flag_value(args, '--account')
170+
service = _get_flag_value(args, '--service')
171+
newData = _get_flag_value(args, '--newData')
172+
173+
click.secho('Updating entries from the iOS keychain...', dim=True)
174+
click.secho('Account: {0}'.format(account), dim=True)
175+
click.secho('Service: {0}'.format(service), dim=True)
176+
click.secho('New Data: {0}'.format(newData), dim=True)
177+
178+
api = state_connection.get_api()
179+
api.ios_keychain_update(account, service, newData);
180+
click.secho('Successfully updated matching keychain item', fg='green')
181+
141182
def add(args: list) -> None:
142183
"""
143184
Adds a new kSecClassGenericPassword keychain entry to the keychain

objection/console/commands.py

+10
Original file line numberDiff line numberDiff line change
@@ -527,6 +527,16 @@
527527
'meta': 'Delete all keychain entries for the current app\'s entitlement group',
528528
'exec': keychain.clear
529529
},
530+
'remove': {
531+
'meta': 'Remove an entry from the iOS keychain',
532+
'flags': ['--account', '--service'],
533+
'exec': keychain.remove
534+
},
535+
'update': {
536+
'meta': 'Update an entry from the iOS keychain',
537+
'flags': ['--account', '--service', '--newData'],
538+
'exec': keychain.update
539+
},
530540
'add': {
531541
'meta': 'Add an entry to the iOS keychain',
532542
'flags': ['--account', '--service', '--data'],

0 commit comments

Comments
 (0)