flatcar-scripts/pkg_auto/impl/md5_cache_lib.sh

1178 lines
35 KiB
Bash

#!/bin/bash
# This file implements parsing the md5-metadata cache files and
# accessing the parsed results. Not the entirety of the cache file is
# parsed, only the parts that were needed at the time of writing
# it. So currently the exposed parts of parsed cache files are EAPI,
# IUSE, KEYWORDS, LICENSE, {B,R,P,I,}DEPEND and _eclasses_. The
# _eclasses_ part discards the checksums, though, so only names are
# available.
if [[ -z ${__MD5_CACHE_LIB_SH_INCLUDED__:-} ]]; then
__MD5_CACHE_LIB_SH_INCLUDED__=x
source "$(dirname "${BASH_SOURCE[0]}")/util.sh"
source "${PKG_AUTO_IMPL_DIR}/debug.sh"
source "${PKG_AUTO_IMPL_DIR}/gentoo_ver.sh"
#
# Cache file
#
# Indices to access fields of the cache file:
# PCF_EAPI_IDX - a string/number describing EAPI
# PCF_KEYWORDS_IDX - a name of an array containing keyword objects
# PCF_IUSE_IDX - a name of an array containing iuse objects
# PCF_BDEPEND_IDX - a name of a group object with build dependencies
# PCF_DEPEND_IDX - a name of a group object with dependencies
# PCF_IDEPEND_IDX - a name of a group object with install dependencies
# PCF_PDEPEND_IDX - a name of a group object with post dependencies
# PCF_RDEPEND_IDX - a name of a group object with runtime dependencies
# PCF_LICENSE_IDX - a name of a group object with licenses
# PCF_ECLASSES_IDX - a name of an array with used eclasses
declare -gri PCF_EAPI_IDX=0 PCF_KEYWORDS_IDX=1 PCF_IUSE_IDX=2 PCF_BDEPEND_IDX=3 PCF_DEPEND_IDX=4 PCF_IDEPEND_IDX=5 PCF_PDEPEND_IDX=6 PCF_RDEPEND_IDX=7 PCF_LICENSE_IDX=8 PCF_ECLASSES_IDX=9
# Declares empty cache files. Can take flags that are passed to
# declare. Usually only -r or -t make sense to pass. Can take several
# names, just like declare. Takes no initializers - this is hardcoded.
function cache_file_declare() {
struct_declare -ga "${@}" "( '0' 'EMPTY_ARRAY' 'EMPTY_ARRAY' 'EMPTY_GROUP' 'EMPTY_GROUP' 'EMPTY_GROUP' 'EMPTY_GROUP' 'EMPTY_GROUP' 'EMPTY_GROUP' 'EMPTY_ARRAY' )"
}
# Unsets cache files, can take several names, just like unset.
function cache_file_unset() {
local name
for name; do
local -n cache_file_ref=${name}
__mcl_unset_array "${cache_file_ref[PCF_KEYWORDS_IDX]}" kw_unset
__mcl_unset_array "${cache_file_ref[PCF_IUSE_IDX]}" iuse_unset
__mcl_unset_array "${cache_file_ref[PCF_ECLASSES_IDX]}" unset
group_unset \
"${cache_file_ref[PCF_BDEPEND_IDX]}" \
"${cache_file_ref[PCF_DEPEND_IDX]}" \
"${cache_file_ref[PCF_IDEPEND_IDX]}" \
"${cache_file_ref[PCF_PDEPEND_IDX]}" \
"${cache_file_ref[PCF_RDEPEND_IDX]}" \
"${cache_file_ref[PCF_LICENSE_IDX]}"
unset -n cache_file_ref
done
unset "${@}"
}
# Parses a cache file under the passed path.
#
# Params:
#
# 1 - cache file
# 2 - path to the cache file
function parse_cache_file() {
local -n cache_file_ref=${1}; shift
local path=${1}; shift
# rest are architectures
if pkg_debug_enabled; then
local -a file_lines
mapfile -t file_lines <"${path}"
pkg_debug_print_lines "parsing ${path@Q}" "${file_lines[@]}"
unset file_lines
fi
local -n pkg_eapi_ref='cache_file_ref[PCF_EAPI_IDX]'
local -n pkg_keywords_ref='cache_file_ref[PCF_KEYWORDS_IDX]'
local -n pkg_iuse_ref='cache_file_ref[PCF_IUSE_IDX]'
local -n pkg_bdepend_group_name_ref='cache_file_ref[PCF_BDEPEND_IDX]'
local -n pkg_depend_group_name_ref='cache_file_ref[PCF_DEPEND_IDX]'
local -n pkg_idepend_group_name_ref='cache_file_ref[PCF_IDEPEND_IDX]'
local -n pkg_pdepend_group_name_ref='cache_file_ref[PCF_PDEPEND_IDX]'
local -n pkg_rdepend_group_name_ref='cache_file_ref[PCF_RDEPEND_IDX]'
local -n pkg_license_group_name_ref='cache_file_ref[PCF_LICENSE_IDX]'
local -n pkg_eclasses_ref='cache_file_ref[PCF_ECLASSES_IDX]'
local l key
while read -r l; do
key=${l%%=*}
pkg_debug "parsing ${key@Q}"
case ${key} in
'EAPI')
pkg_eapi_ref=${l#*=}
pkg_debug "EAPI: ${pkg_eapi_ref}"
;;
'KEYWORDS')
__mcl_parse_keywords "${l#*=}" pkg_keywords_ref "${@}"
;;
'IUSE')
__mcl_parse_iuse "${l#*=}" pkg_iuse_ref
;;
'BDEPEND')
__mcl_parse_dsf "${__MCL_DSF_DEPEND}" "${l#*=}" pkg_bdepend_group_name_ref
;;
'DEPEND')
__mcl_parse_dsf "${__MCL_DSF_DEPEND}" "${l#*=}" pkg_depend_group_name_ref
;;
'IDEPEND')
__mcl_parse_dsf "${__MCL_DSF_DEPEND}" "${l#*=}" pkg_idepend_group_name_ref
;;
'PDEPEND')
__mcl_parse_dsf "${__MCL_DSF_DEPEND}" "${l#*=}" pkg_pdepend_group_name_ref
;;
'RDEPEND')
__mcl_parse_dsf "${__MCL_DSF_DEPEND}" "${l#*=}" pkg_rdepend_group_name_ref
;;
'LICENSE')
__mcl_parse_dsf "${__MCL_DSF_LICENSE}" "${l#*=}" pkg_license_group_name_ref
;;
'_eclasses_')
__mcl_parse_eclasses "${l#*=}" pkg_eclasses_ref
;;
*)
pkg_debug "Not parsing ${key@Q}, ignoring"
;;
esac
done <"${path}"
}
#
# Use requirement (the part in the square brackets in the package
# dependency specification).
#
# Indices to access fields of the use requirement:
# UR_NAME_IDX - a use name
# UR_MODE_IDX - a string describing the mode: "+" is enabled, "="
# is strict relation, "!=" reversed strict relation,
# "?" is "enabled if enabled in the ebuild", "!?" is
# "disabled if disabled in the ebuild", and "-" is
# disabled)
# UR_PRETEND_IDX - a string describing pretend mode: it can be either
# empty (no pretending if use is missing in the
# package), "+" (pretend enabled if missing in the
# package), or "-" (pretend disabled if missing in
# the package)
declare -gri UR_NAME_IDX=0 UR_MODE_IDX=1 UR_PRETEND_IDX=2
# Declares empty use requirements. Can take flags that are passed to
# declare. Usually only -r or -t make sense to pass. Can take several
# names, just like declare. Takes no initializers - this is hardcoded.
function ur_declare() {
struct_declare -ga "${@}" "( 'ITS_UNSET' '+' '' )"
}
# Unsets use requirements, can take several names, just like unset.
function ur_unset() {
unset "${@}"
}
# Copies use requirement into another.
#
# Params:
#
# 1 - use requirement to be clobbered
# 2 - the source use requirement
function ur_copy() {
local -n to_clobber_ref=${1}; shift
local -n to_copy_ref=${1}; shift
local -i idx
for idx in UR_NAME_IDX UR_MODE_IDX UR_PRETEND_IDX; do
to_clobber_ref[idx]=${to_copy_ref[idx]}
done
}
# Stringifies use requirement.
#
# Params:
#
# 1 - use requirement
# 2 - name of a variable where the string form will be stored
function ur_to_string() {
local -n ur_ref=${1}; shift
local -n str_ref=${1}; shift
case ${ur_ref[UR_MODE_IDX]} in
'!'*)
str_ref='!'
;;
'-'*)
str_ref='-'
;;
*)
str_ref=''
;;
esac
str_ref+=${ur_ref[UR_NAME_IDX]}
local p=${ur_ref[UR_PRETEND_IDX]}
if [[ -n ${p} ]]; then
str_ref+="(${p})"
fi
case ${ur_ref[UR_MODE_IDX]} in
*'=')
str_ref+='='
;;
*'?')
str_ref+='?'
;;
esac
}
#
# Package depedency specification (or PDS)
#
# Enumeration describing blocker mode of a package. Self-describing.
declare -gri PDS_NO_BLOCK=0 PDS_WEAK_BLOCK=1 PDS_STRONG_BLOCK=2
# Indices to access fields of the package dependency specification:
# PDS_BLOCKS_IDX - a number describing blocker mode (use PDS_NO_BLOCK,
# PDS_WEAK_BLOCK and PDS_STRONG_BLOCK)
# PDS_OP_IDX - a string describing the relational operator, can be
# empty or one of "<", "<=", "=", "=*", "~", ">=", or
# ">" ("=*" is same as "=" with asterisk appended to
# version)
# if empty, the version will also be empty
# PDS_NAME_IDX - a qualified package name (so category/name)
# PDS_VER_IDX - a version of the package, may be empty (if so,
# operator is also empty)
# PDS_SLOT_IDX - a string describing the slot operator, without the
# preceding colon
# PDS_UR_IDX - a name of an array variable containing use
# requirements
declare -gri PDS_BLOCKS_IDX=0 PDS_OP_IDX=1 PDS_NAME_IDX=2 PDS_VER_IDX=3 PDS_SLOT_IDX=4 PDS_UR_IDX=5
# Declares empty package definition specifications. Can take flags
# that are passed to declare. Usually only -r or -t make sense to
# pass. Can take several names, just like declare. Takes no
# initializers - this is hardcoded.
function pds_declare() {
struct_declare -ga "${@}" "( ${PDS_NO_BLOCK@Q} '' 'ITS_UNSET' '' '' 'EMPTY_ARRAY' )"
}
# Unsets package definition specifications, can take several names,
# just like unset.
function pds_unset() {
local use_reqs_name name
for name; do
local -n pds_ref=${name}
use_reqs_name=${pds_ref[PDS_UR_IDX]}
local -n use_reqs_ref=${use_reqs_name}
ur_unset "${use_reqs_ref[@]}"
unset -n use_reqs_ref
if [[ ${use_reqs_name} != 'EMPTY_ARRAY' ]]; then
unset "${use_reqs_name}"
fi
unset -n pds_ref
done
unset "${@}"
}
# Copies package dependency specification into another.
#
# Params:
#
# 1 - package dependency specification to be clobbered
# 2 - the source package dependency specification
function pds_copy() {
local -n to_clobber_ref=${1}; shift
local -n to_copy_ref=${1}; shift
local -i idx
for idx in PDS_BLOCKS_IDX PDS_OP_IDX PDS_NAME_IDX PDS_VER_IDX PDS_SLOT_IDX; do
to_clobber_ref[idx]=${to_copy_ref[idx]}
done
if [[ ${to_copy_ref[PDS_UR_IDX]} = 'EMPTY_ARRAY' || ${#to_copy_ref[PDS_UR_IDX]} -eq 0 ]]; then
to_clobber_ref[PDS_UR_IDX]='EMPTY_ARRAY'
else
local pc_ur_array_name
gen_varname pc_ur_array_name
declare -ga "${pc_ur_array_name}=()"
local -n urs_to_copy_ref=${to_copy_ref[PDS_UR_IDX]} urs_ref=${pc_ur_array_name}
local ur_name pc_ur_name
for ur_name in "${urs_to_copy_ref[@]}"; do
gen_varname pc_ur_name
ur_declare "${pc_ur_name}"
ur_copy "${pc_ur_name}" "${ur_name}"
urs_ref+=( "${pc_ur_name}" )
done
to_clobber_ref[PDS_UR_IDX]=${pc_ur_array_name}
fi
}
# Adds use requirements to the package dependency specification.
#
# Params:
#
# 1 - package dependency specification
# @ - use requirements
function pds_add_urs() {
local -n pds_ref=${1}; shift
# rest are use requirements
if [[ ${#} -eq 0 ]]; then
return 0
fi
local use_reqs_name=${pds_ref[PDS_UR_IDX]}
if [[ ${use_reqs_name} = 'EMPTY_ARRAY' ]]; then
local ura_name
gen_varname ura_name
declare -ga "${ura_name}=()"
pds_ref[PDS_UR_IDX]=${ura_name}
use_reqs_name=${ura_name}
unset ura_name
fi
local -n use_reqs_ref=${use_reqs_name}
use_reqs_ref+=( "${@}" )
}
# Stringifies package dependency specification.
#
# Params:
#
# 1 - package dependency specification
# 2 - name of a variable where the string form will be stored
function pds_to_string() {
local -n pds_ref=${1}; shift
local -n str_ref=${1}; shift
case ${pds_ref[PDS_BLOCKS_IDX]} in
"${PDS_NO_BLOCK}")
str_ref=''
;;
"${PDS_WEAK_BLOCK}")
str_ref='!'
;;
"${PDS_STRONG_BLOCK}")
str_ref='!!'
;;
esac
local op=${pds_ref[PDS_OP_IDX]}
local v=${pds_ref[PDS_VER_IDX]}
if [[ ${op} = '=*' ]]; then
op='='
# if there's an op, then we assume version is not empty - so
# version will never end up being just *
v+='*'
fi
str_ref+=${op}${pds_ref[PDS_NAME_IDX]}
if [[ -n ${v} ]]; then
str_ref+=-${v}
fi
local s=${pds_ref[PDS_SLOT_IDX]}
if [[ -n ${s} ]]; then
str_ref+=:${s}
fi
local -n urs_ref=${pds_ref[PDS_UR_IDX]}
if [[ ${#urs_ref[@]} -gt 0 ]]; then
str_ref+='['
local u ur_str
for u in "${urs_ref[@]}"; do
ur_to_string "${u}" ur_str
str_ref+=${ur_str},
done
unset ur_str u
str_ref=${str_ref:0:$(( ${#str_ref} - 1 ))}']'
fi
}
#
# Group. A structure for describing {B,R,I,P,}DEPEND and LICENSE
# fields. Contains items, which can be either package dependency
# specifications or another groups. So it is a recursive structure.
#
# Enumeration describing type of a group. Self-describing.
declare -gri GROUP_ALL_OF=0 GROUP_ANY_OF=1
# Enumeration describing whether items in group are for enabled or disabled USE. Self-describing.
declare -gri GROUP_USE_ENABLED=0 GROUP_USE_DISABLED=1
# Indices to access fields of the group:
# GROUP_TYPE_IDX - a number describing a type of the group (use
# GROUP_ALL_OF and GROUP_ANY_OF)
# GROUP_USE_IDX - a USE name of the group, may be empty, and must
# be empty for GROUP_ANY_OF groups
# GROUP_ENABLED_IDX - a number describing mode of the USE, should be
# ignored if USE name is empty (use
# GROUP_USE_ENABLED and GROUP_USE_DISABLED)
# GROUP_ITEMS_IDX - a name of an array containing items
declare -gri GROUP_TYPE_IDX=0 GROUP_USE_IDX=1 GROUP_ENABLED_IDX=2 GROUP_ITEMS_IDX=3
# Declares empty groups. Can take flags that are passed to
# declare. Usually only -r or -t make sense to pass. Can take several
# names, just like declare. Takes no initializers - this is hardcoded.
function group_declare() {
struct_declare -ga "${@}" "( ${GROUP_ALL_OF@Q} '' ${GROUP_USE_ENABLED@Q} 'EMPTY_ARRAY' )"
}
# An empty readonly group.
group_declare -r EMPTY_GROUP
# Unsets groups, can take several names, just like unset.
function group_unset() {
local -a to_unset=()
local name items_name
for name; do
if [[ ${name} == 'EMPTY_GROUP' ]]; then
continue
fi
local -n group_ref=${name}
items_name=${group_ref[GROUP_ITEMS_IDX]}
unset -n group_ref
to_unset+=( "${name}" )
if [[ ${items_name} == 'EMPTY_ARRAY' ]]; then
continue
fi
to_unset+=( "${items_name}" )
local -n items_ref=${items_name}
item_unset "${items_ref[@]}"
unset -n items_ref
done
unset "${to_unset[@]}"
}
# Copies group into another.
#
# Params:
#
# 1 - group to be clobbered
# 2 - the source group
function group_copy() {
local -n to_clobber_ref=${1}; shift
local -n to_copy_ref=${1}; shift
local -i idx
for idx in GROUP_TYPE_IDX GROUP_USE_IDX GROUP_ENABLED_IDX; do
to_clobber_ref[idx]=${to_copy_ref[idx]}
done
if [[ ${to_copy_ref[GROUP_ITEMS_IDX]} = 'EMPTY_ARRAY' || ${#to_copy_ref[GROUP_ITEMS_IDX]} -eq 0 ]]; then
to_clobber_ref[GROUP_ITEMS_IDX]='EMPTY_ARRAY'
else
local gc_items_name
gen_varname gc_items_name
declare -ga "${gc_items_name}=()"
local -n items_to_copy_ref=${to_copy_ref[GROUP_ITEMS_IDX]}
local -n items_ref=${gc_items_name}
local item_name_to_copy gc_item_name
for item_name_to_copy in "${items_to_copy_ref[@]}"; do
gen_varname gc_item_name
item_declare "${gc_item_name}"
item_copy "${gc_item_name}" "${item_name_to_copy}"
items_ref+=( "${gc_item_name}" )
done
unset -n items_ref items_to_copy_ref
to_clobber_ref[GROUP_ITEMS_IDX]="${gc_items_name}"
fi
}
# Adds items to the group.
#
# Params:
#
# 1 - group
# @ - items
function group_add_items() {
local -n group_ref=${1}; shift
# rest are items to add
if [[ ${#} -eq 0 ]]; then
return 0
fi
local items_name=${group_ref[GROUP_ITEMS_IDX]}
if [[ ${items_name} = 'EMPTY_ARRAY' ]]; then
local ia_name
gen_varname ia_name
declare -ga "${ia_name}=()"
group_ref[GROUP_ITEMS_IDX]=${ia_name}
items_name=${ia_name}
unset ia_name
fi
local -n items_ref=${items_name}
items_ref+=( "${@}" )
}
# Stringifies group.
#
# Params:
#
# 1 - group
# 2 - name of a variable where the string form will be stored
function group_to_string() {
local -n group_ref=${1}; shift
local -n str_ref=${1}; shift
local t=${group_ref[GROUP_TYPE_IDX]}
case ${t} in
"${GROUP_ALL_OF}")
local u=${group_ref[GROUP_USE_IDX]}
if [[ -n ${u} ]]; then
local e=${group_ref[GROUP_ENABLED_IDX]}
case ${e} in
"${GROUP_USE_ENABLED}")
str_ref=''
;;
"${GROUP_USE_DISABLED}")
str_ref='!'
esac
unset e
str_ref+="${u}? "
else
str_ref=''
fi
unset u
;;
"${GROUP_ANY_OF}")
str_ref='|| '
;;
esac
str_ref+='( '
local -n item_names_ref=${group_ref[GROUP_ITEMS_IDX]}
if [[ ${#item_names_ref[@]} -gt 0 ]]; then
local item_name item_str
for item_name in "${item_names_ref[@]}"; do
item_to_string "${item_name}" item_str
str_ref+="${item_str} "
done
unset item_str item_name
fi
str_ref+=')'
}
#
# Item. A string of <mode>:<data>. Mode is a single char and describes
# the data.
#
# Modes:
# "e" - empty, data is meaningless and should be just empty.
# "g" - group, data is a group (a name of a variable containing a
# group)
# "l" - license, data is a name of a license (plain string)
# "p" - package dependency specification, data is pds (a name of a
# variable containing a pds)
# Declares empty items. Can take flags that are passed to
# declare. Usually only -r or -t make sense to pass. Can take several
# names, just like declare. Takes no initializers - this is hardcoded.
function item_declare() {
struct_declare -g "${@}" 'e:'
}
# Unsets items, can take several names, just like unset.
function item_unset() {
local name
for name; do
local -n item_ref=${name}
case ${item_ref} in
e:*)
# noop
:
;;
g:*)
group_unset "${item_ref:2}"
;;
l:*)
# noop, license is just a string
;;
p:*)
pds_unset "${item_ref:2}"
;;
esac
unset -n item_ref
done
unset "${@}"
}
# Copies item into another.
#
# Params:
#
# 1 - item to be clobbered
# 2 - the source item
function item_copy() {
local -n to_clobber_ref=${1}; shift
local -n to_copy_ref=${1}; shift
local ic_name
local t=${to_copy_ref:0:1} v=${to_copy_ref:2}
case ${t} in
'e'|'l')
to_clobber_ref=${to_copy_ref}
;;
'g')
gen_varname ic_name
group_declare "${ic_name}"
group_copy "${ic_name}" "${v}"
to_clobber_ref="g:${ic_name}"
;;
'p')
gen_varname ic_name
pds_declare "${ic_name}"
pds_copy "${ic_name}" "${v}"
to_clobber_ref="p:${ic_name}"
;;
esac
}
# Stringifies item.
#
# Params:
#
# 1 - item
# 2 - name of a variable where the string form will be stored
function item_to_string() {
local -n item_ref=${1}; shift
local -n str_ref=${1}; shift
local t=${item_ref:0:1}
case ${t} in
e)
str_ref=''
;;
g)
local group_name=${item_ref:2}
local group_str
group_to_string "${group_name}" group_str
str_ref=${group_str}
unset group_str group_name
;;
l)
str_ref=${item_ref:2}
;;
p)
local pds_name=${item_ref:2}
local pds_str=''
pds_to_string "${pds_name}" pds_str
str_ref=${pds_str}
unset pds_str pds_name
;;
esac
}
#
# Keyword
#
# n - name of keyword
# m - stable (amd64), unstable (~amd64), broken (-amd64), unknown (absent in KEYWORDS)
# Enumeration describing mode of the keyword.
# KW_STABLE - like "amd64"
# KW_UNSTABLE - like "~amd64"
# KW_BROKEN - like "-amd64"
# KW_UNKNOWN - missing from KEYWORDS entirely
declare -gri KW_STABLE=0 KW_UNSTABLE=1 KW_BROKEN=2 KW_UNKNOWN=3
# Indices to access fields of the keyword:
# KW_NAME_IDX - name of the architecture
# KW_LEVEL_IDX - mode of the keyword (use KW_STABLE, KW_UNSTABLE,
# KW_BROKEN and KW_UNKNOWN)
declare -gri KW_NAME_IDX=0 KW_LEVEL_IDX=1
# Declares empty keywords. Can take flags that are passed to
# declare. Usually only -r or -t make sense to pass. Can take several
# names, just like declare. Takes no initializers - this is hardcoded.
function kw_declare() {
struct_declare -ga "${@}" "( 'ITS_UNSET' ${KW_UNSTABLE@Q} )"
}
# Unsets keywords, can take several names, just like unset.
function kw_unset() {
unset "${@}"
}
# Stringifies keyword.
#
# Params:
#
# 1 - keyword
# 2 - name of a variable where the string form will be stored
function kw_to_string() {
local -n kw_ref=${1}; shift
local -n str_ref=${1}; shift
local n=${kw_ref[KW_NAME_IDX]}
case ${kw_ref[KW_LEVEL_IDX]} in
"${KW_STABLE}")
str_ref=${n}
;;
"${KW_UNSTABLE}")
str_ref="~${n}"
;;
"${KW_BROKEN}")
str_ref="-${n}"
;;
"${KW_UNKNOWN}")
str_ref=''
;;
esac
}
#
# IUSE
#
# n - name of IUSE flag
# m - 0 or 1 (0 disabled, 1 enabled)
# Enumeration describing mode of the IUSE. Self-describing.
declare -gri IUSE_DISABLED=0 IUSE_ENABLED=1
# Indices to access fields of the IUSE:
# IUSE_NAME_IDX - IUSE name
# IUSE_MODE_IDX - IUSE mode (use IUSE_DISABLED and IUSE_ENABLED)
declare -gri IUSE_NAME_IDX=0 IUSE_MODE_IDX=1
# Declares empty IUSEs. Can take flags that are passed to
# declare. Usually only -r or -t make sense to pass. Can take several
# names, just like declare. Takes no initializers - this is hardcoded.
function iuse_declare() {
struct_declare -ga "${@}" "( 'ITS_UNSET' ${IUSE_DISABLED@Q} )"
}
# Unsets IUSEs, can take several names, just like unset.
function iuse_unset() {
unset "${@}"
}
# Stringifies IUSE.
#
# Params:
#
# 1 - IUSE
# 2 - name of a variable where the string form will be stored
function iuse_to_string() {
local -n iuse_ref=${1}; shift
local -n str_ref=${1}; shift
case ${iuse_ref[IUSE_MODE_IDX]} in
"${IUSE_ENABLED}")
str_ref='+'
;;
"${IUSE_DISABLED}")
str_ref=''
;;
esac
str_ref+=${iuse_ref[IUSE_NAME_IDX]}
}
#
# Implementation details
#
# parse dependency specification format (DSF)
declare -gri __MCL_DSF_DEPEND=0 __MCL_DSF_LICENSE=1
function __mcl_parse_dsf() {
local -i dsf_type=${1}; shift
local dep=${1}; shift
local -n top_group_out_var_name_ref=${1}; shift
local -a group_stack
local pd_group pd_item pd_group_created='' pd_pds
gen_varname pd_group
group_declare "${pd_group}"
group_stack+=( "${pd_group}" )
local -a tokens
mapfile -t tokens <<<"${dep// /$'\n'}"
local -i last_index
local token
for token in "${tokens[@]}"; do
if [[ ${token} = '||' ]]; then
# "any of" group, so create the group, make it an item, add
# to current group and mark the new group as current
gen_varname pd_group
group_declare "${pd_group}"
local -n g_ref=${pd_group}
g_ref[GROUP_TYPE_IDX]=${GROUP_ANY_OF}
unset -n g_ref
gen_varname pd_item
item_declare "${pd_item}"
local -n i_ref=${pd_item}
i_ref="g:${pd_group}"
unset -n i_ref
group_add_items "${group_stack[-1]}" "${pd_item}"
group_stack+=( "${pd_group}" )
pd_group_created=x
elif [[ ${token} =~ ^!?[A-Za-z0-9][A-Za-z0-9+_-]*\?$ ]]; then
# "use" group, so create the group, make it an item, add
# to current group and mark the new group as current
local -i disabled=GROUP_USE_ENABLED
local use=${token%?}
if [[ ${use} = '!'* ]]; then
disabled=GROUP_USE_DISABLED
use=${use:1}
fi
gen_varname pd_group
group_declare "${pd_group}"
local -n g_ref=${pd_group}
g_ref[GROUP_TYPE_IDX]=${GROUP_ALL_OF}
g_ref[GROUP_USE_IDX]=${use}
g_ref[GROUP_ENABLED_IDX]=${disabled}
unset -n g_ref
unset use disabled
gen_varname pd_item
item_declare "${pd_item}"
local -n i_ref=${pd_item}
i_ref="g:${pd_group}"
unset -n i_ref
group_add_items "${group_stack[-1]}" "${pd_item}"
group_stack+=( "${pd_group}" )
pd_group_created=x
elif [[ ${token} = '(' ]]; then
# beginning of a group; usually it is already created,
# because it is an "any of" or "use" group, but it is
# legal to specify just a "all of" group
if [[ -n ${pd_group_created} ]]; then
pd_group_created=
else
gen_varname pd_group
group_declare "${pd_group}"
gen_varname pd_item
item_declare "${pd_item}"
local -n i_ref=${pd_item}
i_ref="g:${pd_group}"
unset -n i_ref
group_add_items "${group_stack[-1]}" "${pd_item}"
group_stack+=( "${pd_group}" )
fi
elif [[ ${token} = ')' ]]; then
# end of a group, pop it from the stack
last_index=$(( ${#group_stack[@]} - 1 ))
unset "group_stack[${last_index}]"
elif [[ ${token} =~ ^[A-Za-z0-9_][A-Za-z0-9+_.-]*$ ]]; then
# license
if [[ dsf_type -ne __MCL_DSF_LICENSE ]]; then
fail "license tokens are only allowed for LICENSE keys (token: ${token@Q})"
fi
gen_varname pd_item
item_declare "${pd_item}"
local -n i_ref=${pd_item}
i_ref="l:${token}"
unset -n i_ref
group_add_items "${group_stack[-1]}" "${pd_item}"
elif [[ ${token} =~ ^!?!?(<|<=|=|~|>=|>)?[A-Za-z0-9_][A-Za-z0-9+_.-]*/[A-Za-z0-9_] ]]; then
# pds
if [[ dsf_type -ne __MCL_DSF_DEPEND ]]; then
fail "package dependency specification is only allowed for DEPEND-like keys (token: ${token@Q})"
fi
local -i blocks=PDS_NO_BLOCK
local operator='' name='' version='' slot=''
local -a use_requirements=()
case ${token} in
'!!'*)
blocks=PDS_STRONG_BLOCK
token=${token:2}
;;
'!'*)
blocks=PDS_WEAK_BLOCK
token=${token:1}
;;
esac
case ${token} in
'<='*|'>='*)
operator=${token:0:2}
token=${token:2}
;;
'<'*|'='*|'~'*|'>'*)
operator=${token:0:1}
token=${token:1}
;;
esac
if [[ ${token} = *']' ]]; then
local use_reqs_string=${token#*'['}
use_reqs_string=${use_reqs_string%']'}
token=${token%"[${use_reqs_string}]"}
local -a use_reqs
mapfile -t use_reqs <<<"${use_reqs_string//,/$'\n'}"
unset use_reqs_string
local ur name mode pretend pd_ur
for ur in "${use_reqs[@]}"; do
name=''
mode=''
pretend=''
case ${ur} in
'-'*)
mode='-'
ur=${ur:1}
;;
'!'*)
mode='!'
ur=${ur:1}
;;
esac
if [[ ${mode} != '-' ]]; then
case ${ur} in
*'=')
mode+='='
ur=${ur%'='}
;;
*'?')
mode+='?'
ur=${ur%'?'}
;;
esac
fi
if [[ -z ${mode} ]]; then
mode='+'
fi
if [[ ${ur} =~ \(([+-])\)$ ]]; then
pretend=${BASH_REMATCH[1]}
ur=${ur%"(${pretend})"}
fi
name=${ur}
gen_varname pd_ur
ur_declare "${pd_ur}"
local -n u_ref=${pd_ur}
u_ref[UR_NAME_IDX]=${name}
u_ref[UR_MODE_IDX]=${mode}
u_ref[UR_PRETEND_IDX]=${pretend}
unset -n u_ref
use_requirements+=( "${pd_ur}" )
done
unset pd_ur pretend mode name ur use_reqs
fi
if [[ ${token} = *:* ]]; then
slot=${token#*:}
token=${token%:"${slot}"}
fi
if [[ ${token} = *'*' ]]; then
operator='=*'
token=${token%'*'}
fi
local ver_ere_with_dash="-(${VER_ERE_UNBOUNDED})$"
if [[ ${token} =~ ${ver_ere_with_dash} ]]; then
version=${BASH_REMATCH[1]}
token=${token%"-${version}"}
fi
name=${token}
gen_varname pd_pds
pds_declare "${pd_pds}"
local -n p_ref=${pd_pds}
p_ref[PDS_BLOCKS_IDX]=${blocks}
p_ref[PDS_OP_IDX]=${operator}
p_ref[PDS_NAME_IDX]=${name}
p_ref[PDS_VER_IDX]=${version}
p_ref[PDS_SLOT_IDX]=${slot}
unset -n p_ref
pds_add_urs "${pd_pds}" "${use_requirements[@]}"
unset use_requirements slot version name operator blocks
gen_varname pd_item
item_declare "${pd_item}"
local -n i_ref=${pd_item}
i_ref="p:${pd_pds}"
unset -n i_ref
group_add_items "${group_stack[-1]}" "${pd_item}"
else
fail "unknown token ${token@Q}"
fi
done
if [[ ${#group_stack[@]} -ne 1 ]]; then
fail "botched parsing, group stack has ${#group_stack[@]} groups instead of 1"
fi
top_group_out_var_name_ref=${group_stack[0]}
if pkg_debug_enabled; then
local pd_group_str
group_to_string "${group_stack[0]}" pd_group_str
pkg_debug_print "dsf: ${pd_group_str}"
fi
}
function __mcl_parse_eclasses() {
local eclasses_string=${1}; shift
local -n eclasses_out_var_name_ref=${1}; shift
local eclasses_var_name
gen_varname eclasses_var_name
declare -ga "${eclasses_var_name}=()"
local -n eclasses_ref=${eclasses_var_name}
local -a tokens
mapfile -t tokens <<<"${eclasses_string//$'\t'/$'\n'}"
local token
local -i eclass_name_now=1
for token in "${tokens[@]}"; do
if [[ eclass_name_now -eq 1 ]]; then
eclasses_ref+=( "${token}" )
fi
eclass_name_now=$((eclass_name_now ^ 1))
done
eclasses_out_var_name_ref=${eclasses_var_name}
if pkg_debug_enabled; then
local joined_eclasses_string
join_by joined_eclasses_string ' ' "${eclasses_ref[@]}"
pkg_debug_print "eclasses: ${joined_eclasses_string}"
fi
}
function __mcl_parse_keywords() {
local keywords_string=${1}; shift
local -n keywords_out_var_name_ref=${1}; shift
# rest are architectures
local keywords_var_name
gen_varname keywords_var_name
declare -ga "${keywords_var_name}=()"
local -n keywords_ref=${keywords_var_name}
local -A keywords_set=()
local -a tokens
mapfile -t tokens <<<"${keywords_string// /$'\n'}"
local token
for token in "${tokens[@]}"; do
keywords_set["${token}"]=x
done
local has_hyphen_star=${keywords_set['-*']:-}
local arch mark kw_level_pair kw kw_name
local -i level
for arch; do
for kw_level_pair in "${arch}@${KW_STABLE}" "~${arch}@${KW_UNSTABLE}" "-${arch}@${KW_BROKEN}"; do
kw=${kw_level_pair%@*}
level=${kw_level_pair#*@}
mark=${keywords_set["${kw}"]:-}
if [[ -n ${mark} ]]; then
gen_varname kw_name
kw_declare "${kw_name}"
local -n k_ref=${kw_name}
k_ref[KW_NAME_IDX]=${arch}
k_ref[KW_LEVEL_IDX]=${level}
unset -n k_ref
keywords_ref+=( "${kw_name}" )
break
fi
done
if [[ -z ${mark} ]]; then
gen_varname kw_name
kw_declare "${kw_name}"
local -n k_ref=${kw_name}
k_ref[KW_NAME_IDX]=${arch}
if [[ -n ${has_hyphen_star} ]]; then
k_ref[KW_LEVEL_IDX]=${KW_BROKEN}
else
k_ref[KW_LEVEL_IDX]=${KW_UNKNOWN}
fi
unset -n k_ref
keywords_ref+=( "${kw_name}" )
fi
done
keywords_out_var_name_ref=${keywords_var_name}
if pkg_debug_enabled; then
local -a all_kws_strings=()
local kw_name pk_kw_str
local joined_kws_string
for kw_name in "${keywords_ref[@]}"; do
kw_to_string "${kw_name}" pk_kw_str
all_kws_strings+=( "${pk_kw_str}" )
done
join_by joined_kws_string ' ' "${all_kws_strings[@]}"
pkg_debug_print "keywords: ${joined_kws_string}"
fi
}
function __mcl_parse_iuse() {
local iuse_string=${1}; shift
local -n iuse_out_var_name_ref=${1}; shift
local iuse_var_name
gen_varname iuse_var_name
declare -ga "${iuse_var_name}=()"
local -n iuse_ref=${iuse_var_name}
local -a tokens
mapfile -t tokens <<<"${iuse_string// /$'\n'}"
local token pi_iuse
for token in "${tokens[@]}"; do
gen_varname pi_iuse
iuse_declare "${pi_iuse}"
local -n i_ref=${pi_iuse}
if [[ ${token} = '+'* ]]; then
i_ref[IUSE_MODE_IDX]=${IUSE_ENABLED}
token=${token:1}
fi
i_ref[IUSE_NAME_IDX]=${token}
unset -n i_ref
iuse_ref+=( "${pi_iuse}" )
done
iuse_out_var_name_ref=${iuse_var_name}
if pkg_debug_enabled; then
local -a all_iuse_strings=()
local iuse_name pi_iuse_str
local joined_iuse_string
for iuse_name in "${iuse_ref[@]}"; do
iuse_to_string "${iuse_name}" pi_iuse_str
all_iuse_strings+=( "${pi_iuse_str}" )
done
join_by joined_iuse_string ' ' "${all_iuse_strings[@]}"
pkg_debug_print "IUSE: ${joined_iuse_string}"
fi
}
function __mcl_unset_array() {
local array_name=${1}; shift
local item_unset_func=${1}; shift
if [[ ${array_name} = EMPTY_ARRAY ]]; then
return 0
fi
local -n array_ref=${array_name}
"${item_unset_func}" "${array_ref[@]}"
unset -n array_ref
unset "${array_name}"
}
fi