forked from davinerd/MacTomb
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmactomb.sh
828 lines (696 loc) · 22.3 KB
/
mactomb.sh
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
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
#!/bin/bash
##############################################################################
# MacTomb #
# Copyright (C) 2015 Davide `Anathema` Barbato #
# #
# This program is free software; you can redistribute it and/or modify #
# it under the terms of the GNU General Public License as published by #
# the Free Software Foundation; either version 3 of the License. #
# #
# This program is distributed in the hope that it will be useful, #
# but WITHOUT ANY WARRANTY; without even the implied warranty of #
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the #
# GNU General Public License for more details. #
# #
# You should have received a copy of the GNU General Public License #
# along with this program; if not, write to the Free Software Foundation, #
# Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA #
##############################################################################
banner() {
echo "..:: MacTomb v$VERSION ::.."
echo -e "by Davide Barbato\n"
}
e_echo() {
echo -e "[x] $1"
}
s_echo() {
echo -e "[*] $1"
}
p_echo() {
echo -e "[-] $1"
}
usage() {
banner
echo -e "Usage:"
echo -e "${0} <${COMMAND[@]}> -v\n"
echo -e "-v\tMac OS X Notification"
echo -e "Use 'help' to show the full help"
}
help() {
banner
echo "Help!"
echo -e '''
list:
list all opened mactombs\n
chpass:
-f <file>\t\tChange passphrase of mactomb <file>\n
compress:
-f <file>\t\tCompress a mactomb <file> (will make it read-only)\n
decompress:
-f <file>\t\tDecompress a mactomb <file>\n
rename:
-f <file>\t\tmactomb file (already created)
Optional:
-n <volname>\tSpecify the new volume name to assign to the mactomb <file> (default is "untitled")
-b <script>\t\tThe bash script in which replaces all the occurence of the old volum name with the new one\n
encrypt:
-f <file>\t\tmactomb file to encrypt\n
create:
-f <file>\t\tFile to create (the mactomb file)
-s <size[m|g|t]\tSize of the file (m=mb, g=gb, t=tb)
Optional:
-p <profile>\tFolder/file to copy into the newly created mactomb <file>
-c\t\t\tCreate a zlib compressed mactomb <file> (will make it read-only)
-n <volname>\tSpecify the volume name to assign to the mactomb <file>\n
app:
-f <file>\tEncrypted DMG to use as mactomb file (already created)
-a <app>\tBinary and arguments of the app you want to use inside the mactomb file
-b <script>\tThe bash script used to launch the <app> inside the mactomb file <file>\n
forge:
Will call both "create" and "app" if all flags are specified. Can be called on \n already created files, in this case skipping "create" and/or "app"
Optional:
-o <app>\tThe Automator app used to launch the bash <script> by Mac OS X\n
Example
bash $0 forge -f ~/mytomb.dmg -s 100m -a "/Applications/Firefox.app/Contents/MacOS/firefox-bin -p secure_profile" -b ~/run.sh -o ~/runmy.app
'''
return 2
}
compression_banner() {
echo '''
##################################################
# WARNING #
# #
# Compression will make the mactomb read-only #
# and DMG #
# #
##################################################
'''
}
decompression_banner() {
echo '''
##################################################
# WARNING #
# #
# Decompression will overwrite your compressed #
# mactomb #
# #
##################################################
'''
}
sparsebundle_banner() {
echo '''
##################################################
# WARNING #
# #
# The size specified on the command line is the #
# MAXIMUM size the tomb can reach. #
# #
# Resize DOES NOT work on sparsebundle tomb #
# #
##################################################
'''
}
check_size() {
local dim=${SIZE:~0}
local size=${SIZE%?}
# check if the size ends with a well know character and if it's a number
if [[ "(m, g, t)" =~ "$dim" ]] && [[ "$size" =~ ^[0-9]+$ ]]; then
return 0
fi
return 1
}
check_file() {
local tombfile="${FILENAME}"
if [ ! "${tombfile}" ]; then
F_MESSAGE="Please specify the filename (-f)"
return 1
fi
if [ ! -e "${tombfile}" ]; then
F_MESSAGE+="'${tombfile}' not found."
return 1
fi
# this check is not really strong, but I was not able to find another better one
if [[ -d "${tombfile}" && "${tombfile##*.}" != "sparsebundle" ]]; then
F_MESSAGE+="'${tombfile}' is a directory."
return 1
fi
return 0
}
list() {
local PLISTBUDDY="/usr/libexec/PlistBuddy"
# colours! even if I don't need all of them...colours!
local BLUE="\x1b[0;34m"
local RED="\x1b[0;31m"
local GREEN="\x1b[0;32m"
local YELLOW="\x1b[1;33m"
local WHITE="\x1b[1;37m"
local LIGHT_RED="\x1b[1;31m"
local LIGHT_GREEN="\x1b[1;32m"
local LIGHT_BLUE="\x1b[1;34m"
local LIGHT_CYAN="\x1b[1;36m"
local NO_COLOUR="\x1b[0m"
local mountpoint mnt space_tot used avail perc oid
if [ ! -x "$PLISTBUDDY" ]; then
F_MESSAGE="PlistBuddy not found in $PLISTBUDDY. Maybe it's on a different path or not installed?"
return 1
fi
local tempfile=$(mktemp /tmp/$RANDOM.XXX)
${HDIUTIL} info -plist > $tempfile
local -i idx=0
local -i cnt=0
local compressed="No"
while True; do
imgpath=$(${PLISTBUDDY} -c Print:images:$idx:image-path $tempfile 2>/dev/null)
if [ ! "$imgpath" ]; then
break
fi
encrypted=$(${PLISTBUDDY} -c Print:images:$idx:image-encrypted $tempfile)
removable=$(${PLISTBUDDY} -c Print:images:$idx:removable $tempfile)
writeable=$(${PLISTBUDDY} -c Print:images:$idx:writeable $tempfile)
imgtype=$(${PLISTBUDDY} -c Print:images:$idx:image-type $tempfile)
oid=$(${PLISTBUDDY} -c Print:images:$idx:owner-uid $tempfile)
local out=$(dscl . -search /Users UniqueID $oid)
local owner=$(cut -d ' ' -f1 <<< $out)
local -i j=0
while True; do
mountpoint=$(${PLISTBUDDY} -c Print:images:$idx:system-entities:$j $tempfile 2>/dev/null)
if [ ! "$mountpoint" ]; then
# we can skip the while loop since it means there are no system-entities left
# then it doesn't contain the mount-point for sure
break
fi
mountpoint=$(${PLISTBUDDY} -c Print:images:$idx:system-entities:$j:mount-point $tempfile 2>/dev/null)
# mount-point is not mandatory inside system-entities
if [ "$mountpoint" ]; then
read -ra space_tot -d ''<<< $(df -h "$mountpoint" | awk -F ' ' '{print $2}')
read -ra used -d '' <<< $(df -h "$mountpoint" | awk -F ' ' '{print $3}')
read -ra avail -d '' <<< $(df -h "$mountpoint" | awk -F ' ' '{print $4}')
read -ra perc -d '' <<< $(df -h "$mountpoint" | awk -F ' ' '{print $5}')
# pretty lame...
mnt=$mountpoint
fi
j+=1
done
# assume that macbombs are encrypted, removable and writable.
# it's too loose, but for now it's ok
if [[ "$encrypted" == "true" && "$removable" == "true" ]] && [[ "$writeable" == "true" || "$imgtype" =~ "compressed" ]]; then
echo "***************"
echo -e "${GREEN}Image Path$NO_COLOUR:\t$imgpath"
echo -e "${GREEN}Mount Point$NO_COLOUR:\t$mnt"
if [[ "$imgtype" =~ "compressed" ]]; then
compressed="Yes ($imgtype)"
fi
echo -e "${GREEN}Compressed$NO_COLOUR:\t$compressed"
echo -e "${GREEN}${space_tot[0]}$NO_COLOUR:\t\t${space_tot[1]}"
echo -e "${GREEN}${used[0]}$NO_COLOUR:\t\t${used[1]}"
echo -e "${GREEN}${avail[0]}$NO_COLOUR:\t\t${avail[1]}"
# it's nice to have different colours based on the percentage of used space
if [ ${perc[1]%?} -ge 50 ] && [ ${perc[1]%?} -lt 80 ]; then
local colour=${YELLOW}
elif [ ${perc[1]%?} -ge 80 ]; then
local colour=${RED}
fi
echo -e "${GREEN}${perc[0]}$NO_COLOUR:\t$colour${perc[1]}$NO_COLOUR"
echo -e "${GREEN}Owner$NO_COLOUR:\t\t$owner"
cnt+=1
fi
idx+=1
done
rm -rf $tempfile
if [ $cnt -gt 0 ]; then
echo
F_MESSAGE="There are nr.$cnt mactomb(s) open"
else
F_MESSAGE="There are no mactombs opened"
fi
return 0
}
# this function can be used even for not-encrypted DMG
chpass() {
F_MESSAGE="Cannot change passphrase: "
check_file
if [ "$?" -eq 1 ]; then
return 1
fi
local ret=$(${HDIUTIL} chpass "${FILENAME}" 2>&1)
if [[ "$ret" =~ "chpass failed" ]]; then
F_MESSAGE+="$ret"
return 1
fi
F_MESSAGE="Successfully changed your passphrase!"
return 0
}
# this function can be used even for not-encrypted DMG
rename() {
F_MESSAGE="Cannot rename '$FILENAME': "
if [ ! "${FILENAME}" ]; then
F_MESSAGE="Please specify the filename (-f)"
return 1
fi
if [ ! -e "${FILENAME}" ]; then
F_MESSAGE+="file not found."
return 1
fi
local ret disk oldlabel
# if the mactomb is already mounted we don't want to unmount it
# at the end of the renaming process
local already_mounted=$(${HDIUTIL} info | grep "$FILENAME")
# a quick check to avoid going through the process of renameing a compressed mactomb.
# if the mactomb is mounted, this check fails.
if [ ! "${already_mounted}" ]; then
p_echo "Getting information..."
ret=$(${HDIUTIL} imageinfo "${FILENAME}" 2>&1 | grep "Compressed:")
if [[ "$ret" =~ "true" ]]; then
F_MESSAGE+="compressed mactombs are read-only"
return 1
fi
fi
p_echo "Renaming to '$VOLNAME'"
# if already attached we don't care, hdiutil is smart enough
ret=$(${HDIUTIL} attach "${FILENAME}" 2>&1)
if [[ "$ret" =~ "attach failed" ]]; then
F_MESSAGE+=$ret
return 1
fi
disk=$(grep Volumes <<< "$ret" | awk -F ' ' '{print $1}')
oldlabel=$(grep Volumes <<< "$ret" | awk -F ' ' '{print $3}')
oldlabel=$(basename $oldlabel)
if [[ "$oldlabel" == "$VOLNAME" ]]; then
if [ ! "$already_mounted" ]; then
${HDIUTIL} detach "$disk" &> /dev/null
fi
F_MESSAGE+="same label"
return 1
fi
ret=$(/usr/sbin/diskutil rename "${disk}" "${VOLNAME}" 2>&1)
F_MESSAGE="Successfully renamed your mactomb!"
if [ ! "$already_mounted" ]; then
# we don't want to let it sits open on your system!
${HDIUTIL} detach "$disk" &> /dev/null
fi
# this can happens if your mactomb is compressed and already mounted
if [[ "$ret" =~ "Failed to rename volume" ]]; then
F_MESSAGE+=$ret
return 1
fi
if [ "$BASHSCRIPT" ]; then
if [ ! -w "$BASHSCRIPT" ]; then
e_echo "'$BASHSCRIPT' does not exist or you don't have write permission! Not editing"
else
# yes: it will automatically replace all the occurence of $oldlabel with $VOLNAME
sed -i '' -e "s@$oldlabel@$VOLNAME@" "${BASHSCRIPT}"
fi
fi
return 0
}
# this function can be used even for not-encrypted DMG
resize() {
F_MESSAGE="Cannot resize '$FILENAME': "
if [[ ! "${FILENAME}" || ! "${SIZE}" ]]; then
F_MESSAGE="Please specify the filename (-f) and the size (-s)"
return 1
fi
check_file
if [ "$?" -eq 1 ]; then
return 1
fi
if [ "${FILENAME##*.}" == "sparsebundle" ]; then
F_MESSAGE="Sparsebundle cannot be resized!"
return 1
fi
check_size
if [ "$?" -eq 1 ]; then
F_MESSAGE+="wrong size numer!"
return 1
fi
local num=${SIZE%?}
local new_size=$(( ($num*1024)*1024 ))
local param
# risky...but nice!
eval $(stat -s "$FILENAME")
if [ $new_size -lt $st_size ]; then
param="-shrinkonly"
elif [ $new_size -gt $st_size ]; then
param="-growonly"
else
F_MESSAGE+="size is the same. Not resizing"
return 1
fi
${HDIUTIL} resize -size ${SIZE} ${param} "${FILENAME}"
if [ "$?" -ne 0 ]; then
F_MESSAGE="Mactomb file '$FILENAME' not resized"
return 1
fi
F_MESSAGE="Mactomb file '$FILENAME' succesfully resized!"
return 0
}
compress() {
F_MESSAGE="Failed compressing the mactomb file '${FILENAME}': "
check_file
if [ "$?" -eq 1 ]; then
return 1
fi
compression_banner
local tmp="/tmp/$RANDOM$RANDOM.dmg"
p_echo "Compressing...(you'll asked to insert a new passphrase: choose a new one or insert the old one)"
ret=$(${HDIUTIL} convert "${FILENAME}" -format $CFORMAT -imagekey zlib-level=$CLEVEL -o "${tmp}" -encryption "$ENC" 2>&1)
if [[ "$ret" =~ "convert failed" || "$ret" =~ "convert canceled" ]]; then
F_MESSAGE+=$ret
return 1
fi
mv -f "$tmp" "${FILENAME}"
F_MESSAGE="Mactomb file '${FILENAME}' successfully compressed!"
return 0
}
decompress() {
F_MESSAGE="Failed decompressing the mactomb file '${FILENAME}': "
check_file
if [ "$?" -eq 1 ]; then
return 1
fi
decompression_banner
local tmp="/tmp/$RANDOM$RANDOM.dmg"
p_echo "Decompressing...(you'll asked to insert a new passphrase: choose a new one or insert the old one)"
local ret=$(${HDIUTIL} convert "${FILENAME}" -format UDRW -o "${tmp}" -encryption "$ENC" 2>&1)
if [[ "$ret" =~ "convert failed" || "$ret" =~ "convert canceled" ]]; then
F_MESSAGE+=$ret
return 1
fi
mv -f "$tmp" "${FILENAME}"
F_MESSAGE="Mactomb file '${FILENAME}' successfully decompressed!"
return 0
}
encrypt() {
F_MESSAGE="Failed encrypting the mactomb file '${FILENAME}': "
check_file
if [ "$?" -eq 1 ]; then
return 1
fi
local encr=$(${HDIUTIL} imageinfo "${FILENAME}" | grep Encrypted)
if [ ! "$encr" ]; then
F_MESSAGE+="error getting information."
return 1
elif [[ "$encr" =~ "true" ]]; then
F_MESSAGE+="already encrypted."
return 1
fi
local tmp="/tmp/$RANDOM$RANDOM.dmg"
local ret=$(${HDIUTIL} convert "${FILENAME}" -format UDRW -o "${tmp}" -encryption "$ENC" 2>&1)
if [[ "$ret" =~ "convert failed" || "$ret" =~ "convert canceled" ]]; then
rm -rf "${tmp}"
F_MESSAGE+=$ret
return 1
fi
mv -f "$tmp" "${FILENAME}"
F_MESSAGE="Mactomb file '${FILENAME}' successfully encrypted!"
return 0
}
create() {
F_MESSAGE="Failed creating the mactomb file '${FILENAME}': "
if [[ ! "${FILENAME}" || ! "$SIZE" ]]; then
F_MESSAGE="You must specify a filename and/or a size!"
return 1
fi
if [ -d "${FILENAME}" ]; then
F_MESSAGE+="File is a directory"
return 1
fi
if [[ -e "${FILENAME}" || -e "${FILENAME}".dmg || -e "${FILENAME}".sparsebundle ]]; then
F_MESSAGE+="File already exists!"
return 1
fi
check_size
if [ $? -eq 1 ]; then
F_MESSAGE+="Wrong size number!"
return 1
fi
local ret=$(mount | grep "$VOLNAME")
if [[ "$ret" ]]; then
F_MESSAGE="Volume name '$VOLNAME' already used. Please pick up a different name (-n) or unmount it"
return 1
fi
if [[ "${FILENAME##*.}" == "dmg" || "${IMGFORMAT}" == "DMG" ]]; then
IMGFORMAT="UDIF"
fi
if [[ ${IMGFORMAT} == "SPARSEBUNDLE" ]]; then
sparsebundle_banner
fi
# this can be quite huge block to read but I was not able to find a more elegant solution
if [[ "${COMPRESS}" -eq 1 && "${PROFILE}" ]]; then
compression_banner
p_echo "Creating, copying and compressing the mactomb..."
ret=$(${HDIUTIL} create "$FILENAME" -type "$IMGFORMAT" -encryption "$ENC" -size "$SIZE" -fs "$FS" -nospotlight -volname "$VOLNAME" \
-format $CFORMAT -imagekey zlib-level=$CLEVEL -srcfolder ${PROFILE} 2>&1)
if [[ "$ret" =~ "create failed" || "$ret" =~ "create canceled" ]]; then
F_MESSAGE+=$ret
return 1
fi
F_MESSAGE="mactomb file '${FILENAME}' successfully created!"
elif [[ "${COMPRESS}" -eq 0 && "${PROFILE}" ]]; then
p_echo "Creating the mactomb..."
ret=$(${HDIUTIL} create "$FILENAME" -type "$IMGFORMAT" -encryption "$ENC" -size "$SIZE" -fs "$FS" -nospotlight -volname "$VOLNAME" -attach 2>&1)
if [[ "$ret" =~ "create failed" || "$ret" =~ "attach failed" || "$ret" =~ "create canceled" ]]; then
F_MESSAGE+=$ret
return 1
fi
s_echo "mactomb file '${FILENAME}' successfully created!"
p_echo "Copying profile file(s) into the mactomb..."
local abs_vol_path="/Volumes/$VOLNAME"
# enforce a check - don't trust hdiutil
if [[ -d "$abs_vol_path" ]]; then
# do not let the script fails if cp fails
if [ -e "$PROFILE" ]; then
cp -rv "$PROFILE" "$abs_vol_path"
echo
if [ "$?" -eq 1 ]; then
e_echo "File(s) not copied!"
else
s_echo "File(s) successfully copied!"
fi
else
e_echo "Cannot find $PROFILE. File(s) not copied."
fi
# I really don't care about the exit status.
${HDIUTIL} detach "$abs_vol_path" &> /dev/null
else
F_MESSAGE="Problem mounting the mactomb file, file(s) not copied."
return 1
fi
F_MESSAGE="Enjoy your mactomb"
elif [[ ! "${PROFILE}" && "${COMPRESS}" -eq 1 ]]; then
compression_banner
# problem is: if you specify -format UDZO, hdiutil requires -srcfolder to be set.
# so we need to create a temp tomb and then compress (hdiutil convert)
local tmp="/tmp/$RANDOM$RANDOM"
# since 'convert' doesn't preserve encryption, let's create a normal container
# that will be encrypted by 'convert'
ret=$(${HDIUTIL} create "$tmp" -size "$SIZE" -fs "$FS" -nospotlight -volname "$VOLNAME" 2>&1)
if [[ "$ret" =~ "create failed" || "$ret" =~ "create canceled" ]]; then
rm -rf "${tmp}"
F_MESSAGE+=$ret
return 1
fi
ret=$(${HDIUTIL} convert "$tmp" -format $CFORMAT -imagekey zlib-level=$CLEVEL -o "${FILENAME}" -encryption "$ENC" 2>&1)
if [[ "$ret" =~ "convert failed" || "$ret" =~ "convert canceled" ]]; then
rm -rf "${tmp}"
F_MESSAGE+=$ret
return 1
fi
# removing the temp file
rm -rf "${tmp}"
F_MESSAGE="mactomb file '${FILENAME}' successfully created!"
else
ret=$(${HDIUTIL} create "$FILENAME" -type "$IMGFORMAT" -encryption "$ENC" -size "$SIZE" -fs "$FS" -nospotlight -volname "$VOLNAME" 2>&1)
if [[ "$ret" =~ "create failed" || "$ret" =~ "create canceled" ]]; then
F_MESSAGE+=$ret
return 1
fi
F_MESSAGE="mactomb file '${FILENAME}' successfully created!"
fi
return 0
}
app() {
F_MESSAGE="Failed creating the script: "
if [[ ! "$APPCMD" || ! "$BASHSCRIPT" || ! "$FILENAME" ]]; then
F_MESSAGE="Please specify -a -b -f."
return 1
fi
if [ ! -e "${FILENAME}" ]; then
F_MESSAGE+="$FILENAME file not found."
return 1
fi
# check if we included a command (with spaces) or a binary path
read -ra app_arr -d '' <<< "$APPCMD"
if [ ! -e "${app_arr[0]}" ]; then
F_MESSAGE+="Cannot find ${app_arr[0]}."
return 1
fi
if [ -d "${BASHSCRIPT}" ]; then
F_MESSAGE+="$BASHSCRIPT is a directory."
return 1
fi
if [ -e "${BASHSCRIPT}" ]; then
F_MESSAGE+="$BASHSCRIPT already exists."
return 1
fi
# ensure absolute path for the mactomb file
if [[ $(dirname "$FILENAME") == "." ]]; then
FILENAME="$(PWD)"/"${FILENAME}"
fi
# ensure absolute path for the app
if [[ $(dirname "$APPCMD") == "." ]]; then
APPCMD="$(PWD)"/"${APPCMD}"
fi
if [ -e "${APPCMD}" ]; then
F_MESSAGE+="$APPCMD already exists."
return 1
fi
if [ -e "${APPCMD}" ]; then
F_MESSAGE+="$APPCMD is a directory"
return 1
fi
local abs_vol_path="/Volumes/$VOLNAME"
for i in ${!app_arr[@]}; do
if [[ "${app_arr[$i]}" =~ "\$VOLNAME" ]]; then
app_arr[$i]=$(sed -e "s@\$VOLNAME@$abs_vol_path@" <<< ${app_arr[$i]})
fi
done
$(cat << EOF > "$BASHSCRIPT"
#!/bin/bash
if [ -e "$FILENAME" ]; then
${HDIUTIL} attach "$FILENAME"
if [ \$? -eq 0 ]; then
${app_arr[@]}
${HDIUTIL} detach "$abs_vol_path"
fi
fi)
# ensure our bash script is executable
chmod +x "$BASHSCRIPT"
F_MESSAGE="File $BASHSCRIPT successfully created!"
return 0
}
forge() {
F_MESSAGE="Tell me what to do!"
if [[ "${FILENAME}" && "$SIZE" ]]; then
p_echo "Creating the mactomb file..."
create
if [ "$?" -eq 1 ]; then
return 1
fi
fi
if [[ "$APPCMD" && "$BASHSCRIPT" && "$FILENAME" ]]; then
p_echo "Creating the output script..."
app
if [ "$?" -eq 1 ]; then
return 1
fi
fi
if [[ "$OUTSCRIPT" ]]; then
# we can't create the Automator app without bash script!
if [[ ! "$BASHSCRIPT" ]]; then
F_MESSAGE="No bash script set. Please use the -b flag"
return 1
fi
p_echo "Creating the Automator app to call the output script..."
# ensure we have the .app extension to let Mac recognise it as app
if [[ "${OUTSCRIPT##*.}" != "app" ]]; then
OUTSCRIPT+=".app"
fi
# ensure absolute path for the output app
if [[ $(dirname "$OUTSCRIPT") == "." ]]; then
OUTSCRIPT="$(PWD)"/"${OUTSCRIPT}"
fi
if [ -e "$OUTSCRIPT" ]; then
F_MESSAGE="$OUTSCRIPT already exists."
return 1
fi
if [ -d "$OUTSCRIPT" ]; then
F_MESSAGE="$OUTSCRIPT is a directory."
return 1
fi
# keep the template safe
cp -r "$AUTOMATOR" "${OUTSCRIPT}"
# let's the magic happen!
sed -i '' -e "s@SCRIPT_TO_RUN@$BASHSCRIPT@" "${OUTSCRIPT}/Contents/document.wflow"
F_MESSAGE="Mactomb forged! Now double click on $automatr to start your app inside the mactomb."
fi
return 0
}
COMMAND=('create', 'app', 'help', 'forge', 'resize', 'list', 'chpass', 'compress', 'decompress', 'rename', 'encrypt')
HDIUTIL=/usr/bin/hdiutil
# if 1, the script will use the Mac OS X notification method
NOTIFICATION=0
ENC="AES-256"
FS="HFS+"
# going default with sparsebundle! Use "DMG" for old style tomb
IMGFORMAT="SPARSEBUNDLE"
# compression?
COMPRESS=0
# compression format (should be the default)
CFORMAT="UDZO"
# zlib compression level (9 = highest)
CLEVEL=9
# path to the app used for the script
APPCMD=""
# output script for automatically call the app defined in APPCMD
BASHSCRIPT=""
# output Automator app that is in charge to call $BASHSCRIPT
OUTSCRIPT=""
# automator template
AUTOMATOR="template.app"
# default volume name for HFS+. Change this if you don't like it
VOLNAME="untitled"
VERSION=1.4
CMD=$1
shift
while getopts "a:f:s:p:o:b:n:t:hvc" opt; do
case "${opt}" in
f)
FILENAME=$OPTARG;;
s)
SIZE=$OPTARG;;
n)
VOLNAME=$OPTARG;;
a)
APPCMD=$OPTARG;;
c)
COMPRESS=1;;
p)
PROFILE=$OPTARG;;
b)
BASHSCRIPT=$OPTARG;;
o)
OUTSCRIPT=$OPTARG;;
t)
IMGFORMAT=$OPTARG;;
v)
NOTIFICATION=1;;
\?)
usage
exit 1
;;
esac
done
if [[ ! "${COMMAND[@]}" =~ "${CMD}" ]]; then
usage
exit 1
fi
# add a checks just to be sure
if [ ! -e "${HDIUTIL}" ]; then
e_echo "Weird, hdiutil not found in '${HDIUTIL}'. Can't make it."
exit 1
fi
$CMD
ret=$?
if [ "$NOTIFICATION" -eq 1 ]; then
if [ "$ret" -eq 0 ] || [ "$ret" -eq 1 ]; then
osascript -e 'display notification "'"${F_MESSAGE}"'" with title "MacTomb" subtitle "'${CMD}'"'
fi
else
if [ "$ret" -eq 1 ]; then
e_echo "${F_MESSAGE}"
# enforce check to 0 if we want some commands not to return a message (like 'help')
elif [ "$ret" -eq 0 ]; then
s_echo "${F_MESSAGE}"
fi
fi