forked from jdfwarrior/Workflows
-
Notifications
You must be signed in to change notification settings - Fork 0
/
workflows.php
488 lines (443 loc) · 12.5 KB
/
workflows.php
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
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
<?php
/**
* Name: Workflows
* Description: This PHP class object provides several useful functions for retrieving, parsing,
* and formatting data to be used with Alfred 2 Workflows.
* Author: David Ferguson (@jdfwarrior)
* Revised: 6/6/2013
* Version: 0.3.3
*/
class Workflows {
private $cache;
private $data;
private $bundle;
private $path;
private $home;
private $results;
/**
* Description:
* Class constructor function. Intializes all class variables. Accepts one optional parameter
* of the workflow bundle id in the case that you want to specify a different bundle id. This
* would adjust the output directories for storing data.
*
* @param $bundleid - optional bundle id if not found automatically
* @return none
*/
function __construct( $bundleid=null )
{
$this->path = exec('pwd');
$this->home = exec('printf "$HOME"');
if ( file_exists( 'info.plist' ) ):
$this->bundle = $this->get( 'bundleid', 'info.plist' );
endif;
if ( !is_null( $bundleid ) ):
$this->bundle = $bundleid;
endif;
/*
* if file not exist in MacOs Monterey
* create it by yourself
* mkdir '/Library/Caces/com.runningwithcrayons.Alfred-2/Workflow Data/'
* mkdir '/Library/Application Support/Alfred 2/Workflow Data/'
*/
$this->cache = $this->home. "/Library/Caches/com.runningwithcrayons.Alfred-2/Workflow Data/".$this->bundle;
$this->data = $this->home. "/Library/Application Support/Alfred 2/Workflow Data/".$this->bundle;
if ( !file_exists( $this->cache ) ):
exec("mkdir '".$this->cache."'");
endif;
if ( !file_exists( $this->data ) ):
exec("mkdir '".$this->data."'");
endif;
$this->results = array();
}
/**
* Description:
* Accepts no parameter and returns the value of the bundle id for the current workflow.
* If no value is available, then false is returned.
*
* @param none
* @return false if not available, bundle id value if available.
*/
public function bundle()
{
if ( is_null( $this->bundle ) ):
return false;
else:
return $this->bundle;
endif;
}
/**
* Description:
* Accepts no parameter and returns the value of the path to the cache directory for your
* workflow if it is available. Returns false if the value isn't available.
*
* @param none
* @return false if not available, path to the cache directory for your workflow if available.
*/
public function cache()
{
if ( is_null( $this->bundle ) ):
return false;
else:
if ( is_null( $this->cache ) ):
return false;
else:
return $this->cache;
endif;
endif;
}
/**
* Description:
* Accepts no parameter and returns the value of the path to the storage directory for your
* workflow if it is available. Returns false if the value isn't available.
*
* @param none
* @return false if not available, path to the storage directory for your workflow if available.
*/
public function data()
{
if ( is_null( $this->bundle ) ):
return false;
else:
if ( is_null( $this->data ) ):
return false;
else:
return $this->data;
endif;
endif;
}
/**
* Description:
* Accepts no parameter and returns the value of the path to the current directory for your
* workflow if it is available. Returns false if the value isn't available.
*
* @param none
* @return false if not available, path to the current directory for your workflow if available.
*/
public function path()
{
if ( is_null( $this->path ) ):
return false;
else:
return $this->path;
endif;
}
/**
* Description:
* Accepts no parameter and returns the value of the home path for the current user
* Returns false if the value isn't available.
*
* @param none
* @return false if not available, home path for the current user if available.
*/
public function home()
{
if ( is_null( $this->home ) ):
return false;
else:
return $this->home;
endif;
}
/**
* Description:
* Returns an array of available result items
*
* @param none
* @return array - list of result items
*/
public function results()
{
return $this->results;
}
/**
* Description:
* Convert an associative array into XML format
*
* @param $a - An associative array to convert
* @param $format - format of data being passed (json or array), defaults to array
* @return - XML string representation of the array
*/
public function toxml( $a=null, $format='array' ) {
if ( $format == 'json' ):
$a = json_decode( $a, TRUE );
endif;
if ( is_null( $a ) && !empty( $this->results ) ):
$a = $this->results;
elseif ( is_null( $a ) && empty( $this->results ) ):
return false;
endif;
$items = new SimpleXMLElement("<items></items>"); // Create new XML element
foreach( $a as $b ): // Lop through each object in the array
$c = $items->addChild( 'item' ); // Add a new 'item' element for each object
$c_keys = array_keys( $b ); // Grab all the keys for that item
foreach( $c_keys as $key ): // For each of those keys
if ( $key == 'uid' ):
if ( $b[$key] === null || $b[$key] === '' ):
continue;
else:
$c->addAttribute( 'uid', $b[$key] );
endif;
elseif ( $key == 'arg' ):
$c->addAttribute( 'arg', $b[$key] );
$c->$key = $b[$key];
elseif ( $key == 'type' ):
$c->addAttribute( 'type', $b[$key] );
elseif ( $key == 'valid' ):
if ( $b[$key] == 'yes' || $b[$key] == 'no' ):
$c->addAttribute( 'valid', $b[$key] );
endif;
elseif ( $key == 'autocomplete' ):
if ( $b[$key] === null || $b[$key] === '' ):
continue;
else:
$c->addAttribute( 'autocomplete', $b[$key] == null ? '' : $b[$key] );
endif;
elseif ( $key == 'icon' ):
if ( substr( $b[$key], 0, 9 ) == 'fileicon:' ):
$val = substr( $b[$key], 9 );
$c->$key = $val;
$c->$key->addAttribute( 'type', 'fileicon' );
elseif ( substr( $b[$key], 0, 9 ) == 'filetype:' ):
$val = substr( $b[$key], 9 );
$c->$key = $val;
$c->$key->addAttribute( 'type', 'filetype' );
else:
$c->$key = $b[$key];
endif;
else:
$c->$key = $b[$key];
endif;
endforeach;
endforeach;
return $items->asXML(); // Return XML string representation of the array
}
/**
* Description:
* Remove all items from an associative array that do not have a value
*
* @param $a - Associative array
* @return bool
*/
private function empty_filter( $a ) {
if ( $a == '' || $a == null ): // if $a is empty or null
return false; // return false, else, return true
else:
return true;
endif;
}
/**
* Description:
* Save values to a specified plist. If the first parameter is an associative
* array, then the second parameter becomes the plist file to save to. If the
* first parameter is string, then it is assumed that the first parameter is
* the label, the second parameter is the value, and the third parameter is
* the plist file to save the data to.
*
* @param $a - associative array of values to save
* @param $b - the value of the setting
* @param $c - the plist to save the values into
* @return string - execution output
*/
public function set( $a=null, $b=null, $c=null )
{
if ( is_array( $a ) ):
if ( file_exists( $b ) ):
if ( file_exists( $this->path.'/'.$b ) ):
$b = $this->path.'/'.$b;
endif;
elseif ( file_exists( $this->data."/".$b ) ):
$b = $this->data."/".$b;
elseif ( file_exists( $this->cache."/".$b ) ):
$b = $this->cache."/".$b;
else:
$b = $this->data."/".$b;
endif;
else:
if ( file_exists( $c ) ):
if ( file_exists( $this->path.'/'.$c ) ):
$c = $this->path.'/'.$c;
endif;
elseif ( file_exists( $this->data."/".$c ) ):
$c = $this->data."/".$c;
elseif ( file_exists( $this->cache."/".$c ) ):
$c = $this->cache."/".$c;
else:
$c = $this->data."/".$c;
endif;
endif;
if ( is_array( $a ) ):
foreach( $a as $k => $v ):
exec( 'defaults write "'. $b .'" '. $k .' "'. $v .'"');
endforeach;
else:
exec( 'defaults write "'. $c .'" '. $a .' "'. $b .'"');
endif;
}
/**
* Description:
* Read a value from the specified plist
*
* @param $a - the value to read
* @param $b - plist to read the values from
* @return bool false if not found, string if found
*/
public function get( $a, $b ) {
if ( file_exists( $b ) ):
if ( file_exists( $this->path.'/'.$b ) ):
$b = $this->path.'/'.$b;
endif;
elseif ( file_exists( $this->data."/".$b ) ):
$b = $this->data."/".$b;
elseif ( file_exists( $this->cache."/".$b ) ):
$b = $this->cache."/".$b;
else:
return false;
endif;
exec( 'defaults read "'. $b .'" '.$a, $out ); // Execute system call to read plist value
if ( $out == "" ):
return false;
endif;
$out = $out[0];
return $out; // Return item value
}
/**
* Description:
* Read data from a remote file/url, essentially a shortcut for curl
*
* @param $url - URL to request
* @param $options - Array of curl options
* @return result from curl_exec
*/
public function request( $url=null, $options=null )
{
if ( is_null( $url ) ):
return false;
endif;
$defaults = array( // Create a list of default curl options
CURLOPT_RETURNTRANSFER => true, // Returns the result as a string
CURLOPT_URL => $url, // Sets the url to request
CURLOPT_FRESH_CONNECT => true
);
if ( $options ):
foreach( $options as $k => $v ):
$defaults[$k] = $v;
endforeach;
endif;
array_filter( $defaults, // Filter out empty options from the array
array( $this, 'empty_filter' ) );
$ch = curl_init(); // Init new curl object
curl_setopt_array( $ch, $defaults ); // Set curl options
$out = curl_exec( $ch ); // Request remote data
$err = curl_error( $ch );
curl_close( $ch ); // End curl request
if ( $err ):
return $err;
else:
return $out;
endif;
}
/**
* Description:
* Allows searching the local hard drive using mdfind
*
* @param $query - search string
* @return array - array of search results
*/
public function mdfind( $query )
{
exec('mdfind "'.$query.'"', $results);
return $results;
}
/**
* Description:
* Accepts data and a string file name to store data to local file as cache
*
* @param array - data to save to file
* @param file - filename to write the cache data to
* @return none
*/
public function write( $a, $b )
{
if ( file_exists( $b ) ):
if ( file_exists( $this->path.'/'.$b ) ):
$b = $this->path.'/'.$b;
endif;
elseif ( file_exists( $this->data."/".$b ) ):
$b = $this->data."/".$b;
elseif ( file_exists( $this->cache."/".$b ) ):
$b = $this->cache."/".$b;
else:
$b = $this->data."/".$b;
endif;
if ( is_array( $a ) ):
$a = json_encode( $a );
file_put_contents( $b, $a );
return true;
elseif ( is_string( $a ) ):
file_put_contents( $b, $a );
return true;
else:
return false;
endif;
}
/**
* Description:
* Returns data from a local cache file
*
* @param file - filename to read the cache data from
* @return false if the file cannot be found, the file data if found. If the file
* format is json encoded, then a json object is returned.
*/
public function read( $a, $array = false )
{
if ( file_exists( $a ) ):
if ( file_exists( $this->path.'/'.$a ) ):
$a = $this->path.'/'.$a;
endif;
elseif ( file_exists( $this->data."/".$a ) ):
$a = $this->data."/".$a;
elseif ( file_exists( $this->cache."/".$a ) ):
$a = $this->cache."/".$a;
else:
return false;
endif;
$out = file_get_contents( $a );
if ( !is_null( json_decode( $out ) ) && !$array ):
$out = json_decode( $out );
elseif ( !is_null( json_decode( $out ) ) && !$array ):
$out = json_decode( $out, true );
endif;
return $out;
}
/**
* Description:
* Helper function that just makes it easier to pass values into a function
* and create an array result to be passed back to Alfred
*
* @param $uid - the uid of the result, should be unique
* @param $arg - the argument that will be passed on
* @param $title - The title of the result item
* @param $sub - The subtitle text for the result item
* @param $icon - the icon to use for the result item
* @param $valid - sets whether the result item can be actioned
* @param $auto - the autocomplete value for the result item
* @return array - array item to be passed back to Alfred
*/
public function result( $uid, $arg, $title, $sub, $icon, $valid='yes', $auto=null, $type=null )
{
$temp = array(
'uid' => $uid,
'arg' => $arg,
'title' => $title,
'subtitle' => $sub,
'icon' => $icon,
'valid' => $valid,
'autocomplete' => $auto,
'type' => $type
);
if ( is_null( $type ) ):
unset( $temp['type'] );
endif;
array_push( $this->results, $temp );
return $temp;
}
}