forked from CodelyTV/dotly
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathpackage.sh
543 lines (473 loc) · 18.5 KB
/
package.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
#!/usr/bin/env bash
DOTFILES_PACKAGE_MANAGERS_PATH="${DOTFILES_PACKAGE_MANAGERS_PATH:-${DOTFILES_PATH:-${HOME}/.dotfiles}/package/managers}"
if [[ -n "${PACKAGE_MANAGERS_SRC[*]:-}" ]]; then
if
! array::exists_value "${SLOTH_PATH:-${DOTLY_PATH:-/dev/null}}/scripts/package/src/package_managers" "${PACKAGE_MANAGERS_SRC[@]}" ||
! array::exists_value "$DOTFILES_PACKAGE_MANAGERS_PATH" "${PACKAGE_MANAGERS_SRC[@]}"
then
export PACKAGE_MANAGERS_SRC=(
"${SLOTH_PATH:-${DOTLY_PATH:-/dev/null}}/scripts/package/src/package_managers"
"$DOTFILES_PACKAGE_MANAGERS_PATH"
"${PACKAGE_MANAGERS_SRC[@]}"
)
fi
else
export PACKAGE_MANAGERS_SRC=(
"${SLOTH_PATH:-${DOTLY_PATH:-/dev/null}}/scripts/package/src/package_managers"
"$DOTFILES_PACKAGE_MANAGERS_PATH"
)
fi
if [[ -z "${SLOTH_PACKAGE_MANAGERS_PRECEDENCE:-}" ]]; then
if platform::is_macos; then
export SLOTH_PACKAGE_MANAGERS_PRECEDENCE=(
mas brew cargo pipx pip volta npm
)
else
export SLOTH_PACKAGE_MANAGERS_PRECEDENCE=(
brew apt snap dnf pacman yum cargo pipx pip gem volta npm
)
fi
fi
#;
# package::manager_exists()
# Check if a given package manager exists in PACKAGE_MANAGERS_SRC
# @param string package_manager
# @return string|void Full path to package manager or nothing
#"
package::manager_exists() {
local package_manager_src
local -r package_manager="${1:-}"
for package_manager_src in "${PACKAGE_MANAGERS_SRC[@]}"; do
[[ -f "${package_manager_src}/${package_manager}.sh" ]] &&
printf "%s" "${package_manager_src}/${package_manager}.sh" &&
return
done
}
#;
# package::load_manager()
# Load a package manager library if exists, if not, exit the script with a critical warning. Recommended to use first package::manager_exists() if is not critical
# @param string package_manager
# @return void
#"
package::load_manager() {
local package_manager_file_path
local -r package_manager="${1:-}"
package_manager_file_path="$(package::manager_exists "$package_manager")"
if [[ -n "$package_manager_file_path" ]]; then
dot::load_library "$package_manager_file_path"
else
output::error "🚨 Package Manager \`$package_manager\` does not exists"
exit 4
fi
}
#;
# package::get_all_package_managers()
# Output a full list of all package managers. If any param provided will list only package managers with subcommand(s) (functions)
# @param any commands command or command that must have the package managers to list them as available
#"
package::get_all_package_managers() {
local package_manager_src package_manager package_command has_all
for package_manager_src in $(find "${PACKAGE_MANAGERS_SRC[@]}" -maxdepth 1 -mindepth 1 -name "*.sh" -print0 2> /dev/null | xargs -0); do
# Get package manager name
#shellcheck disable=SC2030
package_manager="$(basename "$package_manager_src")"
package_manager="${package_manager%%.sh}"
has_all=true
if [[ $# -gt 0 ]]; then
for package_command in "$@"; do
! script::function_exists "$package_manager_src" "${package_manager}::${package_command}" && has_all=false && break
done
fi
if ${has_all:-true}; then
printf "%s\n" "$package_manager"
fi
done
}
#;
# package::get_available_package_managers()
# Output a full list of available package managers (those with ::is_available and return success)
#"
package::get_available_package_managers() {
local package_manager_src package_manager_filename package_manager
for package_manager_src in $(package::get_all_package_managers "is_available"); do
package_manager_filename="$(basename "$package_manager_src")"
package_manager="${package_manager_filename%%.sh}"
if package::command "$package_manager" "is_available"; then
printf "%s\n" "$package_manager"
fi
done
}
#;
# package::manager_preferred()
# Get the first avaible of the preferred package managers
# @return string package manager
#"
package::manager_preferred() {
local all_available_pkgmgrs
readarray -t all_available_pkgmgrs < <(package::get_available_package_managers)
eval "$(array::uniq_unordered "${SLOTH_PACKAGE_MANAGERS_PRECEDENCE[@]}" "${all_available_pkgmgrs[@]}")"
if [[ ${#uniq_values[@]} -gt 0 ]]; then
printf "%s" "${uniq_values[0]}"
fi
}
#;
# package::command_exists()
# Execute if a command (function) is defined for a given package manager
# @param string package_manager
# @param string command The function to be check
# @param string package_mananager_src Optional param to provide the package manager library (useful to avoid check if exists again)
# @return boolean
#"
package::command_exists() {
local -r package_manager="${1:-}"
local -r command="${2:-}"
local -r package_command="${package_manager}::${command}"
local -r package_manager_src="${3:-$(package::manager_exists "$package_manager")}"
if
[[
"$package_manager" == "none" ||
-z "$package_manager" ||
-z "$command" ||
! -f "$package_manager_src" ]] ||
! script::function_exists "$package_manager_src" "$package_command"
then
return 1
fi
return 0
}
#;
# package::command()
# Execute if exists a function for package_manager (example: execute install command for a package manager if that function is defined for the given package manager). If execute install, the output is send also to the log.
# @param string package_manager
# @param string command The function to be executed
# @param any args Arguments for command
# @return void
#"
package::command() {
local -r package_manager="${1:-}"
local -r command="${2:-}"
local -r args=("${@:3}")
# If function does not exists for the package manager it will return 0 (true) always
if package::command_exists "$package_manager" "${command}"; then
package::load_manager "$package_manager"
if [[ "$command" == "install" ]]; then
"${package_manager}::${command}" "${args[@]:-}" 2>&1 | log::file "Installing ${args[*]} using $package_manager" || return
elif [[ "$command" == "uninstall" ]]; then
"${package_manager}::${command}" "${args[@]:-}" 2>&1 | log::file "Uninstalling ${args[*]} using $package_manager" || return
else
"${package_manager}::${command}" "${args[@]:-}"
fi
return $?
fi
return 1
}
#;
# package::managers_self_update()
# Update packages manager list of packages (no packages). Should not be a upgrade of all apps
# @param string package_manager If this value is empty update all available package managers
# @return void
#"
package::manager_self_update() {
local package_manager="${1:-}"
if [[ -n "$package_manager" ]]; then
package::command_exists "$package_manager" self_update && package::command "$package_manager" self_update
else
for package_manager_src in $(package::get_available_package_managers); do
[[ -n "$package_manager" ]] && package::manager_self_update "$package_manager"
done
fi
}
#;
# package::which_package_manager()
# Output which package manager was used to install a package
# @param string package
# @param boolean avoid_registry_check If true avoid check if package is installed with the registry
# @return boolean If package is not installed
#"
package::which_package_manager() {
local package_manager
local -r package_name="${1:-}"
local -r avoid_registry_check=${2:-false}
[[ -z "$package_name" ]] && return 1
# Check every package manager first because maybe registry has used a package manager
for package_manager in $(package::get_all_package_managers "is_available" "is_installed"); do
package::command "$package_manager" "is_available" &&
package::command "$package_manager" is_installed "$package_name" &&
printf "%s" "$package_manager" &&
return
done
# Because registry::is_installed is defined in core. This is a expected behavior and we do it at
# the end because probably a package was installed with a package manager spite of being a reicpe
if
! $avoid_registry_check &&
[[ -n "$(registry::recipe_exists "$package_name")" ]] &&
registry::command_exists "$package_name" "is_installed"
then
registry::is_installed "$package_name" && printf "%s" "registry" && return
return 1
fi
return 1
}
#;
# package::is_installed()
# Check if a package is installed with a recipe or any of the available package managers. It does not check if a binary package_name is available
# @param string package_name
# @param string package_manager Can be "any" to use any package manager or registry (same as empty). "auto" to use any one except registry. "recipe" or "registry" are aliases. Can be any other valid package manager in 'scripts/package/src/package_managers'.
# @return boolean
#"
package::is_installed() {
local package_manager
local -r package_name="${1:-}"
package_manager="${2:-}"
[[ -z "$package_name" ]] && return 1
# Allow to use recipe(s) instead of registry
[[ -n "$package_manager" && $package_manager == "recipe"[s] ]] && package_manager="registry"
# Any package manager is the same as empty package_manager
[[ $package_manager == "any" ]] && package_manager=""
if
[[ -z "$package_manager" || $package_manager == "registry" ]] &&
[[ -n "$(registry::recipe_exists "$package_name")" ]] &&
registry::command_exists "$package_name" "is_installed"
then
registry::is_installed "$package_name" && return
return 1
# auto package manager means any but not registry
elif [[ $package_manager == "auto" || -z "$package_manager" ]]; then
package::which_package_manager "$package_name" true > /dev/null 2>&1 && return 0
elif [[ -n "$package_manager" ]]; then
package::command_exists "$package_manager" "is_installed" && package::command "$package_manager" "is_installed" "$package_name" && return
fi
return 1
}
#;
# package::_install()
# "Private" function for package::install that do the repetive task of installing a package
# @param string package_manager
# @param string package
# @return boolean
#"
package::_install() {
local package_manager package
package_manager="${1:-}"
package="${2:-}"
[[ -z "$package_manager" || -z "$package" ]] && return 1
shift 2
if
package::command_exists "$package_manager" "is_available" &&
package::command_exists "$package_manager" "package_exists" &&
package::command_exists "$package_manager" "is_installed" &&
package::command_exists "$package_manager" "install" &&
package::command "$package_manager" "is_available"
then
if
package::command "$package_manager" "package_exists" "$package" &&
package::command "$package_manager" "install" "$package" "$@"
then
package::command "$package_manager" "is_installed" "$package" && return
fi
elif
package::command_exists "$package_manager" "is_available" &&
package::command_exists "$package_manager" "install" &&
package::command_exists "$package_manager" "is_installed" &&
package::command "$package_manager" "is_available"
then
package::command "$package_manager" "install" "$package" "$@" &&
package::command "$package_manager" "is_installed" &&
return
fi
# Not exists or not installed
return 1
}
#;
# package::install()
# Try to install with any available package manager, but if you provided a package manager (second param) it will only try to use that package manager. This avoids to install from registry recipe (use package::install_recipe_first).
# @param string package Package to install
# @param string package_manager Can be "any" to use any package manager or registry. "auto" to use any one except registry. "recipe" or "registry" are aliases. Can be any other valid package manager in 'scripts/package/src/package_managers'.
# @param any args Arguments for package manager wrapper
# @return boolen
#"
package::install() {
local _args all_available_pkgmgrs uniq_values package_manager package
if [[ $* == *"--force"* ]]; then
mapfile -t _args < <(array::substract "--force" "$@")
package::force_install "${_args[@]}" &&
return
log::error "Unable to force install \`$package\` with \`$package_manager\`"
return 1
else
[[ -z "${1:-}" ]] && return 1
package="$1"
shift
package_manager="${1:-}"
fi
if [[ -n "$package_manager" ]]; then
shift
# Allow to use recipe(s) instead of registry
[[ $package_manager == "recipe"[s] ]] && package_manager="registry"
[[ $package_manager == "any" ]] && package_manager=""
fi
if
[[
-n "$package_manager" &&
$package_manager != "auto" &&
$package_manager != "registry" ]]
then
if [[ -n "$(package::manager_exists "$package_manager")" ]]; then
package::_install "$package_manager" "$package" "$@" &&
return 0
else
output::error "Package manager not found"
return 1
fi
elif
[[
-z "$package_manager" ||
$package_manager == "registry" ]] &&
[[ -n "$(registry::recipe_exists "$package")" ]]
then
registry::install "$package" "$@" && registry::is_installed "$package" "$@" && return 0
else
if platform::command_exists readarray; then
readarray -t all_available_pkgmgrs < <(package::get_available_package_managers)
else
#shellcheck disable=SC2207
all_available_pkgmgrs=($(package::get_available_package_managers))
fi
eval "$(array::uniq_unordered "${SLOTH_PACKAGE_MANAGERS_PRECEDENCE[@]}" "${all_available_pkgmgrs[@]}")"
# Try to install respecting package managers precedence
for package_manager in "${uniq_values[@]}"; do
if
[[ -n "$(package::manager_exists "$package_manager")" ]] &&
package::_install "$package_manager" "$package" "$@"
then
return 0
fi
done
return 1
fi
return 1
}
#;
# package::force_install()
# Install using forced method if implemented for the given package, if second parameter is used as the package manager to be used or it will fail
# @param string package_name
# @param string package_manager Can be "any" to use any package manager or registry. "auto" to use any one except registry. "recipe" or "registry" are aliases. Can be any other valid package manager in 'scripts/package/src/package_managers'.
# @param any args Additional arguments to be passed to install function (package_manager is required then)
# @return boolean True if installed and false if still installed
#"
package::force_install() {
local package_manager package_name recipe_path
[[ $# -lt 1 ]] && log::error "There is no package name" && return 1
mapfile -t _args < <(array::substract "--force" "$@")
package_name="${_args[0]:-}"
package_manager="${_args[1]:-}"
[[ -z "$package_name" ]] && return 1
recipe_path="$(registry::recipe_exists "$package_name")"
if [[ -n "$recipe_path" || $package_manager == "registry" || $package_manager == "recipe" ]]; then
package_manager=registry
registry::force_install "$package_name" "${_args[@]:2}" &&
return
elif
[[ -n "$package_manager" ]] &&
package::command_exists "$package_manager" "force_install"
then
package::command "$package_manager" "force_install" "$package_name" "${_args[@]:2}" && return
elif
[[ -n "$package_manager" ]] &&
package::command_exists "$package_manager" "uninstall" &&
package::uninstall "$package_name" "$package_manager" "${_args[@]:2}" &&
package::command_exists "$package_manager" "install"
then
package::command "$package_manager" "install" "$package_name" "${_args[@]:2}" && return
fi
log::error "Unable to force install \`$package_name\` with \`$package_manager\`"
return 1
}
#;
# package::uninstall()
# Uninstall the given package, if second parameter is used as the package manager to be used or it will fail
# @param string package_name
# @param string package_manager Can be "any" to use any package manager or registry. "auto" to use any one except registry. "recipe" or "registry" are aliases. Can be any other valid package manager in 'scripts/package/src/package_managers'.
# @param any args Additional arguments to be passed to uninstall function (package_manager is required then)
# @return boolean True if uninstalled and false if still installed
#"
package::uninstall() {
local package_manager
[[ $# -lt 1 ]] && return 1
local -r package_name="$1"
shift
package_manager="${1:-}"
if [[ -n "$package_manager" ]]; then
shift
# Allow to use recipe(s) instead of registry
[[ $package_manager == "recipe"[s] ]] && package_manager="registry"
[[ $package_manager == "any" ]] && package_manager=""
fi
if [[ -z "$package_manager" || $package_manager == "registry" ]]; then
local -r recipe_path="$(registry::recipe_exists "$package_name")"
if
[[ -n "$recipe_path" ]] &&
registry::command_exists "$package_name" "uninstall"
then
registry::uninstall "$package_name" "$@" && ! registry::is_installed "$package_name" && return 0
fi
else
[[ $package_manager == "auto" || -z "$package_manager" ]] && package_manager="$(package::which_package_manager "$package_name" true || true)"
echo "- $package_manager -"
if
[[
-z "$package_manager" ||
-z "$(package::manager_exists "$package_manager")" ]]
then
printf "%s" "Package manager $package_manager"
# Could not determine which package manager to be used or package manager not exists
return 1
fi
if package::command_exists "$package_manager" "uninstall"; then
package::command "$package_manager" "uninstall" "$package_name" "$@" && ! package::is_installed "$package_name" && return 0
fi
fi
# Recipe or package not uninstalled or does not have uninstall wrapper (function) (can happen with both package managers and registry)
return 1
}
#;
# package::remove()
# Alias of package::uninstall
#"
package::remove() {
package::uninstall "$@"
}
#;
# package::command_dump_check()
# Used to check if package manager exists and create the subdir where the dump file will be placed
# @param string package_manager
# @param string file_path The directory where the dump file will be created
# @return void
#"
package::common_dump_check() {
local -r package_manager="${1:-}"
local -r file_path="${2:-}"
if
[[ -n "$package_manager" ]] &&
[[ -n "$file_path" ]] &&
[[ -n "$(package::manager_exists "$package_manager")" ]]
then
mkdir -p "$(dirname "$file_path")"
fi
}
#;
# package::common_import_check()
# Check if the file exists for the given package manager
# @param string package_manager
# @param string file_path File to check if exists
# @return boolean
#"
package::common_import_check() {
local package_manager file_path
local -r package_manager="${1:-}"
local -r file_path="${2:-}"
[[ -n "$package_manager" ]] &&
[[ -n "$file_path" ]] &&
[[ -n "$(package::manager_exists "$package_manager")" ]] &&
[[ -f "$file_path" ]]
}