mirror of
				https://github.com/arcan1s/ahriman.git
				synced 2025-10-26 11:23:44 +00:00 
			
		
		
		
	Compare commits
	
		
			1 Commits
		
	
	
		
			b9203c38e0
			...
			feature/pk
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| aac3ccfc30 | 
							
								
								
									
										42
									
								
								.github/workflows/docker-image.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								.github/workflows/docker-image.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,42 @@ | |||||||
|  | name: docker image | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: [ master ] | ||||||
|  |     tags: | ||||||
|  |       - '*' | ||||||
|  |       - '!*rc*' | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   docker-image: | ||||||
|  |  | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|  |     steps: | ||||||
|  |       - name: extract docker metadata | ||||||
|  |         id: meta | ||||||
|  |         uses: docker/metadata-action@v3 | ||||||
|  |         with: | ||||||
|  |           images: | | ||||||
|  |             arcan1s/ahriman | ||||||
|  |           tags: | | ||||||
|  |             type=ref,event=tag | ||||||
|  |             type=edge | ||||||
|  |  | ||||||
|  |       - name: setup QEMU | ||||||
|  |         uses: docker/setup-qemu-action@v1 | ||||||
|  |  | ||||||
|  |       - name: setup docker buildx | ||||||
|  |         uses: docker/setup-buildx-action@v1 | ||||||
|  |  | ||||||
|  |       - name: login to docker hub | ||||||
|  |         uses: docker/login-action@v1 | ||||||
|  |         with: | ||||||
|  |           username: ${{ secrets.DOCKERHUB_USERNAME }} | ||||||
|  |           password: ${{ secrets.DOCKERHUB_TOKEN }} | ||||||
|  |  | ||||||
|  |       - name: build an image and push | ||||||
|  |         uses: docker/build-push-action@v2 | ||||||
|  |         with: | ||||||
|  |           push: true | ||||||
|  |           tags: ${{ steps.meta.outputs.tags }} | ||||||
							
								
								
									
										51
									
								
								.github/workflows/docker.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										51
									
								
								.github/workflows/docker.yml
									
									
									
									
										vendored
									
									
								
							| @ -1,51 +0,0 @@ | |||||||
| name: Docker image |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   push: |  | ||||||
|     branches: [ master ] |  | ||||||
|     tags: |  | ||||||
|       - '*' |  | ||||||
|       - '!*rc*' |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   docker-image: |  | ||||||
|  |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|  |  | ||||||
|     permissions: |  | ||||||
|       packages: write |  | ||||||
|  |  | ||||||
|     steps: |  | ||||||
|       - uses: docker/setup-qemu-action@v2 |  | ||||||
|  |  | ||||||
|       - uses: docker/setup-buildx-action@v2 |  | ||||||
|  |  | ||||||
|       - name: Login to docker hub |  | ||||||
|         uses: docker/login-action@v2 |  | ||||||
|         with: |  | ||||||
|           username: ${{ secrets.DOCKERHUB_USERNAME }} |  | ||||||
|           password: ${{ secrets.DOCKERHUB_TOKEN }} |  | ||||||
|  |  | ||||||
|       - name: Login to github container registry |  | ||||||
|         uses: docker/login-action@v2 |  | ||||||
|         with: |  | ||||||
|           registry: ghcr.io |  | ||||||
|           username: ${{ github.repository_owner }} |  | ||||||
|           password: ${{ secrets.GITHUB_TOKEN }} |  | ||||||
|  |  | ||||||
|       - name: Extract docker metadata |  | ||||||
|         id: meta |  | ||||||
|         uses: docker/metadata-action@v3 |  | ||||||
|         with: |  | ||||||
|           images: | |  | ||||||
|             arcan1s/ahriman |  | ||||||
|             ghcr.io/arcan1s/ahriman |  | ||||||
|           tags: | |  | ||||||
|             type=semver,pattern={{raw}} |  | ||||||
|             type=edge |  | ||||||
|  |  | ||||||
|       - name: Build an image and push |  | ||||||
|         uses: docker/build-push-action@v4 |  | ||||||
|         with: |  | ||||||
|           push: true |  | ||||||
|           tags: ${{ steps.meta.outputs.tags }} |  | ||||||
							
								
								
									
										12
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							| @ -1,4 +1,4 @@ | |||||||
| name: Release | name: release | ||||||
|  |  | ||||||
| on: | on: | ||||||
|   push: |   push: | ||||||
| @ -11,25 +11,25 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v3 |       - uses: actions/checkout@v2 | ||||||
|  |  | ||||||
|       - name: Extract version |       - name: extract version | ||||||
|         id: version |         id: version | ||||||
|         run: echo ::set-output name=VERSION::${GITHUB_REF#refs/tags/} |         run: echo ::set-output name=VERSION::${GITHUB_REF#refs/tags/} | ||||||
|  |  | ||||||
|       - name: Create changelog |       - name: create changelog | ||||||
|         id: changelog |         id: changelog | ||||||
|         uses: jaywcjlove/changelog-generator@main |         uses: jaywcjlove/changelog-generator@main | ||||||
|         with: |         with: | ||||||
|           token: ${{ secrets.GITHUB_TOKEN }} |           token: ${{ secrets.GITHUB_TOKEN }} | ||||||
|           filter: 'Release \d+\.\d+\.\d+' |           filter: 'Release \d+\.\d+\.\d+' | ||||||
|  |  | ||||||
|       - name: Create archive |       - name: create archive | ||||||
|         run: make archive |         run: make archive | ||||||
|         env: |         env: | ||||||
|           VERSION: ${{ steps.version.outputs.VERSION }} |           VERSION: ${{ steps.version.outputs.VERSION }} | ||||||
|  |  | ||||||
|       - name: Publish release |       - name: release | ||||||
|         uses: softprops/action-gh-release@v1 |         uses: softprops/action-gh-release@v1 | ||||||
|         with: |         with: | ||||||
|           body: | |           body: | | ||||||
|  | |||||||
| @ -1,4 +1,4 @@ | |||||||
| name: Setup | name: setup | ||||||
| 
 | 
 | ||||||
| on: | on: | ||||||
|   push: |   push: | ||||||
| @ -18,9 +18,9 @@ jobs: | |||||||
|       options: --privileged -w /build |       options: --privileged -w /build | ||||||
| 
 | 
 | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v3 |       - uses: actions/checkout@v2 | ||||||
| 
 | 
 | ||||||
|       - name: Setup the minimal service in arch linux container |       - name: setup the minimal service in arch linux container | ||||||
|         run: .github/workflows/setup.sh minimal |         run: .github/workflows/setup.sh minimal | ||||||
| 
 | 
 | ||||||
|   run-setup: |   run-setup: | ||||||
| @ -34,7 +34,7 @@ jobs: | |||||||
|       options: --privileged -w /build |       options: --privileged -w /build | ||||||
| 
 | 
 | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v3 |       - uses: actions/checkout@v2 | ||||||
| 
 | 
 | ||||||
|       - name: Setup the service in arch linux container |       - name: setup the service in arch linux container | ||||||
|         run: .github/workflows/setup.sh |         run: .github/workflows/setup.sh | ||||||
| @ -1,4 +1,4 @@ | |||||||
| name: Tests | name: tests | ||||||
| 
 | 
 | ||||||
| on: | on: | ||||||
|   push: |   push: | ||||||
| @ -18,7 +18,7 @@ jobs: | |||||||
|       options: -w /build |       options: -w /build | ||||||
| 
 | 
 | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v3 |       - uses: actions/checkout@v2 | ||||||
| 
 | 
 | ||||||
|       - name: Run check and tests in arch linux container |       - name: run check and tests in arch linux container | ||||||
|         run: .github/workflows/tests.sh |         run: .github/workflows/tests.sh | ||||||
							
								
								
									
										12
									
								
								.github/workflows/setup.sh
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/workflows/setup.sh
									
									
									
									
										vendored
									
									
								
							| @ -10,9 +10,9 @@ echo -e '[arcanisrepo]\nServer = http://repo.arcanis.me/$arch\nSigLevel = Never' | |||||||
| # refresh the image | # refresh the image | ||||||
| pacman --noconfirm -Syu | pacman --noconfirm -Syu | ||||||
| # main dependencies | # main dependencies | ||||||
| pacman --noconfirm -Sy base-devel devtools git pyalpm python-cerberus python-inflection python-passlib python-requests python-srcinfo python-systemd sudo | pacman --noconfirm -Sy base-devel devtools-git-poc git pyalpm python-cerberus python-inflection python-passlib python-requests python-srcinfo sudo | ||||||
| # make dependencies | # make dependencies | ||||||
| pacman --noconfirm -Sy python-build python-flit python-installer python-wheel | pacman --noconfirm -Sy python-build python-installer python-wheel | ||||||
| # optional dependencies | # optional dependencies | ||||||
| if [[ -z $MINIMAL_INSTALL ]]; then | if [[ -z $MINIMAL_INSTALL ]]; then | ||||||
|     # VCS support |     # VCS support | ||||||
| @ -34,6 +34,8 @@ pacman --noconfirm -U ahriman-1.0.0-1-any.pkg.tar.zst | |||||||
| # create machine-id which is required by build tools | # create machine-id which is required by build tools | ||||||
| systemd-machine-id-setup | systemd-machine-id-setup | ||||||
|  |  | ||||||
|  | # special thing for the container, because /dev/log interface is not available there | ||||||
|  | sed -i "s/handlers = syslog_handler/handlers = console_handler/g" /etc/ahriman.ini.d/logging.ini | ||||||
| # initial setup command as root | # initial setup command as root | ||||||
| [[ -z $MINIMAL_INSTALL ]] && WEB_ARGS=("--web-port" "8080") | [[ -z $MINIMAL_INSTALL ]] && WEB_ARGS=("--web-port" "8080") | ||||||
| ahriman -a x86_64 service-setup --packager "ahriman bot <ahriman@example.com>" --repository "github" "${WEB_ARGS[@]}" | ahriman -a x86_64 service-setup --packager "ahriman bot <ahriman@example.com>" --repository "github" "${WEB_ARGS[@]}" | ||||||
| @ -46,11 +48,13 @@ if [[ -z $MINIMAL_INSTALL ]]; then | |||||||
|     # run web service (detached) |     # run web service (detached) | ||||||
|     sudo -u ahriman -- ahriman -a x86_64 web & |     sudo -u ahriman -- ahriman -a x86_64 web & | ||||||
|     WEB_PID=$! |     WEB_PID=$! | ||||||
|  |     sleep 15s  # wait for the web service activation | ||||||
| fi | fi | ||||||
| # add the first package | # add the first package | ||||||
| sudo -u ahriman -- ahriman package-add --now yay | # the build itself does not really work in the container | ||||||
|  | sudo -u ahriman -- ahriman package-add --now ahriman | ||||||
| # check if package was actually installed | # check if package was actually installed | ||||||
| test -n "$(find "/var/lib/ahriman/repository/x86_64" -name "yay*pkg*")" | test -n "$(find "/var/lib/ahriman/repository/x86_64" -name "ahriman*pkg*")" | ||||||
| # run package check | # run package check | ||||||
| sudo -u ahriman -- ahriman repo-update | sudo -u ahriman -- ahriman repo-update | ||||||
| # stop web service lol | # stop web service lol | ||||||
|  | |||||||
							
								
								
									
										804
									
								
								.pylintrc
									
									
									
									
									
								
							
							
						
						
									
										804
									
								
								.pylintrc
									
									
									
									
									
								
							| @ -1,78 +1,28 @@ | |||||||
| [MAIN] | [MASTER] | ||||||
|  |  | ||||||
| # Analyse import fallback blocks. This can be used to support both Python 2 and |  | ||||||
| # 3 compatible code, which means that the block might have code that exists |  | ||||||
| # only in one or another interpreter, leading to false positives when analysed. |  | ||||||
| analyse-fallback-blocks=no |  | ||||||
|  |  | ||||||
| # Clear in-memory caches upon conclusion of linting. Useful if running pylint |  | ||||||
| # in a server-like mode. |  | ||||||
| clear-cache-post-run=no |  | ||||||
|  |  | ||||||
| # Load and enable all available extensions. Use --list-extensions to see a list |  | ||||||
| # all available extensions. |  | ||||||
| #enable-all-extensions= |  | ||||||
|  |  | ||||||
| # In error mode, messages with a category besides ERROR or FATAL are |  | ||||||
| # suppressed, and no reports are done by default. Error mode is compatible with |  | ||||||
| # disabling specific errors. |  | ||||||
| #errors-only= |  | ||||||
|  |  | ||||||
| # Always return a 0 (non-error) status code, even if lint errors are found. |  | ||||||
| # This is primarily useful in continuous integration scripts. |  | ||||||
| #exit-zero= |  | ||||||
|  |  | ||||||
| # A comma-separated list of package or module names from where C extensions may | # A comma-separated list of package or module names from where C extensions may | ||||||
| # be loaded. Extensions are loading into the active Python interpreter and may | # be loaded. Extensions are loading into the active Python interpreter and may | ||||||
| # run arbitrary code. | # run arbitrary code. | ||||||
| extension-pkg-allow-list= |  | ||||||
|  |  | ||||||
| # A comma-separated list of package or module names from where C extensions may |  | ||||||
| # be loaded. Extensions are loading into the active Python interpreter and may |  | ||||||
| # run arbitrary code. (This is an alternative name to extension-pkg-allow-list |  | ||||||
| # for backward compatibility.) |  | ||||||
| extension-pkg-whitelist= | extension-pkg-whitelist= | ||||||
|  |  | ||||||
| # Return non-zero exit code if any of these messages/categories are detected, | # Specify a score threshold to be exceeded before program exits with error. | ||||||
| # even if score is above --fail-under value. Syntax same as enable. Messages | fail-under=10.0 | ||||||
| # specified are enabled, while categories only check already-enabled messages. |  | ||||||
| fail-on= |  | ||||||
|  |  | ||||||
| # Specify a score threshold under which the program will exit with error. | # Add files or directories to the blacklist. They should be base names, not | ||||||
| fail-under=10 | # paths. | ||||||
|  |  | ||||||
| # Interpret the stdin as a python script, whose filename needs to be passed as |  | ||||||
| # the module_or_package argument. |  | ||||||
| #from-stdin= |  | ||||||
|  |  | ||||||
| # Files or directories to be skipped. They should be base names, not paths. |  | ||||||
| ignore=CVS | ignore=CVS | ||||||
|  |  | ||||||
| # Add files or directories matching the regular expressions patterns to the | # Add files or directories matching the regex patterns to the blacklist. The | ||||||
| # ignore-list. The regex matches against paths and can be in Posix or Windows | # regex matches against base names, not paths. | ||||||
| # format. Because '\\' represents the directory delimiter on Windows systems, | ignore-patterns= | ||||||
| # it can't be used as an escape character. |  | ||||||
| ignore-paths= |  | ||||||
|  |  | ||||||
| # Files or directories matching the regular expression patterns are skipped. |  | ||||||
| # The regex matches against base names, not paths. The default value ignores |  | ||||||
| # Emacs file locks |  | ||||||
| ignore-patterns=^\.# |  | ||||||
|  |  | ||||||
| # List of module names for which member attributes should not be checked |  | ||||||
| # (useful for modules/projects where namespaces are manipulated during runtime |  | ||||||
| # and thus existing member attributes cannot be deduced by static analysis). It |  | ||||||
| # supports qualified module names, as well as Unix pattern matching. |  | ||||||
| ignored-modules= |  | ||||||
|  |  | ||||||
| # Python code to execute, usually for sys.path manipulation such as | # Python code to execute, usually for sys.path manipulation such as | ||||||
| # pygtk.require(). | # pygtk.require(). | ||||||
| #init-hook= | #init-hook= | ||||||
|  |  | ||||||
| # Use multiple processes to speed up Pylint. Specifying 0 will auto-detect the | # Use multiple processes to speed up Pylint. Specifying 0 will auto-detect the | ||||||
| # number of processors available to use, and will cap the count on Windows to | # number of processors available to use. | ||||||
| # avoid hangs. | jobs=0 | ||||||
| jobs=1 |  | ||||||
|  |  | ||||||
| # Control the amount of potential inferred values when inferring a single | # Control the amount of potential inferred values when inferring a single | ||||||
| # object. This can help the performance when dealing with large functions or | # object. This can help the performance when dealing with large functions or | ||||||
| @ -86,19 +36,6 @@ load-plugins= | |||||||
| # Pickle collected data for later comparisons. | # Pickle collected data for later comparisons. | ||||||
| persistent=yes | persistent=yes | ||||||
|  |  | ||||||
| # Minimum Python version to use for version dependent checks. Will default to |  | ||||||
| # the version used to run pylint. |  | ||||||
| py-version=3.11 |  | ||||||
|  |  | ||||||
| # Discover python modules and packages in the file system subtree. |  | ||||||
| recursive=no |  | ||||||
|  |  | ||||||
| # Add paths to the list of the source roots. Supports globbing patterns. The |  | ||||||
| # source root is an absolute path or a path relative to the current working |  | ||||||
| # directory used to determine a package namespace for modules located under the |  | ||||||
| # source root. |  | ||||||
| source-roots= |  | ||||||
|  |  | ||||||
| # When enabled, pylint would attempt to guess common misconfiguration and emit | # When enabled, pylint would attempt to guess common misconfiguration and emit | ||||||
| # user-friendly hints instead of false-positive error messages. | # user-friendly hints instead of false-positive error messages. | ||||||
| suggestion-mode=yes | suggestion-mode=yes | ||||||
| @ -107,8 +44,120 @@ suggestion-mode=yes | |||||||
| # active Python interpreter and may run arbitrary code. | # active Python interpreter and may run arbitrary code. | ||||||
| unsafe-load-any-extension=no | unsafe-load-any-extension=no | ||||||
|  |  | ||||||
| # In verbose mode, extra non-checker-related info will be displayed. |  | ||||||
| #verbose= | [MESSAGES CONTROL] | ||||||
|  |  | ||||||
|  | # Only show warnings with the listed confidence levels. Leave empty to show | ||||||
|  | # all. Valid levels: HIGH, INFERENCE, INFERENCE_FAILURE, UNDEFINED. | ||||||
|  | confidence= | ||||||
|  |  | ||||||
|  | # Disable the message, report, category or checker with the given id(s). You | ||||||
|  | # can either give multiple identifiers separated by comma (,) or put this | ||||||
|  | # option multiple times (only on the command line, not in the configuration | ||||||
|  | # file where it should appear only once). You can also use "--disable=all" to | ||||||
|  | # disable everything first and then reenable specific checks. For example, if | ||||||
|  | # you want to run only the similarities checker, you can use "--disable=all | ||||||
|  | # --enable=similarities". If you want to run only the classes checker, but have | ||||||
|  | # no Warning level messages displayed, use "--disable=all --enable=classes | ||||||
|  | # --disable=W". | ||||||
|  | disable=raw-checker-failed, | ||||||
|  |         bad-inline-option, | ||||||
|  |         locally-disabled, | ||||||
|  |         file-ignored, | ||||||
|  |         suppressed-message, | ||||||
|  |         useless-suppression, | ||||||
|  |         deprecated-pragma, | ||||||
|  |         use-symbolic-message-instead, | ||||||
|  |         missing-module-docstring, | ||||||
|  |         line-too-long, | ||||||
|  |         no-name-in-module, | ||||||
|  |         import-outside-toplevel, | ||||||
|  |         invalid-name, | ||||||
|  |         raise-missing-from, | ||||||
|  |         wrong-import-order, | ||||||
|  |         too-few-public-methods, | ||||||
|  |         too-many-instance-attributes, | ||||||
|  |         broad-except, | ||||||
|  |         too-many-ancestors, | ||||||
|  |         fixme, | ||||||
|  |         too-many-arguments, | ||||||
|  |         duplicate-code, | ||||||
|  |         cyclic-import, | ||||||
|  |         confusing-with-statement, | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Enable the message, report, category or checker with the given id(s). You can | ||||||
|  | # either give multiple identifier separated by comma (,) or put this option | ||||||
|  | # multiple time (only on the command line, not in the configuration file where | ||||||
|  | # it should appear only once). See also the "--disable" option for examples. | ||||||
|  | enable=c-extension-no-member | ||||||
|  |  | ||||||
|  |  | ||||||
|  | [REPORTS] | ||||||
|  |  | ||||||
|  | # Python expression which should return a score less than or equal to 10. You | ||||||
|  | # have access to the variables 'error', 'warning', 'refactor', and 'convention' | ||||||
|  | # which contain the number of messages in each category, as well as 'statement' | ||||||
|  | # which is the total number of statements analyzed. This score is used by the | ||||||
|  | # global evaluation report (RP0004). | ||||||
|  | evaluation=10.0 - ((float(5 * error + warning + refactor + convention) / statement) * 10) | ||||||
|  |  | ||||||
|  | # Template used to display messages. This is a python new-style format string | ||||||
|  | # used to format the message information. See doc for all details. | ||||||
|  | #msg-template= | ||||||
|  |  | ||||||
|  | # Set the output format. Available formats are text, parseable, colorized, json | ||||||
|  | # and msvs (visual studio). You can also give a reporter class, e.g. | ||||||
|  | # mypackage.mymodule.MyReporterClass. | ||||||
|  | output-format=text | ||||||
|  |  | ||||||
|  | # Tells whether to display a full report or only the messages. | ||||||
|  | reports=no | ||||||
|  |  | ||||||
|  | # Activate the evaluation score. | ||||||
|  | score=yes | ||||||
|  |  | ||||||
|  |  | ||||||
|  | [REFACTORING] | ||||||
|  |  | ||||||
|  | # Maximum number of nested blocks for function / method body | ||||||
|  | max-nested-blocks=5 | ||||||
|  |  | ||||||
|  | # Complete name of functions that never returns. When checking for | ||||||
|  | # inconsistent-return-statements if a never returning function is called then | ||||||
|  | # it will be considered as an explicit return statement and no message will be | ||||||
|  | # printed. | ||||||
|  | never-returning-functions=sys.exit | ||||||
|  |  | ||||||
|  |  | ||||||
|  | [FORMAT] | ||||||
|  |  | ||||||
|  | # Expected format of line ending, e.g. empty (any line ending), LF or CRLF. | ||||||
|  | expected-line-ending-format= | ||||||
|  |  | ||||||
|  | # Regexp for a line that is allowed to be longer than the limit. | ||||||
|  | ignore-long-lines=^\s*(# )?<?https?://\S+>?$ | ||||||
|  |  | ||||||
|  | # Number of spaces of indent required inside a hanging or continued line. | ||||||
|  | indent-after-paren=4 | ||||||
|  |  | ||||||
|  | # String used as indentation unit. This is usually "    " (4 spaces) or "\t" (1 | ||||||
|  | # tab). | ||||||
|  | indent-string='    ' | ||||||
|  |  | ||||||
|  | # Maximum number of characters on a single line. | ||||||
|  | max-line-length=100 | ||||||
|  |  | ||||||
|  | # Maximum number of lines in a module. | ||||||
|  | max-module-lines=400 | ||||||
|  |  | ||||||
|  | # Allow the body of a class to be on the same line as the declaration if body | ||||||
|  | # contains single statement. | ||||||
|  | single-line-class-stmt=no | ||||||
|  |  | ||||||
|  | # Allow the body of an if to be on the same line as the test if there is no | ||||||
|  | # else. | ||||||
|  | single-line-if-stmt=no | ||||||
|  |  | ||||||
|  |  | ||||||
| [BASIC] | [BASIC] | ||||||
| @ -117,15 +166,13 @@ unsafe-load-any-extension=no | |||||||
| argument-naming-style=snake_case | argument-naming-style=snake_case | ||||||
|  |  | ||||||
| # Regular expression matching correct argument names. Overrides argument- | # Regular expression matching correct argument names. Overrides argument- | ||||||
| # naming-style. If left empty, argument names will be checked with the set | # naming-style. | ||||||
| # naming style. |  | ||||||
| #argument-rgx= | #argument-rgx= | ||||||
|  |  | ||||||
| # Naming style matching correct attribute names. | # Naming style matching correct attribute names. | ||||||
| attr-naming-style=snake_case | attr-naming-style=snake_case | ||||||
|  |  | ||||||
| # Regular expression matching correct attribute names. Overrides attr-naming- | # Regular expression matching correct attribute names. Overrides attr-naming- | ||||||
| # style. If left empty, attribute names will be checked with the set naming |  | ||||||
| # style. | # style. | ||||||
| #attr-rgx= | #attr-rgx= | ||||||
|  |  | ||||||
| @ -145,30 +192,20 @@ bad-names-rgxs= | |||||||
| class-attribute-naming-style=any | class-attribute-naming-style=any | ||||||
|  |  | ||||||
| # Regular expression matching correct class attribute names. Overrides class- | # Regular expression matching correct class attribute names. Overrides class- | ||||||
| # attribute-naming-style. If left empty, class attribute names will be checked | # attribute-naming-style. | ||||||
| # with the set naming style. |  | ||||||
| #class-attribute-rgx= | #class-attribute-rgx= | ||||||
|  |  | ||||||
| # Naming style matching correct class constant names. |  | ||||||
| class-const-naming-style=UPPER_CASE |  | ||||||
|  |  | ||||||
| # Regular expression matching correct class constant names. Overrides class- |  | ||||||
| # const-naming-style. If left empty, class constant names will be checked with |  | ||||||
| # the set naming style. |  | ||||||
| #class-const-rgx= |  | ||||||
|  |  | ||||||
| # Naming style matching correct class names. | # Naming style matching correct class names. | ||||||
| class-naming-style=PascalCase | class-naming-style=PascalCase | ||||||
|  |  | ||||||
| # Regular expression matching correct class names. Overrides class-naming- | # Regular expression matching correct class names. Overrides class-naming- | ||||||
| # style. If left empty, class names will be checked with the set naming style. | # style. | ||||||
| #class-rgx= | #class-rgx= | ||||||
|  |  | ||||||
| # Naming style matching correct constant names. | # Naming style matching correct constant names. | ||||||
| const-naming-style=UPPER_CASE | const-naming-style=UPPER_CASE | ||||||
|  |  | ||||||
| # Regular expression matching correct constant names. Overrides const-naming- | # Regular expression matching correct constant names. Overrides const-naming- | ||||||
| # style. If left empty, constant names will be checked with the set naming |  | ||||||
| # style. | # style. | ||||||
| #const-rgx= | #const-rgx= | ||||||
|  |  | ||||||
| @ -180,8 +217,7 @@ docstring-min-length=-1 | |||||||
| function-naming-style=snake_case | function-naming-style=snake_case | ||||||
|  |  | ||||||
| # Regular expression matching correct function names. Overrides function- | # Regular expression matching correct function names. Overrides function- | ||||||
| # naming-style. If left empty, function names will be checked with the set | # naming-style. | ||||||
| # naming style. |  | ||||||
| #function-rgx= | #function-rgx= | ||||||
|  |  | ||||||
| # Good variable names which should always be accepted, separated by a comma. | # Good variable names which should always be accepted, separated by a comma. | ||||||
| @ -203,22 +239,21 @@ include-naming-hint=no | |||||||
| inlinevar-naming-style=any | inlinevar-naming-style=any | ||||||
|  |  | ||||||
| # Regular expression matching correct inline iteration names. Overrides | # Regular expression matching correct inline iteration names. Overrides | ||||||
| # inlinevar-naming-style. If left empty, inline iteration names will be checked | # inlinevar-naming-style. | ||||||
| # with the set naming style. |  | ||||||
| #inlinevar-rgx= | #inlinevar-rgx= | ||||||
|  |  | ||||||
| # Naming style matching correct method names. | # Naming style matching correct method names. | ||||||
| method-naming-style=snake_case | method-naming-style=snake_case | ||||||
|  |  | ||||||
| # Regular expression matching correct method names. Overrides method-naming- | # Regular expression matching correct method names. Overrides method-naming- | ||||||
| # style. If left empty, method names will be checked with the set naming style. | # style. | ||||||
| #method-rgx= | #method-rgx= | ||||||
|  |  | ||||||
| # Naming style matching correct module names. | # Naming style matching correct module names. | ||||||
| module-naming-style=snake_case | module-naming-style=snake_case | ||||||
|  |  | ||||||
| # Regular expression matching correct module names. Overrides module-naming- | # Regular expression matching correct module names. Overrides module-naming- | ||||||
| # style. If left empty, module names will be checked with the set naming style. | # style. | ||||||
| #module-rgx= | #module-rgx= | ||||||
|  |  | ||||||
| # Colon-delimited sets of names that determine each other's naming style when | # Colon-delimited sets of names that determine each other's naming style when | ||||||
| @ -234,56 +269,209 @@ no-docstring-rgx=^_ | |||||||
| # These decorators are taken in consideration only for invalid-name. | # These decorators are taken in consideration only for invalid-name. | ||||||
| property-classes=abc.abstractproperty | property-classes=abc.abstractproperty | ||||||
|  |  | ||||||
| # Regular expression matching correct type alias names. If left empty, type |  | ||||||
| # alias names will be checked with the set naming style. |  | ||||||
| #typealias-rgx= |  | ||||||
|  |  | ||||||
| # Regular expression matching correct type variable names. If left empty, type |  | ||||||
| # variable names will be checked with the set naming style. |  | ||||||
| #typevar-rgx= |  | ||||||
|  |  | ||||||
| # Naming style matching correct variable names. | # Naming style matching correct variable names. | ||||||
| variable-naming-style=snake_case | variable-naming-style=snake_case | ||||||
|  |  | ||||||
| # Regular expression matching correct variable names. Overrides variable- | # Regular expression matching correct variable names. Overrides variable- | ||||||
| # naming-style. If left empty, variable names will be checked with the set | # naming-style. | ||||||
| # naming style. |  | ||||||
| #variable-rgx= | #variable-rgx= | ||||||
|  |  | ||||||
|  |  | ||||||
| [CLASSES] | [TYPECHECK] | ||||||
|  |  | ||||||
| # Warn about protected attribute access inside special methods | # List of decorators that produce context managers, such as | ||||||
| check-protected-access-in-special-methods=no | # contextlib.contextmanager. Add to this list to register other decorators that | ||||||
|  | # produce valid context managers. | ||||||
|  | contextmanager-decorators=contextlib.contextmanager | ||||||
|  |  | ||||||
| # List of method names used to declare (i.e. assign) instance attributes. | # List of members which are set dynamically and missed by pylint inference | ||||||
| defining-attr-methods=__init__, | # system, and so shouldn't trigger E1101 when accessed. Python regular | ||||||
|                       __new__, | # expressions are accepted. | ||||||
|                       setUp, | generated-members= | ||||||
|                       asyncSetUp, |  | ||||||
|                       __post_init__ |  | ||||||
|  |  | ||||||
| # List of member names, which should be excluded from the protected access | # Tells whether missing members accessed in mixin class should be ignored. A | ||||||
| # warning. | # mixin class is detected if its name ends with "mixin" (case insensitive). | ||||||
| exclude-protected=_asdict,_fields,_replace,_source,_make,os._exit | ignore-mixin-members=yes | ||||||
|  |  | ||||||
| # List of valid names for the first argument in a class method. | # Tells whether to warn about missing members when the owner of the attribute | ||||||
| valid-classmethod-first-arg=cls | # is inferred to be None. | ||||||
|  | ignore-none=yes | ||||||
|  |  | ||||||
| # List of valid names for the first argument in a metaclass class method. | # This flag controls whether pylint should warn about no-member and similar | ||||||
| valid-metaclass-classmethod-first-arg=mcs | # checks whenever an opaque object is returned when inferring. The inference | ||||||
|  | # can return multiple potential results while evaluating a Python object, but | ||||||
|  | # some branches might not be evaluated, which results in partial inference. In | ||||||
|  | # that case, it might be useful to still emit no-member and other checks for | ||||||
|  | # the rest of the inferred objects. | ||||||
|  | ignore-on-opaque-inference=yes | ||||||
|  |  | ||||||
|  | # List of class names for which member attributes should not be checked (useful | ||||||
|  | # for classes with dynamically set attributes). This supports the use of | ||||||
|  | # qualified names. | ||||||
|  | ignored-classes=optparse.Values,thread._local,_thread._local | ||||||
|  |  | ||||||
|  | # List of module names for which member attributes should not be checked | ||||||
|  | # (useful for modules/projects where namespaces are manipulated during runtime | ||||||
|  | # and thus existing member attributes cannot be deduced by static analysis). It | ||||||
|  | # supports qualified module names, as well as Unix pattern matching. | ||||||
|  | ignored-modules= | ||||||
|  |  | ||||||
|  | # Show a hint with possible names when a member name was not found. The aspect | ||||||
|  | # of finding the hint is based on edit distance. | ||||||
|  | missing-member-hint=yes | ||||||
|  |  | ||||||
|  | # The minimum edit distance a name should have in order to be considered a | ||||||
|  | # similar match for a missing member name. | ||||||
|  | missing-member-hint-distance=1 | ||||||
|  |  | ||||||
|  | # The total number of similar names that should be taken in consideration when | ||||||
|  | # showing a hint for a missing member. | ||||||
|  | missing-member-max-choices=1 | ||||||
|  |  | ||||||
|  | # List of decorators that change the signature of a decorated function. | ||||||
|  | signature-mutators= | ||||||
|  |  | ||||||
|  |  | ||||||
|  | [SIMILARITIES] | ||||||
|  |  | ||||||
|  | # Ignore comments when computing similarities. | ||||||
|  | ignore-comments=yes | ||||||
|  |  | ||||||
|  | # Ignore docstrings when computing similarities. | ||||||
|  | ignore-docstrings=yes | ||||||
|  |  | ||||||
|  | # Ignore imports when computing similarities. | ||||||
|  | ignore-imports=no | ||||||
|  |  | ||||||
|  | # Minimum lines number of a similarity. | ||||||
|  | min-similarity-lines=4 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | [LOGGING] | ||||||
|  |  | ||||||
|  | # The type of string formatting that logging methods do. `old` means using % | ||||||
|  | # formatting, `new` is for `{}` formatting. | ||||||
|  | logging-format-style=old | ||||||
|  |  | ||||||
|  | # Logging modules to check that the string format arguments are in logging | ||||||
|  | # function parameter format. | ||||||
|  | logging-modules=logging | ||||||
|  |  | ||||||
|  |  | ||||||
|  | [MISCELLANEOUS] | ||||||
|  |  | ||||||
|  | # List of note tags to take in consideration, separated by a comma. | ||||||
|  | notes=FIXME, | ||||||
|  |       XXX, | ||||||
|  |       TODO | ||||||
|  |  | ||||||
|  | # Regular expression of note tags to take in consideration. | ||||||
|  | #notes-rgx= | ||||||
|  |  | ||||||
|  |  | ||||||
|  | [SPELLING] | ||||||
|  |  | ||||||
|  | # Limits count of emitted suggestions for spelling mistakes. | ||||||
|  | max-spelling-suggestions=4 | ||||||
|  |  | ||||||
|  | # Spelling dictionary name. Available dictionaries: none. To make it work, | ||||||
|  | # install the python-enchant package. | ||||||
|  | spelling-dict= | ||||||
|  |  | ||||||
|  | # List of comma separated words that should not be checked. | ||||||
|  | spelling-ignore-words= | ||||||
|  |  | ||||||
|  | # A path to a file that contains the private dictionary; one word per line. | ||||||
|  | spelling-private-dict-file= | ||||||
|  |  | ||||||
|  | # Tells whether to store unknown words to the private dictionary (see the | ||||||
|  | # --spelling-private-dict-file option) instead of raising a message. | ||||||
|  | spelling-store-unknown-words=no | ||||||
|  |  | ||||||
|  |  | ||||||
|  | [VARIABLES] | ||||||
|  |  | ||||||
|  | # List of additional names supposed to be defined in builtins. Remember that | ||||||
|  | # you should avoid defining new builtins when possible. | ||||||
|  | additional-builtins= | ||||||
|  |  | ||||||
|  | # Tells whether unused global variables should be treated as a violation. | ||||||
|  | allow-global-unused-variables=yes | ||||||
|  |  | ||||||
|  | # List of strings which can identify a callback function by name. A callback | ||||||
|  | # name must start or end with one of those strings. | ||||||
|  | callbacks=cb_, | ||||||
|  |           _cb | ||||||
|  |  | ||||||
|  | # A regular expression matching the name of dummy variables (i.e. expected to | ||||||
|  | # not be used). | ||||||
|  | dummy-variables-rgx=_+$|(_[a-zA-Z0-9_]*[a-zA-Z0-9]+?$)|dummy|^ignored_|^unused_ | ||||||
|  |  | ||||||
|  | # Argument names that match this expression will be ignored. Default to name | ||||||
|  | # with leading underscore. | ||||||
|  | ignored-argument-names=_.*|^ignored_|^unused_ | ||||||
|  |  | ||||||
|  | # Tells whether we should check for unused import in __init__ files. | ||||||
|  | init-import=no | ||||||
|  |  | ||||||
|  | # List of qualified module names which can have objects that can redefine | ||||||
|  | # builtins. | ||||||
|  | redefining-builtins-modules=six.moves,past.builtins,future.builtins,builtins,io | ||||||
|  |  | ||||||
|  |  | ||||||
|  | [STRING] | ||||||
|  |  | ||||||
|  | # This flag controls whether inconsistent-quotes generates a warning when the | ||||||
|  | # character used as a quote delimiter is used inconsistently within a module. | ||||||
|  | check-quote-consistency=no | ||||||
|  |  | ||||||
|  | # This flag controls whether the implicit-str-concat should generate a warning | ||||||
|  | # on implicit string concatenation in sequences defined over several lines. | ||||||
|  | check-str-concat-over-line-jumps=no | ||||||
|  |  | ||||||
|  |  | ||||||
|  | [IMPORTS] | ||||||
|  |  | ||||||
|  | # List of modules that can be imported at any level, not just the top level | ||||||
|  | # one. | ||||||
|  | allow-any-import-level= | ||||||
|  |  | ||||||
|  | # Allow wildcard imports from modules that define __all__. | ||||||
|  | allow-wildcard-with-all=no | ||||||
|  |  | ||||||
|  | # Analyse import fallback blocks. This can be used to support both Python 2 and | ||||||
|  | # 3 compatible code, which means that the block might have code that exists | ||||||
|  | # only in one or another interpreter, leading to false positives when analysed. | ||||||
|  | analyse-fallback-blocks=no | ||||||
|  |  | ||||||
|  | # Deprecated modules which should not be used, separated by a comma. | ||||||
|  | deprecated-modules=optparse,tkinter.tix | ||||||
|  |  | ||||||
|  | # Create a graph of external dependencies in the given file (report RP0402 must | ||||||
|  | # not be disabled). | ||||||
|  | ext-import-graph= | ||||||
|  |  | ||||||
|  | # Create a graph of every (i.e. internal and external) dependencies in the | ||||||
|  | # given file (report RP0402 must not be disabled). | ||||||
|  | import-graph= | ||||||
|  |  | ||||||
|  | # Create a graph of internal dependencies in the given file (report RP0402 must | ||||||
|  | # not be disabled). | ||||||
|  | int-import-graph= | ||||||
|  |  | ||||||
|  | # Force import order to recognize a module as part of the standard | ||||||
|  | # compatibility libraries. | ||||||
|  | known-standard-library= | ||||||
|  |  | ||||||
|  | # Force import order to recognize a module as part of a third party library. | ||||||
|  | known-third-party=enchant | ||||||
|  |  | ||||||
|  | # Couples of modules and preferred modules, separated by a comma. | ||||||
|  | preferred-modules= | ||||||
|  |  | ||||||
|  |  | ||||||
| [DESIGN] | [DESIGN] | ||||||
|  |  | ||||||
| # List of regular expressions of class ancestor names to ignore when counting |  | ||||||
| # public methods (see R0903) |  | ||||||
| exclude-too-few-public-methods= |  | ||||||
|  |  | ||||||
| # List of qualified class names to ignore when counting class parents (see |  | ||||||
| # R0901) |  | ||||||
| ignored-parents= |  | ||||||
|  |  | ||||||
| # Maximum number of arguments for function / method. | # Maximum number of arguments for function / method. | ||||||
| max-args=5 | max-args=5 | ||||||
|  |  | ||||||
| @ -315,331 +503,35 @@ max-statements=50 | |||||||
| min-public-methods=2 | min-public-methods=2 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | [CLASSES] | ||||||
|  |  | ||||||
|  | # Warn about protected attribute access inside special methods | ||||||
|  | check-protected-access-in-special-methods=no | ||||||
|  |  | ||||||
|  | # List of method names used to declare (i.e. assign) instance attributes. | ||||||
|  | defining-attr-methods=__init__, | ||||||
|  |                       __new__, | ||||||
|  |                       setUp, | ||||||
|  |                       __post_init__ | ||||||
|  |  | ||||||
|  | # List of member names, which should be excluded from the protected access | ||||||
|  | # warning. | ||||||
|  | exclude-protected=_asdict, | ||||||
|  |                   _fields, | ||||||
|  |                   _replace, | ||||||
|  |                   _source, | ||||||
|  |                   _make | ||||||
|  |  | ||||||
|  | # List of valid names for the first argument in a class method. | ||||||
|  | valid-classmethod-first-arg=cls | ||||||
|  |  | ||||||
|  | # List of valid names for the first argument in a metaclass class method. | ||||||
|  | valid-metaclass-classmethod-first-arg=cls | ||||||
|  |  | ||||||
|  |  | ||||||
| [EXCEPTIONS] | [EXCEPTIONS] | ||||||
|  |  | ||||||
| # Exceptions that will emit a warning when caught. | # Exceptions that will emit a warning when being caught. Defaults to | ||||||
| overgeneral-exceptions=builtins.BaseException,builtins.Exception | # "BaseException, Exception". | ||||||
|  | overgeneral-exceptions=builtins.BaseException, | ||||||
|  |                        builtins.Exception | ||||||
| [FORMAT] |  | ||||||
|  |  | ||||||
| # Expected format of line ending, e.g. empty (any line ending), LF or CRLF. |  | ||||||
| expected-line-ending-format= |  | ||||||
|  |  | ||||||
| # Regexp for a line that is allowed to be longer than the limit. |  | ||||||
| ignore-long-lines=^\s*(# )?<?https?://\S+>?$ |  | ||||||
|  |  | ||||||
| # Number of spaces of indent required inside a hanging or continued line. |  | ||||||
| indent-after-paren=4 |  | ||||||
|  |  | ||||||
| # String used as indentation unit. This is usually "    " (4 spaces) or "\t" (1 |  | ||||||
| # tab). |  | ||||||
| indent-string='    ' |  | ||||||
|  |  | ||||||
| # Maximum number of characters on a single line. |  | ||||||
| max-line-length=100 |  | ||||||
|  |  | ||||||
| # Maximum number of lines in a module. |  | ||||||
| max-module-lines=1000 |  | ||||||
|  |  | ||||||
| # Allow the body of a class to be on the same line as the declaration if body |  | ||||||
| # contains single statement. |  | ||||||
| single-line-class-stmt=no |  | ||||||
|  |  | ||||||
| # Allow the body of an if to be on the same line as the test if there is no |  | ||||||
| # else. |  | ||||||
| single-line-if-stmt=no |  | ||||||
|  |  | ||||||
|  |  | ||||||
| [IMPORTS] |  | ||||||
|  |  | ||||||
| # List of modules that can be imported at any level, not just the top level |  | ||||||
| # one. |  | ||||||
| allow-any-import-level= |  | ||||||
|  |  | ||||||
| # Allow explicit reexports by alias from a package __init__. |  | ||||||
| allow-reexport-from-package=no |  | ||||||
|  |  | ||||||
| # Allow wildcard imports from modules that define __all__. |  | ||||||
| allow-wildcard-with-all=no |  | ||||||
|  |  | ||||||
| # Deprecated modules which should not be used, separated by a comma. |  | ||||||
| deprecated-modules= |  | ||||||
|  |  | ||||||
| # Output a graph (.gv or any supported image format) of external dependencies |  | ||||||
| # to the given file (report RP0402 must not be disabled). |  | ||||||
| ext-import-graph= |  | ||||||
|  |  | ||||||
| # Output a graph (.gv or any supported image format) of all (i.e. internal and |  | ||||||
| # external) dependencies to the given file (report RP0402 must not be |  | ||||||
| # disabled). |  | ||||||
| import-graph= |  | ||||||
|  |  | ||||||
| # Output a graph (.gv or any supported image format) of internal dependencies |  | ||||||
| # to the given file (report RP0402 must not be disabled). |  | ||||||
| int-import-graph= |  | ||||||
|  |  | ||||||
| # Force import order to recognize a module as part of the standard |  | ||||||
| # compatibility libraries. |  | ||||||
| known-standard-library= |  | ||||||
|  |  | ||||||
| # Force import order to recognize a module as part of a third party library. |  | ||||||
| known-third-party=enchant |  | ||||||
|  |  | ||||||
| # Couples of modules and preferred modules, separated by a comma. |  | ||||||
| preferred-modules= |  | ||||||
|  |  | ||||||
|  |  | ||||||
| [LOGGING] |  | ||||||
|  |  | ||||||
| # The type of string formatting that logging methods do. `old` means using % |  | ||||||
| # formatting, `new` is for `{}` formatting. |  | ||||||
| logging-format-style=old |  | ||||||
|  |  | ||||||
| # Logging modules to check that the string format arguments are in logging |  | ||||||
| # function parameter format. |  | ||||||
| logging-modules=logging |  | ||||||
|  |  | ||||||
|  |  | ||||||
| [MESSAGES CONTROL] |  | ||||||
|  |  | ||||||
| # Only show warnings with the listed confidence levels. Leave empty to show |  | ||||||
| # all. Valid levels: HIGH, CONTROL_FLOW, INFERENCE, INFERENCE_FAILURE, |  | ||||||
| # UNDEFINED. |  | ||||||
| confidence=HIGH, |  | ||||||
|            CONTROL_FLOW, |  | ||||||
|            INFERENCE, |  | ||||||
|            INFERENCE_FAILURE, |  | ||||||
|            UNDEFINED |  | ||||||
|  |  | ||||||
| # Disable the message, report, category or checker with the given id(s). You |  | ||||||
| # can either give multiple identifiers separated by comma (,) or put this |  | ||||||
| # option multiple times (only on the command line, not in the configuration |  | ||||||
| # file where it should appear only once). You can also use "--disable=all" to |  | ||||||
| # disable everything first and then re-enable specific checks. For example, if |  | ||||||
| # you want to run only the similarities checker, you can use "--disable=all |  | ||||||
| # --enable=similarities". If you want to run only the classes checker, but have |  | ||||||
| # no Warning level messages displayed, use "--disable=all --enable=classes |  | ||||||
| # --disable=W". |  | ||||||
| disable=raw-checker-failed, |  | ||||||
|         bad-inline-option, |  | ||||||
|         locally-disabled, |  | ||||||
|         file-ignored, |  | ||||||
|         suppressed-message, |  | ||||||
|         useless-suppression, |  | ||||||
|         deprecated-pragma, |  | ||||||
|         use-symbolic-message-instead, |  | ||||||
|         missing-module-docstring, |  | ||||||
|         line-too-long, |  | ||||||
|         no-name-in-module, |  | ||||||
|         import-outside-toplevel, |  | ||||||
|         invalid-name, |  | ||||||
|         raise-missing-from, |  | ||||||
|         wrong-import-order, |  | ||||||
|         too-few-public-methods, |  | ||||||
|         too-many-instance-attributes, |  | ||||||
|         broad-except, |  | ||||||
|         fixme, |  | ||||||
|         too-many-arguments, |  | ||||||
|         duplicate-code, |  | ||||||
|         cyclic-import, |  | ||||||
|  |  | ||||||
| # Enable the message, report, category or checker with the given id(s). You can |  | ||||||
| # either give multiple identifier separated by comma (,) or put this option |  | ||||||
| # multiple time (only on the command line, not in the configuration file where |  | ||||||
| # it should appear only once). See also the "--disable" option for examples. |  | ||||||
| enable=c-extension-no-member |  | ||||||
|  |  | ||||||
|  |  | ||||||
| [METHOD_ARGS] |  | ||||||
|  |  | ||||||
| # List of qualified names (i.e., library.method) which require a timeout |  | ||||||
| # parameter e.g. 'requests.api.get,requests.api.post' |  | ||||||
| timeout-methods=requests.api.delete,requests.api.get,requests.api.head,requests.api.options,requests.api.patch,requests.api.post,requests.api.put,requests.api.request |  | ||||||
|  |  | ||||||
|  |  | ||||||
| [MISCELLANEOUS] |  | ||||||
|  |  | ||||||
| # List of note tags to take in consideration, separated by a comma. |  | ||||||
| notes=FIXME, |  | ||||||
|       XXX, |  | ||||||
|       TODO |  | ||||||
|  |  | ||||||
| # Regular expression of note tags to take in consideration. |  | ||||||
| notes-rgx= |  | ||||||
|  |  | ||||||
|  |  | ||||||
| [REFACTORING] |  | ||||||
|  |  | ||||||
| # Maximum number of nested blocks for function / method body |  | ||||||
| max-nested-blocks=5 |  | ||||||
|  |  | ||||||
| # Complete name of functions that never returns. When checking for |  | ||||||
| # inconsistent-return-statements if a never returning function is called then |  | ||||||
| # it will be considered as an explicit return statement and no message will be |  | ||||||
| # printed. |  | ||||||
| never-returning-functions=sys.exit,argparse.parse_error |  | ||||||
|  |  | ||||||
|  |  | ||||||
| [REPORTS] |  | ||||||
|  |  | ||||||
| # Python expression which should return a score less than or equal to 10. You |  | ||||||
| # have access to the variables 'fatal', 'error', 'warning', 'refactor', |  | ||||||
| # 'convention', and 'info' which contain the number of messages in each |  | ||||||
| # category, as well as 'statement' which is the total number of statements |  | ||||||
| # analyzed. This score is used by the global evaluation report (RP0004). |  | ||||||
| evaluation=max(0, 0 if fatal else 10.0 - ((float(5 * error + warning + refactor + convention) / statement) * 10)) |  | ||||||
|  |  | ||||||
| # Template used to display messages. This is a python new-style format string |  | ||||||
| # used to format the message information. See doc for all details. |  | ||||||
| msg-template= |  | ||||||
|  |  | ||||||
| # Set the output format. Available formats are text, parseable, colorized, json |  | ||||||
| # and msvs (visual studio). You can also give a reporter class, e.g. |  | ||||||
| # mypackage.mymodule.MyReporterClass. |  | ||||||
| #output-format= |  | ||||||
|  |  | ||||||
| # Tells whether to display a full report or only the messages. |  | ||||||
| reports=no |  | ||||||
|  |  | ||||||
| # Activate the evaluation score. |  | ||||||
| score=yes |  | ||||||
|  |  | ||||||
|  |  | ||||||
| [SIMILARITIES] |  | ||||||
|  |  | ||||||
| # Comments are removed from the similarity computation |  | ||||||
| ignore-comments=yes |  | ||||||
|  |  | ||||||
| # Docstrings are removed from the similarity computation |  | ||||||
| ignore-docstrings=yes |  | ||||||
|  |  | ||||||
| # Imports are removed from the similarity computation |  | ||||||
| ignore-imports=yes |  | ||||||
|  |  | ||||||
| # Signatures are removed from the similarity computation |  | ||||||
| ignore-signatures=yes |  | ||||||
|  |  | ||||||
| # Minimum lines number of a similarity. |  | ||||||
| min-similarity-lines=4 |  | ||||||
|  |  | ||||||
|  |  | ||||||
| [SPELLING] |  | ||||||
|  |  | ||||||
| # Limits count of emitted suggestions for spelling mistakes. |  | ||||||
| max-spelling-suggestions=4 |  | ||||||
|  |  | ||||||
| # Spelling dictionary name. No available dictionaries : You need to install |  | ||||||
| # both the python package and the system dependency for enchant to work.. |  | ||||||
| spelling-dict= |  | ||||||
|  |  | ||||||
| # List of comma separated words that should be considered directives if they |  | ||||||
| # appear at the beginning of a comment and should not be checked. |  | ||||||
| spelling-ignore-comment-directives=fmt: on,fmt: off,noqa:,noqa,nosec,isort:skip,mypy: |  | ||||||
|  |  | ||||||
| # List of comma separated words that should not be checked. |  | ||||||
| spelling-ignore-words= |  | ||||||
|  |  | ||||||
| # A path to a file that contains the private dictionary; one word per line. |  | ||||||
| spelling-private-dict-file= |  | ||||||
|  |  | ||||||
| # Tells whether to store unknown words to the private dictionary (see the |  | ||||||
| # --spelling-private-dict-file option) instead of raising a message. |  | ||||||
| spelling-store-unknown-words=no |  | ||||||
|  |  | ||||||
|  |  | ||||||
| [STRING] |  | ||||||
|  |  | ||||||
| # This flag controls whether inconsistent-quotes generates a warning when the |  | ||||||
| # character used as a quote delimiter is used inconsistently within a module. |  | ||||||
| check-quote-consistency=no |  | ||||||
|  |  | ||||||
| # This flag controls whether the implicit-str-concat should generate a warning |  | ||||||
| # on implicit string concatenation in sequences defined over several lines. |  | ||||||
| check-str-concat-over-line-jumps=no |  | ||||||
|  |  | ||||||
|  |  | ||||||
| [TYPECHECK] |  | ||||||
|  |  | ||||||
| # List of decorators that produce context managers, such as |  | ||||||
| # contextlib.contextmanager. Add to this list to register other decorators that |  | ||||||
| # produce valid context managers. |  | ||||||
| contextmanager-decorators=contextlib.contextmanager |  | ||||||
|  |  | ||||||
| # List of members which are set dynamically and missed by pylint inference |  | ||||||
| # system, and so shouldn't trigger E1101 when accessed. Python regular |  | ||||||
| # expressions are accepted. |  | ||||||
| generated-members= |  | ||||||
|  |  | ||||||
| # Tells whether to warn about missing members when the owner of the attribute |  | ||||||
| # is inferred to be None. |  | ||||||
| ignore-none=yes |  | ||||||
|  |  | ||||||
| # This flag controls whether pylint should warn about no-member and similar |  | ||||||
| # checks whenever an opaque object is returned when inferring. The inference |  | ||||||
| # can return multiple potential results while evaluating a Python object, but |  | ||||||
| # some branches might not be evaluated, which results in partial inference. In |  | ||||||
| # that case, it might be useful to still emit no-member and other checks for |  | ||||||
| # the rest of the inferred objects. |  | ||||||
| ignore-on-opaque-inference=yes |  | ||||||
|  |  | ||||||
| # List of symbolic message names to ignore for Mixin members. |  | ||||||
| ignored-checks-for-mixins=no-member, |  | ||||||
|                           not-async-context-manager, |  | ||||||
|                           not-context-manager, |  | ||||||
|                           attribute-defined-outside-init |  | ||||||
|  |  | ||||||
| # List of class names for which member attributes should not be checked (useful |  | ||||||
| # for classes with dynamically set attributes). This supports the use of |  | ||||||
| # qualified names. |  | ||||||
| ignored-classes=optparse.Values,thread._local,_thread._local,argparse.Namespace |  | ||||||
|  |  | ||||||
| # Show a hint with possible names when a member name was not found. The aspect |  | ||||||
| # of finding the hint is based on edit distance. |  | ||||||
| missing-member-hint=yes |  | ||||||
|  |  | ||||||
| # The minimum edit distance a name should have in order to be considered a |  | ||||||
| # similar match for a missing member name. |  | ||||||
| missing-member-hint-distance=1 |  | ||||||
|  |  | ||||||
| # The total number of similar names that should be taken in consideration when |  | ||||||
| # showing a hint for a missing member. |  | ||||||
| missing-member-max-choices=1 |  | ||||||
|  |  | ||||||
| # Regex pattern to define which classes are considered mixins. |  | ||||||
| mixin-class-rgx=.*[Mm]ixin |  | ||||||
|  |  | ||||||
| # List of decorators that change the signature of a decorated function. |  | ||||||
| signature-mutators= |  | ||||||
|  |  | ||||||
|  |  | ||||||
| [VARIABLES] |  | ||||||
|  |  | ||||||
| # List of additional names supposed to be defined in builtins. Remember that |  | ||||||
| # you should avoid defining new builtins when possible. |  | ||||||
| additional-builtins= |  | ||||||
|  |  | ||||||
| # Tells whether unused global variables should be treated as a violation. |  | ||||||
| allow-global-unused-variables=yes |  | ||||||
|  |  | ||||||
| # List of names allowed to shadow builtins |  | ||||||
| allowed-redefined-builtins= |  | ||||||
|  |  | ||||||
| # List of strings which can identify a callback function by name. A callback |  | ||||||
| # name must start or end with one of those strings. |  | ||||||
| callbacks=cb_, |  | ||||||
|           _cb |  | ||||||
|  |  | ||||||
| # A regular expression matching the name of dummy variables (i.e. expected to |  | ||||||
| # not be used). |  | ||||||
| dummy-variables-rgx=_+$|(_[a-zA-Z0-9_]*[a-zA-Z0-9]+?$)|dummy|^ignored_|^unused_ |  | ||||||
|  |  | ||||||
| # Argument names that match this expression will be ignored. |  | ||||||
| ignored-argument-names=_.*|^ignored_|^unused_ |  | ||||||
|  |  | ||||||
| # Tells whether we should check for unused import in __init__ files. |  | ||||||
| init-import=no |  | ||||||
|  |  | ||||||
| # List of qualified module names which can have objects that can redefine |  | ||||||
| # builtins. |  | ||||||
| redefining-builtins-modules=six.moves,past.builtins,future.builtins,builtins,io |  | ||||||
|  | |||||||
| @ -6,7 +6,7 @@ formats: | |||||||
| build: | build: | ||||||
|   os: ubuntu-20.04 |   os: ubuntu-20.04 | ||||||
|   tools: |   tools: | ||||||
|     python: "3.11" |     python: "3.10" | ||||||
|  |  | ||||||
| sphinx: | sphinx: | ||||||
|   builder: html |   builder: html | ||||||
| @ -21,3 +21,4 @@ python: | |||||||
|         - docs |         - docs | ||||||
|         - s3 |         - s3 | ||||||
|         - web |         - web | ||||||
|  |   system_packages: true | ||||||
|  | |||||||
| @ -34,7 +34,7 @@ Again, the most checks can be performed by `make check` command, though some add | |||||||
|         do foo. With very very very long |         do foo. With very very very long | ||||||
|         docstring |         docstring | ||||||
|    |    | ||||||
|         Notes: |         Note: | ||||||
|             Very important note about this function |             Very important note about this function | ||||||
|    |    | ||||||
|         Args: |         Args: | ||||||
| @ -46,7 +46,7 @@ Again, the most checks can be performed by `make check` command, though some add | |||||||
|             int: result |             int: result | ||||||
|    |    | ||||||
|         Raises: |         Raises: | ||||||
|             RuntimeError: a local function error occurs |             RuntimeException: a local function error occurs | ||||||
|    |    | ||||||
|         Examples: |         Examples: | ||||||
|             Very informative example how to use this function, e.g.:: |             Very informative example how to use this function, e.g.:: | ||||||
| @ -103,9 +103,6 @@ Again, the most checks can be performed by `make check` command, though some add | |||||||
|         @property |         @property | ||||||
|         def property(self) -> Any: ... |         def property(self) -> Any: ... | ||||||
|  |  | ||||||
|         @cached_property |  | ||||||
|         def property_cached(self) -> Any: ...  # cached property has to be treated as normal one |  | ||||||
|  |  | ||||||
|         @classmethod |         @classmethod | ||||||
|         def class_method(cls) -> Self: ... |         def class_method(cls) -> Self: ... | ||||||
|  |  | ||||||
| @ -130,12 +127,6 @@ Again, the most checks can be performed by `make check` command, though some add | |||||||
| * Configuration interactions must go through `ahriman.core.configuration.Configuration` class instance. | * Configuration interactions must go through `ahriman.core.configuration.Configuration` class instance. | ||||||
| * In case if class load requires some actions, it is recommended to create class method which can be used for class instantiating. | * In case if class load requires some actions, it is recommended to create class method which can be used for class instantiating. | ||||||
| * The code must follow the exception safety, unless it is explicitly asked by end user. It means that most exceptions must be handled and printed to log, no other actions must be done (e.g. raising another exception). | * The code must follow the exception safety, unless it is explicitly asked by end user. It means that most exceptions must be handled and printed to log, no other actions must be done (e.g. raising another exception). | ||||||
| * Exceptions without parameters should be raised without parentheses, e.g.: |  | ||||||
|  |  | ||||||
|     ```python |  | ||||||
|     raise RuntimeError |  | ||||||
|     ``` |  | ||||||
|  |  | ||||||
| * For the external command `ahriman.core.util.check_output` function must be used. | * For the external command `ahriman.core.util.check_output` function must be used. | ||||||
| * Every temporary file/directory must be removed at the end of processing, no matter what. The `tempfile` module provides good ways to do it. | * Every temporary file/directory must be removed at the end of processing, no matter what. The `tempfile` module provides good ways to do it. | ||||||
| * Import order must be the following: | * Import order must be the following: | ||||||
| @ -164,7 +155,7 @@ Again, the most checks can be performed by `make check` command, though some add | |||||||
| * One file should define only one class, exception is class satellites in case if file length remains less than 400 lines. | * One file should define only one class, exception is class satellites in case if file length remains less than 400 lines. | ||||||
| * It is possible to create file which contains some functions (e.g. `ahriman.core.util`), but in this case you would need to define `__all__` attribute. | * It is possible to create file which contains some functions (e.g. `ahriman.core.util`), but in this case you would need to define `__all__` attribute. | ||||||
| * The file size mentioned above must be applicable in general. In case of big classes consider splitting them into traits. Note, however, that `pylint` includes comments and docstrings into counter, thus you need to check file size by other tools. | * The file size mentioned above must be applicable in general. In case of big classes consider splitting them into traits. Note, however, that `pylint` includes comments and docstrings into counter, thus you need to check file size by other tools. | ||||||
| * No global variable is allowed outside of `ahriman` module. `ahriman.core.context` is also special case. | * No global variable is allowed outside of `ahriman.version` module. `ahriman.core.context` is also special case. | ||||||
| * Single quotes are not allowed. The reason behind this restriction is the fact that docstrings must be written by using double quotes only, and we would like to make style consistent. | * Single quotes are not allowed. The reason behind this restriction is the fact that docstrings must be written by using double quotes only, and we would like to make style consistent. | ||||||
| * If your class writes anything to log, the `ahriman.core.log.LazyLogging` trait must be used. | * If your class writes anything to log, the `ahriman.core.log.LazyLogging` trait must be used. | ||||||
| * Web API methods must be documented by using `aiohttp_apispec` library. Schema testing mostly should be implemented in related view class tests. Recommended example for documentation (excluding comments): | * Web API methods must be documented by using `aiohttp_apispec` library. Schema testing mostly should be implemented in related view class tests. Recommended example for documentation (excluding comments): | ||||||
|  | |||||||
							
								
								
									
										18
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										18
									
								
								Dockerfile
									
									
									
									
									
								
							| @ -6,12 +6,11 @@ ENV AHRIMAN_DEBUG="" | |||||||
| ENV AHRIMAN_FORCE_ROOT="" | ENV AHRIMAN_FORCE_ROOT="" | ||||||
| ENV AHRIMAN_HOST="0.0.0.0" | ENV AHRIMAN_HOST="0.0.0.0" | ||||||
| ENV AHRIMAN_MULTILIB="yes" | ENV AHRIMAN_MULTILIB="yes" | ||||||
| ENV AHRIMAN_OUTPUT="" | ENV AHRIMAN_OUTPUT="syslog" | ||||||
| ENV AHRIMAN_PACKAGER="ahriman bot <ahriman@example.com>" | ENV AHRIMAN_PACKAGER="ahriman bot <ahriman@example.com>" | ||||||
| ENV AHRIMAN_PACMAN_MIRROR="" | ENV AHRIMAN_PACMAN_MIRROR="" | ||||||
| ENV AHRIMAN_PORT="" | ENV AHRIMAN_PORT="" | ||||||
| ENV AHRIMAN_REPOSITORY="aur-clone" | ENV AHRIMAN_REPOSITORY="aur-clone" | ||||||
| ENV AHRIMAN_REPOSITORY_SERVER="" |  | ||||||
| ENV AHRIMAN_REPOSITORY_ROOT="/var/lib/ahriman/ahriman" | ENV AHRIMAN_REPOSITORY_ROOT="/var/lib/ahriman/ahriman" | ||||||
| ENV AHRIMAN_UNIX_SOCKET="" | ENV AHRIMAN_UNIX_SOCKET="" | ||||||
| ENV AHRIMAN_USER="ahriman" | ENV AHRIMAN_USER="ahriman" | ||||||
| @ -29,27 +28,26 @@ RUN useradd -m -d "/home/build" -s "/usr/bin/nologin" build && \ | |||||||
| COPY "docker/install-aur-package.sh" "/usr/local/bin/install-aur-package" | COPY "docker/install-aur-package.sh" "/usr/local/bin/install-aur-package" | ||||||
| ## install package dependencies | ## install package dependencies | ||||||
| ## darcs is not installed by reasons, because it requires a lot haskell packages which dramatically increase image size | ## darcs is not installed by reasons, because it requires a lot haskell packages which dramatically increase image size | ||||||
| RUN pacman -Sy --noconfirm --asdeps devtools git pyalpm python-cerberus python-inflection python-passlib python-requests python-srcinfo && \ | RUN pacman --noconfirm -Sy devtools-git-poc git pyalpm python-cerberus python-inflection python-passlib python-requests python-srcinfo && \ | ||||||
|     pacman -Sy --noconfirm --asdeps python-build python-flit python-installer python-wheel && \ |     pacman --noconfirm -Sy python-build python-installer python-wheel && \ | ||||||
|     pacman -Sy --noconfirm --asdeps breezy mercurial python-aiohttp python-aiohttp-cors python-boto3 python-cryptography python-jinja python-requests-unixsocket python-systemd rsync subversion && \ |     pacman --noconfirm -Sy breezy mercurial python-aiohttp python-aiohttp-cors python-boto3 python-cryptography python-jinja python-requests-unixsocket rsync subversion && \ | ||||||
|     runuser -u build -- install-aur-package python-aioauth-client python-aiohttp-apispec-git python-aiohttp-jinja2  \ |     runuser -u build -- install-aur-package python-aioauth-client python-aiohttp-apispec-git python-aiohttp-jinja2  \ | ||||||
|                                             python-aiohttp-debugtoolbar python-aiohttp-session python-aiohttp-security |                                             python-aiohttp-debugtoolbar python-aiohttp-session python-aiohttp-security | ||||||
|  |  | ||||||
|  | # cleanup unused | ||||||
|  | RUN find "/var/cache/pacman/pkg" -type f -delete | ||||||
|  |  | ||||||
| # install ahriman | # install ahriman | ||||||
| ## copy tree | ## copy tree | ||||||
| COPY --chown=build . "/home/build/ahriman" | COPY --chown=build . "/home/build/ahriman" | ||||||
| ## create package archive and install it | ## create package archive and install it | ||||||
| RUN cd "/home/build/ahriman" && \ | RUN cd "/home/build/ahriman" && \ | ||||||
|     make VERSION=$(python -c "from src.ahriman import __version__; print(__version__)") archlinux && \ |     make VERSION=$(python -c "from src.ahriman.version import __version__; print(__version__)") archlinux && \ | ||||||
|     cp ./*-src.tar.xz "package/archlinux" && \ |     cp ./*-src.tar.xz "package/archlinux" && \ | ||||||
|     cd "package/archlinux" && \ |     cd "package/archlinux" && \ | ||||||
|     runuser -u build -- makepkg --noconfirm --install --skipchecksums && \ |     runuser -u build -- makepkg --noconfirm --install --skipchecksums && \ | ||||||
|     cd / && rm -r "/home/build/ahriman" |     cd / && rm -r "/home/build/ahriman" | ||||||
|  |  | ||||||
| # cleanup unused |  | ||||||
| RUN find "/var/cache/pacman/pkg" -type f -delete |  | ||||||
| RUN pacman -Qdtq | pacman -Rscn --noconfirm - |  | ||||||
|  |  | ||||||
| VOLUME ["/var/lib/ahriman"] | VOLUME ["/var/lib/ahriman"] | ||||||
|  |  | ||||||
| # minimal runtime ahriman setup | # minimal runtime ahriman setup | ||||||
|  | |||||||
							
								
								
									
										6
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										6
									
								
								Makefile
									
									
									
									
									
								
							| @ -3,7 +3,7 @@ | |||||||
|  |  | ||||||
| PROJECT := ahriman | PROJECT := ahriman | ||||||
|  |  | ||||||
| FILES := AUTHORS CONTRIBUTING.md COPYING Makefile README.md SECURITY.md docs package pyproject.toml src tox.ini web.png | FILES := AUTHORS CONTRIBUTING.md COPYING Makefile README.md SECURITY.md docs package src setup.py tox.ini web.png | ||||||
| TARGET_FILES := $(addprefix $(PROJECT)/, $(FILES)) | TARGET_FILES := $(addprefix $(PROJECT)/, $(FILES)) | ||||||
| IGNORE_FILES := package/archlinux src/.mypy_cache | IGNORE_FILES := package/archlinux src/.mypy_cache | ||||||
|  |  | ||||||
| @ -38,7 +38,7 @@ html: specification | |||||||
| 	tox -e docs-html | 	tox -e docs-html | ||||||
|  |  | ||||||
| push: specification archlinux | push: specification archlinux | ||||||
| 	git add package/archlinux/PKGBUILD src/ahriman/__init__.py docs/ahriman-architecture.svg package/share/man/man1/ahriman.1 package/share/bash-completion/completions/_ahriman package/share/zsh/site-functions/_ahriman | 	git add package/archlinux/PKGBUILD src/ahriman/version.py docs/ahriman-architecture.svg docs/ahriman.1 docs/completions/ | ||||||
| 	git commit -m "Release $(VERSION)" | 	git commit -m "Release $(VERSION)" | ||||||
| 	git tag "$(VERSION)" | 	git tag "$(VERSION)" | ||||||
| 	git push | 	git push | ||||||
| @ -56,4 +56,4 @@ version: | |||||||
| ifndef VERSION | ifndef VERSION | ||||||
| 	$(error VERSION is required, but not set) | 	$(error VERSION is required, but not set) | ||||||
| endif | endif | ||||||
| 	sed -i 's/^__version__ = .*/__version__ = "$(VERSION)"/' src/ahriman/__init__.py | 	sed -i 's/^__version__ = .*/__version__ = "$(VERSION)"/' src/ahriman/version.py | ||||||
|  | |||||||
| @ -1,8 +1,8 @@ | |||||||
| # ArcH linux ReposItory MANager | # ArcH linux ReposItory MANager | ||||||
|  |  | ||||||
| [](https://github.com/arcan1s/ahriman/actions/workflows/run-tests.yml) | [](https://github.com/arcan1s/ahriman/actions/workflows/run-tests.yml) | ||||||
| [](https://github.com/arcan1s/ahriman/actions/workflows/run-setup.yml) | [](https://github.com/arcan1s/ahriman/actions/workflows/run-setup.yml) | ||||||
| [](https://hub.docker.com/r/arcan1s/ahriman) | [](https://hub.docker.com/r/arcan1s/ahriman) | ||||||
| [](https://www.codefactor.io/repository/github/arcan1s/ahriman) | [](https://www.codefactor.io/repository/github/arcan1s/ahriman) | ||||||
| [](https://ahriman.readthedocs.io/?badge=latest) | [](https://ahriman.readthedocs.io/?badge=latest) | ||||||
|  |  | ||||||
| @ -16,8 +16,7 @@ Wrapper for managing custom repository inspired by [repo-scripts](https://github | |||||||
| * VCS packages support. | * VCS packages support. | ||||||
| * Official repository support. | * Official repository support. | ||||||
| * Ability to patch AUR packages and even create package from local PKGBUILDs. | * Ability to patch AUR packages and even create package from local PKGBUILDs. | ||||||
| * Various rebuild options with ability to automatically bump package version. | * Sign support with gpg (repository, package, per package settings). | ||||||
| * Sign support with gpg (repository, package), multiple packagers support. |  | ||||||
| * Triggers for repository updates, e.g. synchronization to remote services (rsync, s3 and github) and report generation (email, html, telegram). | * Triggers for repository updates, e.g. synchronization to remote services (rsync, s3 and github) and report generation (email, html, telegram). | ||||||
| * Repository status interface with optional authorization and control options: | * Repository status interface with optional authorization and control options: | ||||||
|  |  | ||||||
|  | |||||||
| @ -15,11 +15,15 @@ database = $AHRIMAN_REPOSITORY_ROOT/ahriman.db | |||||||
| host = $AHRIMAN_HOST | host = $AHRIMAN_HOST | ||||||
|  |  | ||||||
| EOF | EOF | ||||||
|  | sed -i "s|handlers = syslog_handler|handlers = ${AHRIMAN_OUTPUT}_handler|g" "/etc/ahriman.ini.d/logging.ini" | ||||||
|  |  | ||||||
| AHRIMAN_DEFAULT_ARGS=("--architecture" "$AHRIMAN_ARCHITECTURE") | AHRIMAN_DEFAULT_ARGS=("--architecture" "$AHRIMAN_ARCHITECTURE") | ||||||
| AHRIMAN_DEFAULT_ARGS+=("--repository" "$AHRIMAN_REPOSITORY") | if [[ "$AHRIMAN_OUTPUT" == "syslog" ]]; then | ||||||
| if [ -n "$AHRIMAN_OUTPUT" ]; then |     if [ ! -e "/dev/log" ]; then | ||||||
|     AHRIMAN_DEFAULT_ARGS+=("--log-handler" "$AHRIMAN_OUTPUT") |         # by default ahriman uses syslog which is not available inside container | ||||||
|  |         # to make noise less we force quiet mode in case if /dev/log was not mounted | ||||||
|  |         AHRIMAN_DEFAULT_ARGS+=("--quiet") | ||||||
|  |     fi | ||||||
| fi | fi | ||||||
|  |  | ||||||
| # create repository root inside the [[mounted]] directory and set correct ownership | # create repository root inside the [[mounted]] directory and set correct ownership | ||||||
| @ -32,8 +36,8 @@ AHRIMAN_GNUPG_HOME="$(getent passwd "$AHRIMAN_USER" | cut -d : -f 6)/.gnupg" | |||||||
| chown "$AHRIMAN_USER":"$AHRIMAN_USER" "$AHRIMAN_GNUPG_HOME" | chown "$AHRIMAN_USER":"$AHRIMAN_USER" "$AHRIMAN_GNUPG_HOME" | ||||||
|  |  | ||||||
| # run built-in setup command | # run built-in setup command | ||||||
| AHRIMAN_SETUP_ARGS=("--build-as-user" "$AHRIMAN_USER") |  | ||||||
| AHRIMAN_SETUP_ARGS+=("--packager" "$AHRIMAN_PACKAGER") | AHRIMAN_SETUP_ARGS+=("--packager" "$AHRIMAN_PACKAGER") | ||||||
|  | AHRIMAN_SETUP_ARGS+=("--repository" "$AHRIMAN_REPOSITORY") | ||||||
| if [ -z "$AHRIMAN_MULTILIB" ]; then | if [ -z "$AHRIMAN_MULTILIB" ]; then | ||||||
|     AHRIMAN_SETUP_ARGS+=("--no-multilib") |     AHRIMAN_SETUP_ARGS+=("--no-multilib") | ||||||
| fi | fi | ||||||
| @ -43,9 +47,6 @@ fi | |||||||
| if [ -n "$AHRIMAN_PORT" ]; then | if [ -n "$AHRIMAN_PORT" ]; then | ||||||
|     AHRIMAN_SETUP_ARGS+=("--web-port" "$AHRIMAN_PORT") |     AHRIMAN_SETUP_ARGS+=("--web-port" "$AHRIMAN_PORT") | ||||||
| fi | fi | ||||||
| if [ -n "$AHRIMAN_REPOSITORY_SERVER" ]; then |  | ||||||
|     AHRIMAN_SETUP_ARGS+=("--server" "$AHRIMAN_REPOSITORY_SERVER") |  | ||||||
| fi |  | ||||||
| if [ -n "$AHRIMAN_UNIX_SOCKET" ]; then | if [ -n "$AHRIMAN_UNIX_SOCKET" ]; then | ||||||
|     AHRIMAN_SETUP_ARGS+=("--web-unix-socket" "$AHRIMAN_UNIX_SOCKET") |     AHRIMAN_SETUP_ARGS+=("--web-unix-socket" "$AHRIMAN_UNIX_SOCKET") | ||||||
| fi | fi | ||||||
| @ -61,8 +62,8 @@ systemd-machine-id-setup &> /dev/null | |||||||
| # otherwise we prepend executable by sudo command | # otherwise we prepend executable by sudo command | ||||||
| if [ -n "$AHRIMAN_FORCE_ROOT" ]; then | if [ -n "$AHRIMAN_FORCE_ROOT" ]; then | ||||||
|     AHRIMAN_EXECUTABLE=("ahriman") |     AHRIMAN_EXECUTABLE=("ahriman") | ||||||
| elif ahriman help-commands-unsafe -- "$@" &> /dev/null; then | elif ahriman help-commands-unsafe --command="$*" &> /dev/null; then | ||||||
|     AHRIMAN_EXECUTABLE=("sudo" "-E" "-u" "$AHRIMAN_USER" "--" "ahriman") |     AHRIMAN_EXECUTABLE=("sudo" "-u" "$AHRIMAN_USER" "--" "ahriman") | ||||||
| else | else | ||||||
|     AHRIMAN_EXECUTABLE=("ahriman") |     AHRIMAN_EXECUTABLE=("ahriman") | ||||||
| fi | fi | ||||||
|  | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| Before Width: | Height: | Size: 839 KiB After Width: | Height: | Size: 750 KiB | 
| @ -1,9 +1,9 @@ | |||||||
| .TH AHRIMAN "1" "2023\-08\-26" "ahriman" "Generated Python Manual" | .TH AHRIMAN "1" "2023\-04\-09" "ahriman" "Generated Python Manual" | ||||||
| .SH NAME | .SH NAME | ||||||
| ahriman | ahriman | ||||||
| .SH SYNOPSIS | .SH SYNOPSIS | ||||||
| .B ahriman | .B ahriman | ||||||
| [-h] [-a ARCHITECTURE] [-c CONFIGURATION] [--force] [-l LOCK] [--log-handler {console,syslog,journald}] [--report | --no-report] [-q] [--unsafe] [--wait-timeout WAIT_TIMEOUT] [-V] {aur-search,search,help,help-commands-unsafe,help-updates,help-version,version,package-add,add,package-update,package-remove,remove,package-status,status,package-status-remove,package-status-update,status-update,patch-add,patch-list,patch-remove,patch-set-add,repo-backup,repo-check,check,repo-create-keyring,repo-create-mirrorlist,repo-daemon,daemon,repo-rebuild,rebuild,repo-remove-unknown,remove-unknown,repo-report,report,repo-restore,repo-sign,sign,repo-status-update,repo-sync,sync,repo-tree,repo-triggers,repo-update,update,service-clean,clean,repo-clean,service-config,config,repo-config,service-config-validate,config-validate,repo-config-validate,service-key-import,key-import,service-setup,init,repo-init,repo-setup,setup,service-shell,shell,user-add,user-list,user-remove,web} ... | [-h] [-a ARCHITECTURE] [-c CONFIGURATION] [--force] [-l LOCK] [--report | --no-report] [-q] [--unsafe] [-V] {aur-search,search,help,help-commands-unsafe,help-updates,help-version,version,package-add,add,package-update,package-remove,remove,package-status,status,package-status-remove,package-status-update,status-update,patch-add,patch-list,patch-remove,patch-set-add,repo-backup,repo-check,check,repo-daemon,daemon,repo-rebuild,rebuild,repo-remove-unknown,remove-unknown,repo-report,report,repo-restore,repo-sign,sign,repo-status-update,repo-sync,sync,repo-tree,repo-triggers,repo-update,update,service-clean,clean,repo-clean,service-config,config,repo-config,service-config-validate,config-validate,repo-config-validate,service-key-import,key-import,service-setup,init,repo-init,repo-setup,setup,service-shell,shell,user-add,user-list,user-remove,web} ... | ||||||
| .SH DESCRIPTION | .SH DESCRIPTION | ||||||
| ArcH linux ReposItory MANager | ArcH linux ReposItory MANager | ||||||
| 
 | 
 | ||||||
| @ -24,13 +24,9 @@ force run, remove file lock | |||||||
| \fB\-l\fR \fI\,LOCK\/\fR, \fB\-\-lock\fR \fI\,LOCK\/\fR | \fB\-l\fR \fI\,LOCK\/\fR, \fB\-\-lock\fR \fI\,LOCK\/\fR | ||||||
| lock file | lock file | ||||||
| 
 | 
 | ||||||
| .TP |  | ||||||
| \fB\-\-log\-handler\fR \fI\,{console,syslog,journald}\/\fR |  | ||||||
| explicit log handler specification. If none set, the handler will be guessed from environment |  | ||||||
| 
 |  | ||||||
| .TP | .TP | ||||||
| \fB\-\-report\fR, \fB\-\-no\-report\fR | \fB\-\-report\fR, \fB\-\-no\-report\fR | ||||||
| force enable or disable reporting to web service | force enable or disable reporting to web service (default: True) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-q\fR, \fB\-\-quiet\fR | \fB\-q\fR, \fB\-\-quiet\fR | ||||||
| @ -40,11 +36,6 @@ force disable any logging | |||||||
| \fB\-\-unsafe\fR | \fB\-\-unsafe\fR | ||||||
| allow to run ahriman as non\-ahriman user. Some actions might be unavailable | allow to run ahriman as non\-ahriman user. Some actions might be unavailable | ||||||
| 
 | 
 | ||||||
| .TP |  | ||||||
| \fB\-\-wait\-timeout\fR \fI\,WAIT_TIMEOUT\/\fR |  | ||||||
| wait for lock to be free. Negative value will lead to immediate application run even if there is lock file. In case of |  | ||||||
| zero value, the application will wait infinitely |  | ||||||
| 
 |  | ||||||
| .TP | .TP | ||||||
| \fB\-V\fR, \fB\-\-version\fR | \fB\-V\fR, \fB\-\-version\fR | ||||||
| show program's version number and exit | show program's version number and exit | ||||||
| @ -100,12 +91,6 @@ backup repository data | |||||||
| \fBahriman\fR \fI\,repo\-check\/\fR | \fBahriman\fR \fI\,repo\-check\/\fR | ||||||
| check for updates | check for updates | ||||||
| .TP | .TP | ||||||
| \fBahriman\fR \fI\,repo\-create\-keyring\/\fR |  | ||||||
| create keyring package |  | ||||||
| .TP |  | ||||||
| \fBahriman\fR \fI\,repo\-create\-mirrorlist\/\fR |  | ||||||
| create mirrorlist package |  | ||||||
| .TP |  | ||||||
| \fBahriman\fR \fI\,repo\-daemon\/\fR | \fBahriman\fR \fI\,repo\-daemon\/\fR | ||||||
| run application as daemon | run application as daemon | ||||||
| .TP | .TP | ||||||
| @ -187,7 +172,7 @@ return non\-zero exit status if result is empty | |||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-info\fR, \fB\-\-no\-info\fR | \fB\-\-info\fR, \fB\-\-no\-info\fR | ||||||
| show additional package information | show additional package information (default: False) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-sort\-by\fR \fI\,{description,first_submitted,id,last_modified,maintainer,name,num_votes,out_of_date,package_base,package_base_id,popularity,repository,submitter,url,url_path,version}\/\fR | \fB\-\-sort\-by\fR \fI\,{description,first_submitted,id,last_modified,maintainer,name,num_votes,out_of_date,package_base,package_base_id,popularity,repository,submitter,url,url_path,version}\/\fR | ||||||
| @ -204,12 +189,13 @@ show help message for application or command and exit | |||||||
| show help message for specific command | show help message for specific command | ||||||
| 
 | 
 | ||||||
| .SH COMMAND \fI\,'ahriman help\-commands\-unsafe'\/\fR | .SH COMMAND \fI\,'ahriman help\-commands\-unsafe'\/\fR | ||||||
| usage: ahriman help\-commands\-unsafe [\-h] [command ...] | usage: ahriman help\-commands\-unsafe [\-h] [\-\-command COMMAND] | ||||||
| 
 | 
 | ||||||
| list unsafe commands as defined in default args | list unsafe commands as defined in default args | ||||||
| 
 | 
 | ||||||
|  | .SH OPTIONS \fI\,'ahriman help\-commands\-unsafe'\/\fR | ||||||
| .TP | .TP | ||||||
| \fBcommand\fR | \fB\-\-command\fR \fI\,COMMAND\/\fR | ||||||
| instead of showing commands, just test command line for unsafe subcommand and return 0 in case if command is safe and 1 | instead of showing commands, just test command line for unsafe subcommand and return 0 in case if command is safe and 1 | ||||||
| otherwise | otherwise | ||||||
| 
 | 
 | ||||||
| @ -229,8 +215,8 @@ usage: ahriman help\-version [\-h] | |||||||
| print application and its dependencies versions | print application and its dependencies versions | ||||||
| 
 | 
 | ||||||
| .SH COMMAND \fI\,'ahriman package\-add'\/\fR | .SH COMMAND \fI\,'ahriman package\-add'\/\fR | ||||||
| usage: ahriman package\-add [\-h] [\-\-dependencies | \-\-no\-dependencies] [\-e] [\-\-increment | \-\-no\-increment] [\-n] [\-y] | usage: ahriman package\-add [\-h] [\-\-dependencies | \-\-no\-dependencies] [\-e] [\-n] [\-y] | ||||||
|                            [\-s {auto,archive,aur,directory,local,remote,repository}] [\-u USERNAME] |                            [\-s {auto,archive,aur,directory,local,remote,repository}] | ||||||
|                            package [package ...] |                            package [package ...] | ||||||
| 
 | 
 | ||||||
| add existing or new package to the build queue | add existing or new package to the build queue | ||||||
| @ -242,16 +228,12 @@ package source (base name, path to local files, remote URL) | |||||||
| .SH OPTIONS \fI\,'ahriman package\-add'\/\fR | .SH OPTIONS \fI\,'ahriman package\-add'\/\fR | ||||||
| .TP | .TP | ||||||
| \fB\-\-dependencies\fR, \fB\-\-no\-dependencies\fR | \fB\-\-dependencies\fR, \fB\-\-no\-dependencies\fR | ||||||
| process missing package dependencies | process missing package dependencies (default: True) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-e\fR, \fB\-\-exit\-code\fR | \fB\-e\fR, \fB\-\-exit\-code\fR | ||||||
| return non\-zero exit status if result is empty | return non\-zero exit status if result is empty | ||||||
| 
 | 
 | ||||||
| .TP |  | ||||||
| \fB\-\-increment\fR, \fB\-\-no\-increment\fR |  | ||||||
| increment package release (pkgrel) version on duplicate |  | ||||||
| 
 |  | ||||||
| .TP | .TP | ||||||
| \fB\-n\fR, \fB\-\-now\fR | \fB\-n\fR, \fB\-\-now\fR | ||||||
| run update function after | run update function after | ||||||
| @ -264,10 +246,6 @@ download fresh package databases from the mirror before actions, \-yy to force r | |||||||
| \fB\-s\fR \fI\,{auto,archive,aur,directory,local,remote,repository}\/\fR, \fB\-\-source\fR \fI\,{auto,archive,aur,directory,local,remote,repository}\/\fR | \fB\-s\fR \fI\,{auto,archive,aur,directory,local,remote,repository}\/\fR, \fB\-\-source\fR \fI\,{auto,archive,aur,directory,local,remote,repository}\/\fR | ||||||
| explicitly specify the package source for this command | explicitly specify the package source for this command | ||||||
| 
 | 
 | ||||||
| .TP |  | ||||||
| \fB\-u\fR \fI\,USERNAME\/\fR, \fB\-\-username\fR \fI\,USERNAME\/\fR |  | ||||||
| build as user |  | ||||||
| 
 |  | ||||||
| .SH COMMAND \fI\,'ahriman package\-remove'\/\fR | .SH COMMAND \fI\,'ahriman package\-remove'\/\fR | ||||||
| usage: ahriman package\-remove [\-h] package [package ...] | usage: ahriman package\-remove [\-h] package [package ...] | ||||||
| 
 | 
 | ||||||
| @ -298,7 +276,7 @@ return non\-zero exit status if result is empty | |||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-info\fR, \fB\-\-no\-info\fR | \fB\-\-info\fR, \fB\-\-no\-info\fR | ||||||
| show additional package information | show additional package information (default: False) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-s\fR \fI\,{unknown,pending,building,failed,success}\/\fR, \fB\-\-status\fR \fI\,{unknown,pending,building,failed,success}\/\fR | \fB\-s\fR \fI\,{unknown,pending,building,failed,success}\/\fR, \fB\-\-status\fR \fI\,{unknown,pending,building,failed,success}\/\fR | ||||||
| @ -416,22 +394,12 @@ return non\-zero exit status if result is empty | |||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-vcs\fR, \fB\-\-no\-vcs\fR | \fB\-\-vcs\fR, \fB\-\-no\-vcs\fR | ||||||
| fetch actual version of VCS packages | fetch actual version of VCS packages (default: True) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-y\fR, \fB\-\-refresh\fR | \fB\-y\fR, \fB\-\-refresh\fR | ||||||
| download fresh package databases from the mirror before actions, \-yy to force refresh even if up to date | download fresh package databases from the mirror before actions, \-yy to force refresh even if up to date | ||||||
| 
 | 
 | ||||||
| .SH COMMAND \fI\,'ahriman repo\-create\-keyring'\/\fR |  | ||||||
| usage: ahriman repo\-create\-keyring [\-h] |  | ||||||
| 
 |  | ||||||
| create package which contains list of trusted keys as set by configuration. Note, that this action will only create package, the package itself has to be built manually |  | ||||||
| 
 |  | ||||||
| .SH COMMAND \fI\,'ahriman repo\-create\-mirrorlist'\/\fR |  | ||||||
| usage: ahriman repo\-create\-mirrorlist [\-h] |  | ||||||
| 
 |  | ||||||
| create package which contains list of available mirrors as set by configuration. Note, that this action will only create package, the package itself has to be built manually |  | ||||||
| 
 |  | ||||||
| .SH COMMAND \fI\,'ahriman repo\-daemon'\/\fR | .SH COMMAND \fI\,'ahriman repo\-daemon'\/\fR | ||||||
| usage: ahriman repo\-daemon [\-h] [\-i INTERVAL] [\-\-aur | \-\-no\-aur] [\-\-dependencies | \-\-no\-dependencies] | usage: ahriman repo\-daemon [\-h] [\-i INTERVAL] [\-\-aur | \-\-no\-aur] [\-\-dependencies | \-\-no\-dependencies] | ||||||
|                            [\-\-local | \-\-no\-local] [\-\-manual | \-\-no\-manual] [\-\-vcs | \-\-no\-vcs] [\-y] |                            [\-\-local | \-\-no\-local] [\-\-manual | \-\-no\-manual] [\-\-vcs | \-\-no\-vcs] [\-y] | ||||||
| @ -445,31 +413,30 @@ interval between runs in seconds | |||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-aur\fR, \fB\-\-no\-aur\fR | \fB\-\-aur\fR, \fB\-\-no\-aur\fR | ||||||
| enable or disable checking for AUR updates | enable or disable checking for AUR updates (default: True) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-dependencies\fR, \fB\-\-no\-dependencies\fR | \fB\-\-dependencies\fR, \fB\-\-no\-dependencies\fR | ||||||
| process missing package dependencies | process missing package dependencies (default: True) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-local\fR, \fB\-\-no\-local\fR | \fB\-\-local\fR, \fB\-\-no\-local\fR | ||||||
| enable or disable checking of local packages for updates | enable or disable checking of local packages for updates (default: True) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-manual\fR, \fB\-\-no\-manual\fR | \fB\-\-manual\fR, \fB\-\-no\-manual\fR | ||||||
| include or exclude manual updates | include or exclude manual updates (default: True) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-vcs\fR, \fB\-\-no\-vcs\fR | \fB\-\-vcs\fR, \fB\-\-no\-vcs\fR | ||||||
| fetch actual version of VCS packages | fetch actual version of VCS packages (default: True) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-y\fR, \fB\-\-refresh\fR | \fB\-y\fR, \fB\-\-refresh\fR | ||||||
| download fresh package databases from the mirror before actions, \-yy to force refresh even if up to date | download fresh package databases from the mirror before actions, \-yy to force refresh even if up to date | ||||||
| 
 | 
 | ||||||
| .SH COMMAND \fI\,'ahriman repo\-rebuild'\/\fR | .SH COMMAND \fI\,'ahriman repo\-rebuild'\/\fR | ||||||
| usage: ahriman repo\-rebuild [\-h] [\-\-depends\-on DEPENDS_ON] [\-\-dry\-run] [\-\-from\-database] [\-\-increment | \-\-no\-increment] | usage: ahriman repo\-rebuild [\-h] [\-\-depends\-on DEPENDS_ON] [\-\-dry\-run] [\-\-from\-database] [\-e] | ||||||
|                             [\-e] [\-s {unknown,pending,building,failed,success}] [\-u USERNAME] |  | ||||||
| 
 | 
 | ||||||
| force rebuild whole repository | force rebuild whole repository | ||||||
| 
 | 
 | ||||||
| @ -488,22 +455,10 @@ read packages from database instead of filesystem. This feature in particular is | |||||||
| restore repository from another repository instance. Note, however, that in order to restore packages you need to have | restore repository from another repository instance. Note, however, that in order to restore packages you need to have | ||||||
| original ahriman instance run with web service and have run repo\-update at least once. | original ahriman instance run with web service and have run repo\-update at least once. | ||||||
| 
 | 
 | ||||||
| .TP |  | ||||||
| \fB\-\-increment\fR, \fB\-\-no\-increment\fR |  | ||||||
| increment package release (pkgrel) on duplicate |  | ||||||
| 
 |  | ||||||
| .TP | .TP | ||||||
| \fB\-e\fR, \fB\-\-exit\-code\fR | \fB\-e\fR, \fB\-\-exit\-code\fR | ||||||
| return non\-zero exit status if result is empty | return non\-zero exit status if result is empty | ||||||
| 
 | 
 | ||||||
| .TP |  | ||||||
| \fB\-s\fR \fI\,{unknown,pending,building,failed,success}\/\fR, \fB\-\-status\fR \fI\,{unknown,pending,building,failed,success}\/\fR |  | ||||||
| filter packages by status. Requires \-\-from\-database to be set |  | ||||||
| 
 |  | ||||||
| .TP |  | ||||||
| \fB\-u\fR \fI\,USERNAME\/\fR, \fB\-\-username\fR \fI\,USERNAME\/\fR |  | ||||||
| build as user |  | ||||||
| 
 |  | ||||||
| .SH COMMAND \fI\,'ahriman repo\-remove\-unknown'\/\fR | .SH COMMAND \fI\,'ahriman repo\-remove\-unknown'\/\fR | ||||||
| usage: ahriman repo\-remove\-unknown [\-h] [\-\-dry\-run] | usage: ahriman repo\-remove\-unknown [\-h] [\-\-dry\-run] | ||||||
| 
 | 
 | ||||||
| @ -558,15 +513,10 @@ usage: ahriman repo\-sync [\-h] | |||||||
| sync repository files to remote server according to current settings | sync repository files to remote server according to current settings | ||||||
| 
 | 
 | ||||||
| .SH COMMAND \fI\,'ahriman repo\-tree'\/\fR | .SH COMMAND \fI\,'ahriman repo\-tree'\/\fR | ||||||
| usage: ahriman repo\-tree [\-h] [\-p PARTITIONS] | usage: ahriman repo\-tree [\-h] | ||||||
| 
 | 
 | ||||||
| dump repository tree based on packages dependencies | dump repository tree based on packages dependencies | ||||||
| 
 | 
 | ||||||
| .SH OPTIONS \fI\,'ahriman repo\-tree'\/\fR |  | ||||||
| .TP |  | ||||||
| \fB\-p\fR \fI\,PARTITIONS\/\fR, \fB\-\-partitions\fR \fI\,PARTITIONS\/\fR |  | ||||||
| also divide packages by independent partitions |  | ||||||
| 
 |  | ||||||
| .SH COMMAND \fI\,'ahriman repo\-triggers'\/\fR | .SH COMMAND \fI\,'ahriman repo\-triggers'\/\fR | ||||||
| usage: ahriman repo\-triggers [\-h] [trigger ...] | usage: ahriman repo\-triggers [\-h] [trigger ...] | ||||||
| 
 | 
 | ||||||
| @ -578,8 +528,7 @@ instead of running all triggers as set by configuration, just process specified | |||||||
| 
 | 
 | ||||||
| .SH COMMAND \fI\,'ahriman repo\-update'\/\fR | .SH COMMAND \fI\,'ahriman repo\-update'\/\fR | ||||||
| usage: ahriman repo\-update [\-h] [\-\-aur | \-\-no\-aur] [\-\-dependencies | \-\-no\-dependencies] [\-\-dry\-run] [\-e] | usage: ahriman repo\-update [\-h] [\-\-aur | \-\-no\-aur] [\-\-dependencies | \-\-no\-dependencies] [\-\-dry\-run] [\-e] | ||||||
|                            [\-\-increment | \-\-no\-increment] [\-\-local | \-\-no\-local] [\-\-manual | \-\-no\-manual] [\-u USERNAME] |                            [\-\-local | \-\-no\-local] [\-\-manual | \-\-no\-manual] [\-\-vcs | \-\-no\-vcs] [\-y] | ||||||
|                            [\-\-vcs | \-\-no\-vcs] [\-y] |  | ||||||
|                            [package ...] |                            [package ...] | ||||||
| 
 | 
 | ||||||
| check for packages updates and run build process if requested | check for packages updates and run build process if requested | ||||||
| @ -591,11 +540,11 @@ filter check by package base | |||||||
| .SH OPTIONS \fI\,'ahriman repo\-update'\/\fR | .SH OPTIONS \fI\,'ahriman repo\-update'\/\fR | ||||||
| .TP | .TP | ||||||
| \fB\-\-aur\fR, \fB\-\-no\-aur\fR | \fB\-\-aur\fR, \fB\-\-no\-aur\fR | ||||||
| enable or disable checking for AUR updates | enable or disable checking for AUR updates (default: True) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-dependencies\fR, \fB\-\-no\-dependencies\fR | \fB\-\-dependencies\fR, \fB\-\-no\-dependencies\fR | ||||||
| process missing package dependencies | process missing package dependencies (default: True) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-dry\-run\fR | \fB\-\-dry\-run\fR | ||||||
| @ -605,25 +554,17 @@ just perform check for updates, same as check command | |||||||
| \fB\-e\fR, \fB\-\-exit\-code\fR | \fB\-e\fR, \fB\-\-exit\-code\fR | ||||||
| return non\-zero exit status if result is empty | return non\-zero exit status if result is empty | ||||||
| 
 | 
 | ||||||
| .TP |  | ||||||
| \fB\-\-increment\fR, \fB\-\-no\-increment\fR |  | ||||||
| increment package release (pkgrel) on duplicate |  | ||||||
| 
 |  | ||||||
| .TP | .TP | ||||||
| \fB\-\-local\fR, \fB\-\-no\-local\fR | \fB\-\-local\fR, \fB\-\-no\-local\fR | ||||||
| enable or disable checking of local packages for updates | enable or disable checking of local packages for updates (default: True) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-manual\fR, \fB\-\-no\-manual\fR | \fB\-\-manual\fR, \fB\-\-no\-manual\fR | ||||||
| include or exclude manual updates | include or exclude manual updates (default: True) | ||||||
| 
 |  | ||||||
| .TP |  | ||||||
| \fB\-u\fR \fI\,USERNAME\/\fR, \fB\-\-username\fR \fI\,USERNAME\/\fR |  | ||||||
| build as user |  | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-vcs\fR, \fB\-\-no\-vcs\fR | \fB\-\-vcs\fR, \fB\-\-no\-vcs\fR | ||||||
| fetch actual version of VCS packages | fetch actual version of VCS packages (default: True) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-y\fR, \fB\-\-refresh\fR | \fB\-y\fR, \fB\-\-refresh\fR | ||||||
| @ -638,23 +579,23 @@ remove local caches | |||||||
| .SH OPTIONS \fI\,'ahriman service\-clean'\/\fR | .SH OPTIONS \fI\,'ahriman service\-clean'\/\fR | ||||||
| .TP | .TP | ||||||
| \fB\-\-cache\fR, \fB\-\-no\-cache\fR | \fB\-\-cache\fR, \fB\-\-no\-cache\fR | ||||||
| clear directory with package caches | clear directory with package caches (default: False) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-chroot\fR, \fB\-\-no\-chroot\fR | \fB\-\-chroot\fR, \fB\-\-no\-chroot\fR | ||||||
| clear build chroot | clear build chroot (default: False) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-manual\fR, \fB\-\-no\-manual\fR | \fB\-\-manual\fR, \fB\-\-no\-manual\fR | ||||||
| clear manually added packages queue | clear manually added packages queue (default: False) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-packages\fR, \fB\-\-no\-packages\fR | \fB\-\-packages\fR, \fB\-\-no\-packages\fR | ||||||
| clear directory with built packages | clear directory with built packages (default: False) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-pacman\fR, \fB\-\-no\-pacman\fR | \fB\-\-pacman\fR, \fB\-\-no\-pacman\fR | ||||||
| clear directory with pacman local database cache | clear directory with pacman local database cache (default: False) | ||||||
| 
 | 
 | ||||||
| .SH COMMAND \fI\,'ahriman service\-config'\/\fR | .SH COMMAND \fI\,'ahriman service\-config'\/\fR | ||||||
| usage: ahriman service\-config [\-h] [\-\-secure | \-\-no\-secure] | usage: ahriman service\-config [\-h] [\-\-secure | \-\-no\-secure] | ||||||
| @ -664,7 +605,7 @@ dump configuration for the specified architecture | |||||||
| .SH OPTIONS \fI\,'ahriman service\-config'\/\fR | .SH OPTIONS \fI\,'ahriman service\-config'\/\fR | ||||||
| .TP | .TP | ||||||
| \fB\-\-secure\fR, \fB\-\-no\-secure\fR | \fB\-\-secure\fR, \fB\-\-no\-secure\fR | ||||||
| hide passwords and secrets from output | hide passwords and secrets from output (default: True) | ||||||
| 
 | 
 | ||||||
| .SH COMMAND \fI\,'ahriman service\-config\-validate'\/\fR | .SH COMMAND \fI\,'ahriman service\-config\-validate'\/\fR | ||||||
| usage: ahriman service\-config\-validate [\-h] [\-e] | usage: ahriman service\-config\-validate [\-h] [\-e] | ||||||
| @ -692,10 +633,9 @@ key server for key import | |||||||
| 
 | 
 | ||||||
| .SH COMMAND \fI\,'ahriman service\-setup'\/\fR | .SH COMMAND \fI\,'ahriman service\-setup'\/\fR | ||||||
| usage: ahriman service\-setup [\-h] [\-\-build\-as\-user BUILD_AS_USER] [\-\-build\-command BUILD_COMMAND] | usage: ahriman service\-setup [\-h] [\-\-build\-as\-user BUILD_AS_USER] [\-\-build\-command BUILD_COMMAND] | ||||||
|                              [\-\-from\-configuration FROM_CONFIGURATION] [\-\-generate\-salt | \-\-no\-generate\-salt] |                              [\-\-from\-configuration FROM_CONFIGURATION] [\-\-makeflags\-jobs | \-\-no\-makeflags\-jobs] | ||||||
|                              [\-\-makeflags\-jobs | \-\-no\-makeflags\-jobs] [\-\-mirror MIRROR] [\-\-multilib | \-\-no\-multilib] |                              [\-\-mirror MIRROR] [\-\-multilib | \-\-no\-multilib] \-\-packager PACKAGER \-\-repository REPOSITORY | ||||||
|                              \-\-packager PACKAGER \-\-repository REPOSITORY [\-\-server SERVER] [\-\-sign\-key SIGN_KEY] |                              [\-\-sign\-key SIGN_KEY] [\-\-sign\-target {disabled,packages,repository}] [\-\-web\-port WEB_PORT] | ||||||
|                              [\-\-sign\-target {disabled,packages,repository}] [\-\-web\-port WEB_PORT] |  | ||||||
|                              [\-\-web\-unix\-socket WEB_UNIX_SOCKET] |                              [\-\-web\-unix\-socket WEB_UNIX_SOCKET] | ||||||
| 
 | 
 | ||||||
| create initial service configuration, requires root | create initial service configuration, requires root | ||||||
| @ -713,13 +653,9 @@ build command prefix | |||||||
| \fB\-\-from\-configuration\fR \fI\,FROM_CONFIGURATION\/\fR | \fB\-\-from\-configuration\fR \fI\,FROM_CONFIGURATION\/\fR | ||||||
| path to default devtools pacman configuration | path to default devtools pacman configuration | ||||||
| 
 | 
 | ||||||
| .TP |  | ||||||
| \fB\-\-generate\-salt\fR, \fB\-\-no\-generate\-salt\fR |  | ||||||
| generate salt for user passwords |  | ||||||
| 
 |  | ||||||
| .TP | .TP | ||||||
| \fB\-\-makeflags\-jobs\fR, \fB\-\-no\-makeflags\-jobs\fR | \fB\-\-makeflags\-jobs\fR, \fB\-\-no\-makeflags\-jobs\fR | ||||||
| append MAKEFLAGS variable with parallelism set to number of cores | append MAKEFLAGS variable with parallelism set to number of cores (default: True) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-mirror\fR \fI\,MIRROR\/\fR | \fB\-\-mirror\fR \fI\,MIRROR\/\fR | ||||||
| @ -727,7 +663,7 @@ use the specified explicitly mirror instead of including mirrorlist | |||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-multilib\fR, \fB\-\-no\-multilib\fR | \fB\-\-multilib\fR, \fB\-\-no\-multilib\fR | ||||||
| add or do not multilib repository | add or do not multilib repository (default: True) | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fB\-\-packager\fR \fI\,PACKAGER\/\fR | \fB\-\-packager\fR \fI\,PACKAGER\/\fR | ||||||
| @ -737,10 +673,6 @@ packager name and email | |||||||
| \fB\-\-repository\fR \fI\,REPOSITORY\/\fR | \fB\-\-repository\fR \fI\,REPOSITORY\/\fR | ||||||
| repository name | repository name | ||||||
| 
 | 
 | ||||||
| .TP |  | ||||||
| \fB\-\-server\fR \fI\,SERVER\/\fR |  | ||||||
| server to be used for devtools. If none set, local files will be used |  | ||||||
| 
 |  | ||||||
| .TP | .TP | ||||||
| \fB\-\-sign\-key\fR \fI\,SIGN_KEY\/\fR | \fB\-\-sign\-key\fR \fI\,SIGN_KEY\/\fR | ||||||
| sign key id | sign key id | ||||||
| @ -760,15 +692,14 @@ path to unix socket used for interprocess communications | |||||||
| .SH COMMAND \fI\,'ahriman service\-shell'\/\fR | .SH COMMAND \fI\,'ahriman service\-shell'\/\fR | ||||||
| usage: ahriman service\-shell [\-h] [code] | usage: ahriman service\-shell [\-h] [code] | ||||||
| 
 | 
 | ||||||
| drop into python shell | drop into python shell while having created application | ||||||
| 
 | 
 | ||||||
| .TP | .TP | ||||||
| \fBcode\fR | \fBcode\fR | ||||||
| instead of dropping into shell, just execute the specified code | instead of dropping into shell, just execute the specified code | ||||||
| 
 | 
 | ||||||
| .SH COMMAND \fI\,'ahriman user\-add'\/\fR | .SH COMMAND \fI\,'ahriman user\-add'\/\fR | ||||||
| usage: ahriman user\-add [\-h] [\-\-key KEY] [\-\-packager PACKAGER] [\-p PASSWORD] [\-r {unauthorized,read,reporter,full}] | usage: ahriman user\-add [\-h] [\-p PASSWORD] [\-r {unauthorized,read,reporter,full}] [\-s] username | ||||||
|                         username |  | ||||||
| 
 | 
 | ||||||
| update user for web services with the given password and role. In case if password was not entered it will be asked interactively | update user for web services with the given password and role. In case if password was not entered it will be asked interactively | ||||||
| 
 | 
 | ||||||
| @ -777,14 +708,6 @@ update user for web services with the given password and role. In case if passwo | |||||||
| username for web service | username for web service | ||||||
| 
 | 
 | ||||||
| .SH OPTIONS \fI\,'ahriman user\-add'\/\fR | .SH OPTIONS \fI\,'ahriman user\-add'\/\fR | ||||||
| .TP |  | ||||||
| \fB\-\-key\fR \fI\,KEY\/\fR |  | ||||||
| optional PGP key used by this user. The private key must be imported |  | ||||||
| 
 |  | ||||||
| .TP |  | ||||||
| \fB\-\-packager\fR \fI\,PACKAGER\/\fR |  | ||||||
| optional packager id used for build process in form of `Name Surname <mail@example.com>` |  | ||||||
| 
 |  | ||||||
| .TP | .TP | ||||||
| \fB\-p\fR \fI\,PASSWORD\/\fR, \fB\-\-password\fR \fI\,PASSWORD\/\fR | \fB\-p\fR \fI\,PASSWORD\/\fR, \fB\-\-password\fR \fI\,PASSWORD\/\fR | ||||||
| user password. Blank password will be treated as empty password, which is in particular must be used for OAuth2 | user password. Blank password will be treated as empty password, which is in particular must be used for OAuth2 | ||||||
| @ -794,6 +717,10 @@ authorization type. | |||||||
| \fB\-r\fR \fI\,{unauthorized,read,reporter,full}\/\fR, \fB\-\-role\fR \fI\,{unauthorized,read,reporter,full}\/\fR | \fB\-r\fR \fI\,{unauthorized,read,reporter,full}\/\fR, \fB\-\-role\fR \fI\,{unauthorized,read,reporter,full}\/\fR | ||||||
| user access level | user access level | ||||||
| 
 | 
 | ||||||
|  | .TP | ||||||
|  | \fB\-s\fR, \fB\-\-secure\fR | ||||||
|  | set file permissions to user\-only | ||||||
|  | 
 | ||||||
| .SH COMMAND \fI\,'ahriman user\-list'\/\fR | .SH COMMAND \fI\,'ahriman user\-list'\/\fR | ||||||
| usage: ahriman user\-list [\-h] [\-e] [\-r {unauthorized,read,reporter,full}] [username] | usage: ahriman user\-list [\-h] [\-e] [\-r {unauthorized,read,reporter,full}] [username] | ||||||
| 
 | 
 | ||||||
| @ -829,7 +756,7 @@ start web server | |||||||
| .SH COMMENTS | .SH COMMENTS | ||||||
| Argument list can also be read from file by using @ prefix. | Argument list can also be read from file by using @ prefix. | ||||||
| 
 | 
 | ||||||
| .SH AUTHOR | .SH AUTHORS | ||||||
| .nf | .nf | ||||||
| ahriman team | ahriman team | ||||||
| .fi | .fi | ||||||
| @ -20,14 +20,6 @@ ahriman.core.configuration.schema module | |||||||
|    :no-undoc-members: |    :no-undoc-members: | ||||||
|    :show-inheritance: |    :show-inheritance: | ||||||
|  |  | ||||||
| ahriman.core.configuration.shell\_interpolator module |  | ||||||
| ----------------------------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.core.configuration.shell_interpolator |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| ahriman.core.configuration.validator module | ahriman.core.configuration.validator module | ||||||
| ------------------------------------------- | ------------------------------------------- | ||||||
|  |  | ||||||
|  | |||||||
| @ -60,38 +60,6 @@ ahriman.core.database.migrations.m006\_packages\_architecture\_required module | |||||||
|    :no-undoc-members: |    :no-undoc-members: | ||||||
|    :show-inheritance: |    :show-inheritance: | ||||||
|  |  | ||||||
| ahriman.core.database.migrations.m007\_check\_depends module |  | ||||||
| ------------------------------------------------------------ |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.core.database.migrations.m007_check_depends |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| ahriman.core.database.migrations.m008\_packagers module |  | ||||||
| ------------------------------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.core.database.migrations.m008_packagers |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| ahriman.core.database.migrations.m009\_local\_source module |  | ||||||
| ----------------------------------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.core.database.migrations.m009_local_source |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| ahriman.core.database.migrations.m010\_version\_based\_logs\_removal module |  | ||||||
| --------------------------------------------------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.core.database.migrations.m010_version_based_logs_removal |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| Module contents | Module contents | ||||||
| --------------- | --------------- | ||||||
|  |  | ||||||
|  | |||||||
| @ -20,14 +20,6 @@ ahriman.core.formatters.build\_printer module | |||||||
|    :no-undoc-members: |    :no-undoc-members: | ||||||
|    :show-inheritance: |    :show-inheritance: | ||||||
|  |  | ||||||
| ahriman.core.formatters.configuration\_paths\_printer module |  | ||||||
| ------------------------------------------------------------ |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.core.formatters.configuration_paths_printer |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| ahriman.core.formatters.configuration\_printer module | ahriman.core.formatters.configuration\_printer module | ||||||
| ----------------------------------------------------- | ----------------------------------------------------- | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,21 +0,0 @@ | |||||||
| ahriman.core.http package |  | ||||||
| ========================= |  | ||||||
|  |  | ||||||
| Submodules |  | ||||||
| ---------- |  | ||||||
|  |  | ||||||
| ahriman.core.http.sync\_http\_client module |  | ||||||
| ------------------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.core.http.sync_http_client |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| Module contents |  | ||||||
| --------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.core.http |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
| @ -20,14 +20,6 @@ ahriman.core.log.http\_log\_handler module | |||||||
|    :no-undoc-members: |    :no-undoc-members: | ||||||
|    :show-inheritance: |    :show-inheritance: | ||||||
|  |  | ||||||
| ahriman.core.log.journal\_handler module |  | ||||||
| ---------------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.core.log.journal_handler |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| ahriman.core.log.lazy\_logging module | ahriman.core.log.lazy\_logging module | ||||||
| ------------------------------------- | ------------------------------------- | ||||||
|  |  | ||||||
|  | |||||||
| @ -36,14 +36,6 @@ ahriman.core.report.jinja\_template module | |||||||
|    :no-undoc-members: |    :no-undoc-members: | ||||||
|    :show-inheritance: |    :show-inheritance: | ||||||
|  |  | ||||||
| ahriman.core.report.remote\_call module |  | ||||||
| --------------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.core.report.remote_call |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| ahriman.core.report.report module | ahriman.core.report.report module | ||||||
| --------------------------------- | --------------------------------- | ||||||
|  |  | ||||||
|  | |||||||
| @ -14,13 +14,11 @@ Subpackages | |||||||
|    ahriman.core.database |    ahriman.core.database | ||||||
|    ahriman.core.formatters |    ahriman.core.formatters | ||||||
|    ahriman.core.gitremote |    ahriman.core.gitremote | ||||||
|    ahriman.core.http |  | ||||||
|    ahriman.core.log |    ahriman.core.log | ||||||
|    ahriman.core.report |    ahriman.core.report | ||||||
|    ahriman.core.repository |    ahriman.core.repository | ||||||
|    ahriman.core.sign |    ahriman.core.sign | ||||||
|    ahriman.core.status |    ahriman.core.status | ||||||
|    ahriman.core.support |  | ||||||
|    ahriman.core.triggers |    ahriman.core.triggers | ||||||
|    ahriman.core.upload |    ahriman.core.upload | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,37 +0,0 @@ | |||||||
| ahriman.core.support.pkgbuild package |  | ||||||
| ===================================== |  | ||||||
|  |  | ||||||
| Submodules |  | ||||||
| ---------- |  | ||||||
|  |  | ||||||
| ahriman.core.support.pkgbuild.keyring\_generator module |  | ||||||
| ------------------------------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.core.support.pkgbuild.keyring_generator |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| ahriman.core.support.pkgbuild.mirrorlist\_generator module |  | ||||||
| ---------------------------------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.core.support.pkgbuild.mirrorlist_generator |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| ahriman.core.support.pkgbuild.pkgbuild\_generator module |  | ||||||
| -------------------------------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.core.support.pkgbuild.pkgbuild_generator |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| Module contents |  | ||||||
| --------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.core.support.pkgbuild |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
| @ -1,45 +0,0 @@ | |||||||
| ahriman.core.support package |  | ||||||
| ============================ |  | ||||||
|  |  | ||||||
| Subpackages |  | ||||||
| ----------- |  | ||||||
|  |  | ||||||
| .. toctree:: |  | ||||||
|    :maxdepth: 4 |  | ||||||
|  |  | ||||||
|    ahriman.core.support.pkgbuild |  | ||||||
|  |  | ||||||
| Submodules |  | ||||||
| ---------- |  | ||||||
|  |  | ||||||
| ahriman.core.support.keyring\_trigger module |  | ||||||
| -------------------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.core.support.keyring_trigger |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| ahriman.core.support.mirrorlist\_trigger module |  | ||||||
| ----------------------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.core.support.mirrorlist_trigger |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| ahriman.core.support.package\_creator module |  | ||||||
| -------------------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.core.support.package_creator |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| Module contents |  | ||||||
| --------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.core.support |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
| @ -20,14 +20,6 @@ ahriman.core.upload.http\_upload module | |||||||
|    :no-undoc-members: |    :no-undoc-members: | ||||||
|    :show-inheritance: |    :show-inheritance: | ||||||
|  |  | ||||||
| ahriman.core.upload.remote\_service module |  | ||||||
| ------------------------------------------ |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.core.upload.remote_service |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| ahriman.core.upload.rsync module | ahriman.core.upload.rsync module | ||||||
| -------------------------------- | -------------------------------- | ||||||
|  |  | ||||||
|  | |||||||
| @ -60,14 +60,6 @@ ahriman.models.internal\_status module | |||||||
|    :no-undoc-members: |    :no-undoc-members: | ||||||
|    :show-inheritance: |    :show-inheritance: | ||||||
|  |  | ||||||
| ahriman.models.log\_handler module |  | ||||||
| ---------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.models.log_handler |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| ahriman.models.log\_record\_id module | ahriman.models.log\_record\_id module | ||||||
| ------------------------------------- | ------------------------------------- | ||||||
|  |  | ||||||
| @ -116,22 +108,6 @@ ahriman.models.package\_source module | |||||||
|    :no-undoc-members: |    :no-undoc-members: | ||||||
|    :show-inheritance: |    :show-inheritance: | ||||||
|  |  | ||||||
| ahriman.models.packagers module |  | ||||||
| ------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.models.packagers |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| ahriman.models.pacman\_synchronization module |  | ||||||
| --------------------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.models.pacman_synchronization |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| ahriman.models.pkgbuild\_patch module | ahriman.models.pkgbuild\_patch module | ||||||
| ------------------------------------- | ------------------------------------- | ||||||
|  |  | ||||||
| @ -220,14 +196,6 @@ ahriman.models.user\_access module | |||||||
|    :no-undoc-members: |    :no-undoc-members: | ||||||
|    :show-inheritance: |    :show-inheritance: | ||||||
|  |  | ||||||
| ahriman.models.waiter module |  | ||||||
| ---------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.models.waiter |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| Module contents | Module contents | ||||||
| --------------- | --------------- | ||||||
|  |  | ||||||
|  | |||||||
| @ -12,6 +12,17 @@ Subpackages | |||||||
|    ahriman.models |    ahriman.models | ||||||
|    ahriman.web |    ahriman.web | ||||||
|  |  | ||||||
|  | Submodules | ||||||
|  | ---------- | ||||||
|  |  | ||||||
|  | ahriman.version module | ||||||
|  | ---------------------- | ||||||
|  |  | ||||||
|  | .. automodule:: ahriman.version | ||||||
|  |    :members: | ||||||
|  |    :no-undoc-members: | ||||||
|  |    :show-inheritance: | ||||||
|  |  | ||||||
| Module contents | Module contents | ||||||
| --------------- | --------------- | ||||||
|  |  | ||||||
|  | |||||||
| @ -36,14 +36,6 @@ ahriman.web.schemas.error\_schema module | |||||||
|    :no-undoc-members: |    :no-undoc-members: | ||||||
|    :show-inheritance: |    :show-inheritance: | ||||||
|  |  | ||||||
| ahriman.web.schemas.file\_schema module |  | ||||||
| --------------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.web.schemas.file_schema |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| ahriman.web.schemas.internal\_status\_schema module | ahriman.web.schemas.internal\_status\_schema module | ||||||
| --------------------------------------------------- | --------------------------------------------------- | ||||||
|  |  | ||||||
| @ -140,22 +132,6 @@ ahriman.web.schemas.pgp\_key\_schema module | |||||||
|    :no-undoc-members: |    :no-undoc-members: | ||||||
|    :show-inheritance: |    :show-inheritance: | ||||||
|  |  | ||||||
| ahriman.web.schemas.process\_id\_schema module |  | ||||||
| ---------------------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.web.schemas.process_id_schema |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| ahriman.web.schemas.process\_schema module |  | ||||||
| ------------------------------------------ |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.web.schemas.process_schema |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| ahriman.web.schemas.remote\_schema module | ahriman.web.schemas.remote\_schema module | ||||||
| ----------------------------------------- | ----------------------------------------- | ||||||
|  |  | ||||||
| @ -180,14 +156,6 @@ ahriman.web.schemas.status\_schema module | |||||||
|    :no-undoc-members: |    :no-undoc-members: | ||||||
|    :show-inheritance: |    :show-inheritance: | ||||||
|  |  | ||||||
| ahriman.web.schemas.update\_flags\_schema module |  | ||||||
| ------------------------------------------------ |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.web.schemas.update_flags_schema |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| Module contents | Module contents | ||||||
| --------------- | --------------- | ||||||
|  |  | ||||||
|  | |||||||
| @ -20,14 +20,6 @@ ahriman.web.views.service.pgp module | |||||||
|    :no-undoc-members: |    :no-undoc-members: | ||||||
|    :show-inheritance: |    :show-inheritance: | ||||||
|  |  | ||||||
| ahriman.web.views.service.process module |  | ||||||
| ---------------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.web.views.service.process |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| ahriman.web.views.service.rebuild module | ahriman.web.views.service.rebuild module | ||||||
| ---------------------------------------- | ---------------------------------------- | ||||||
|  |  | ||||||
| @ -68,14 +60,6 @@ ahriman.web.views.service.update module | |||||||
|    :no-undoc-members: |    :no-undoc-members: | ||||||
|    :show-inheritance: |    :show-inheritance: | ||||||
|  |  | ||||||
| ahriman.web.views.service.upload module |  | ||||||
| --------------------------------------- |  | ||||||
|  |  | ||||||
| .. automodule:: ahriman.web.views.service.upload |  | ||||||
|    :members: |  | ||||||
|    :no-undoc-members: |  | ||||||
|    :show-inheritance: |  | ||||||
|  |  | ||||||
| Module contents | Module contents | ||||||
| --------------- | --------------- | ||||||
|  |  | ||||||
|  | |||||||
| @ -33,22 +33,20 @@ This package contains everything required for the most of application actions an | |||||||
| * ``ahriman.core.alpm`` package controls pacman related functions. It provides wrappers for ``pyalpm`` library and safe calls for repository tools (``repo-add`` and ``repo-remove``). Also this package contains ``ahriman.core.alpm.remote`` package which provides wrapper for remote sources (e.g. AUR RPC and official repositories RPC). | * ``ahriman.core.alpm`` package controls pacman related functions. It provides wrappers for ``pyalpm`` library and safe calls for repository tools (``repo-add`` and ``repo-remove``). Also this package contains ``ahriman.core.alpm.remote`` package which provides wrapper for remote sources (e.g. AUR RPC and official repositories RPC). | ||||||
| * ``ahriman.core.auth`` package provides classes for authorization methods used by web mostly. Base class is ``ahriman.core.auth.Auth`` which must be called by ``load`` method. | * ``ahriman.core.auth`` package provides classes for authorization methods used by web mostly. Base class is ``ahriman.core.auth.Auth`` which must be called by ``load`` method. | ||||||
| * ``ahriman.core.build_tools`` is a package which provides wrapper for ``devtools`` commands. | * ``ahriman.core.build_tools`` is a package which provides wrapper for ``devtools`` commands. | ||||||
| * ``ahriman.core.configuration`` contains extension for standard ``configparser`` library and some validation related classes. |  | ||||||
| * ``ahriman.core.database`` is everything including data and schema migrations for database. | * ``ahriman.core.database`` is everything including data and schema migrations for database. | ||||||
| * ``ahriman.core.formatters`` package provides ``Printer`` sub-classes for printing data (e.g. package properties) to stdout which are used by some handlers. | * ``ahriman.core.formatters`` package provides ``Printer`` sub-classes for printing data (e.g. package properties) to stdout which are used by some handlers. | ||||||
| * ``ahriman.core.gitremote`` is a package with remote PKGBUILD triggers. Should not be called directly. | * ``ahriman.core.gitremote`` is a package with remote PKGBUILD triggers. Should not be called directly. | ||||||
| * ``ahriman.core.http`` package provides HTTP clients which can be later used by other classes. |  | ||||||
| * ``ahriman.core.log`` is a log utils package. It includes logger loader class, custom HTTP based logger and access logger for HTTP services with additional filters. | * ``ahriman.core.log`` is a log utils package. It includes logger loader class, custom HTTP based logger and access logger for HTTP services with additional filters. | ||||||
| * ``ahriman.core.report`` is a package with reporting triggers. Should not be called directly. | * ``ahriman.core.report`` is a package with reporting triggers. Should not be called directly. | ||||||
| * ``ahriman.core.repository`` contains several traits and base repository (``ahriman.core.repository.Repository`` class) implementation. | * ``ahriman.core.repository`` contains several traits and base repository (``ahriman.core.repository.Repository`` class) implementation. | ||||||
| * ``ahriman.core.sign`` package provides sign feature (only gpg calls are available). | * ``ahriman.core.sign`` package provides sign feature (only gpg calls are available). | ||||||
| * ``ahriman.core.status`` contains helpers and watcher class which are required for web application. Reporter must be initialized by using ``ahriman.core.status.client.Client.load`` method. | * ``ahriman.core.status`` contains helpers and watcher class which are required for web application. Reporter must be initialized by using ``ahriman.core.status.client.Client.load`` method. | ||||||
| * ``ahriman.core.support`` provides plugins for support packages (mirrorlist and keyring) generation. |  | ||||||
| * ``ahriman.core.triggers`` package contains base trigger classes. Classes from this package must be imported in order to implement user extensions. In fact, ``ahriman.core.report`` and ``ahriman.core.upload`` use this package. | * ``ahriman.core.triggers`` package contains base trigger classes. Classes from this package must be imported in order to implement user extensions. In fact, ``ahriman.core.report`` and ``ahriman.core.upload`` use this package. | ||||||
| * ``ahriman.core.upload`` package provides sync feature, should not be called directly. | * ``ahriman.core.upload`` package provides sync feature, should not be called directly. | ||||||
|  |  | ||||||
| This package also provides some generic functions and classes which may be used by other packages: | This package also provides some generic functions and classes which may be used by other packages: | ||||||
|  |  | ||||||
|  | * ``ahriman.core.configuration.Configuration`` is an extension for standard ``configparser`` library. | ||||||
| * ``ahriman.core.exceptions`` provides custom exceptions. | * ``ahriman.core.exceptions`` provides custom exceptions. | ||||||
| * ``ahriman.core.spawn.Spawn`` is a tool which can spawn another ``ahriman`` process. This feature is used by web application. | * ``ahriman.core.spawn.Spawn`` is a tool which can spawn another ``ahriman`` process. This feature is used by web application. | ||||||
| * ``ahriman.core.tree`` is a dependency tree implementation. | * ``ahriman.core.tree`` is a dependency tree implementation. | ||||||
| @ -64,23 +62,20 @@ It provides models for any other part of application. Unlike ``ahriman.core`` pa | |||||||
| Web application. It is important that this package is isolated from any other to allow it to be optional feature (i.e. dependencies which are required by the package are optional). | Web application. It is important that this package is isolated from any other to allow it to be optional feature (i.e. dependencies which are required by the package are optional). | ||||||
|  |  | ||||||
| * ``ahriman.web.middlewares`` provides middlewares for request handlers. | * ``ahriman.web.middlewares`` provides middlewares for request handlers. | ||||||
| * ``ahriman.web.schemas`` provides schemas (actually copy paste from dataclasses) used by swagger documentation. |  | ||||||
| * ``ahriman.web.views`` contains web views derived from aiohttp view class. | * ``ahriman.web.views`` contains web views derived from aiohttp view class. | ||||||
| * ``ahriman.web.apispec`` provides generators for swagger documentation. |  | ||||||
| * ``ahriman.web.cors`` contains helpers for cross origin resource sharing middlewares. |  | ||||||
| * ``ahriman.web.routes`` creates routes for web application. | * ``ahriman.web.routes`` creates routes for web application. | ||||||
| * ``ahriman.web.web`` provides main web application functions (e.g. start, initialization). | * ``ahriman.web.web`` provides main web application functions (e.g. start, initialization). | ||||||
|  |  | ||||||
| Application run | Application run | ||||||
| --------------- | --------------- | ||||||
|  |  | ||||||
| #. Parse command line arguments, find command and related handler which is set by parser. | * Parse command line arguments, find command and related handler which is set by parser. | ||||||
| #. Call ``Handler.execute`` method. | * Call ``Handler.execute`` method. | ||||||
| #. Define list of architectures to run. In case if there is more than one architecture specified run several subprocesses or process in current process otherwise. Class attribute ``ALLOW_MULTI_ARCHITECTURE_RUN`` controls whether application can be run in multiple processes or not - this feature is required for some handlers (e.g. ``Web``) which should be able to spawn child process in daemon mode (it is impossible to do from daemonic processes). | * Define list of architectures to run. In case if there is more than one architecture specified run several subprocesses or process in current process otherwise. Class attribute ``ALLOW_MULTI_ARCHITECTURE_RUN`` controls whether application can be run in multiple processes or not - this feature is required for some handlers (e.g. ``Web``) which should be able to spawn child process in daemon mode (it is impossible to do from daemonic processes). | ||||||
| #. In each child process call lock functions. | * In each child process call lock functions. | ||||||
| #. After success checks pass control to ``Handler.run`` method defined by specific handler class. | * After success checks pass control to ``Handler.run`` method defined by specific handler class. | ||||||
| #. Return result (success or failure) of each subprocess and exit from application. | * Return result (success or failure) of each subprocess and exit from application. | ||||||
| #. Some handlers may override their status and throw ``ExitCode`` exception. This exception is just silently suppressed and changes application exit code to ``1``. | * Some handlers may override their status and throw ``ExitCode`` exception. This exception is just silently suppressed and changes application exit code to ``1``. | ||||||
|  |  | ||||||
| In the most cases handlers spawn god class ``ahriman.application.application.Application`` class and call required methods. | In the most cases handlers spawn god class ``ahriman.application.application.Application`` class and call required methods. | ||||||
|  |  | ||||||
| @ -119,7 +114,7 @@ Schema and data migrations | |||||||
|  |  | ||||||
| The schema migration are applied according to current ``pragma user_info`` values, located at ``ahriman.core.database.migrations`` package and named as ``m000_migration_name.py`` (the preceding ``m`` is required in order to import migration content for tests). Additional class ``ahriman.core.database.migrations.Migrations`` reads all migrations automatically and applies them in alphabetical order. | The schema migration are applied according to current ``pragma user_info`` values, located at ``ahriman.core.database.migrations`` package and named as ``m000_migration_name.py`` (the preceding ``m`` is required in order to import migration content for tests). Additional class ``ahriman.core.database.migrations.Migrations`` reads all migrations automatically and applies them in alphabetical order. | ||||||
|  |  | ||||||
| These migrations can also contain data migrations. Though the recommended way is to migrate data directly from SQL requests, sometimes it is required to have external data (like packages list) in order to set correct data. To do so, special method `migrate_data` is used. | These migrations also contain data migrations. Though the recommended way is to migrate data directly from SQL requests, sometimes it is required to have external data (like packages list) in order to set correct data. To do so, special method `migrate_data` is used. | ||||||
|  |  | ||||||
| Type conversions | Type conversions | ||||||
| ^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^ | ||||||
| @ -131,12 +126,6 @@ By default, it parses rows into python dictionary. In addition, the following ps | |||||||
| Basic flows | Basic flows | ||||||
| ----------- | ----------- | ||||||
|  |  | ||||||
| By default package build operations are performed with ``PACKAGER`` which is specified in ``makepkg.conf``, however, it is possible to override this variable from command line; in this case service performs lookup in the following way: |  | ||||||
|  |  | ||||||
| * If packager is not set, it reads environment variables (e.g. ``SUDO_USER`` and ``USER``), otherwise it uses value from command line. |  | ||||||
| * It checks users for the specified username and tries to extract packager variable from it. |  | ||||||
| * If packager value has been found, it will be passed as ``PACKAGER`` system variable (sudo configuration required). |  | ||||||
|  |  | ||||||
| Add new packages or rebuild existing | Add new packages or rebuild existing | ||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | ||||||
|  |  | ||||||
| @ -151,7 +140,7 @@ This logic can be overwritten by specifying the ``source`` parameter, which is p | |||||||
| Rebuild packages | Rebuild packages | ||||||
| ^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^ | ||||||
|  |  | ||||||
| Same as add function for every package in repository. Optional filters by reverse dependency or build status can be supplied. | Same as add function for every package in repository. Optional filter by reverse dependency can be supplied. | ||||||
|  |  | ||||||
| Remove packages | Remove packages | ||||||
| ^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^ | ||||||
| @ -169,7 +158,6 @@ This feature is divided into to stages: check AUR for updates and run rebuild fo | |||||||
| #. For each level of tree it does: | #. For each level of tree it does: | ||||||
|  |  | ||||||
|    #. Download package data from AUR. |    #. Download package data from AUR. | ||||||
|    #. Bump ``pkgrel`` if there is duplicate version in the local repository (see explanation below). |  | ||||||
|    #. Build every package in clean chroot. |    #. Build every package in clean chroot. | ||||||
|    #. Sign packages if required. |    #. Sign packages if required. | ||||||
|    #. Add packages to database and sign database if required. |    #. Add packages to database and sign database if required. | ||||||
| @ -177,20 +165,6 @@ This feature is divided into to stages: check AUR for updates and run rebuild fo | |||||||
|  |  | ||||||
| After any step any package data is being removed. | After any step any package data is being removed. | ||||||
|  |  | ||||||
| pkgrel bump rules |  | ||||||
| ^^^^^^^^^^^^^^^^^ |  | ||||||
|  |  | ||||||
| The application is able to automatically bump package release (``pkgrel``) during build process if there is duplicate version in repository. The version will be incremented as following: |  | ||||||
|  |  | ||||||
| #. Get version of the remote package. |  | ||||||
| #. Get version of the local package if any. |  | ||||||
| #. If local version is not set, proceed with remote one. |  | ||||||
| #. If local version is set and epoch or package version (``pkgver``) are different, proceed with remote version. |  | ||||||
| #. If local version is set and remote version is newer than local one, proceed with remote. |  | ||||||
| #. Extract ``pkgrel`` value. |  | ||||||
| #. If it has ``major.minor`` notation (e.g. ``1.1``), then increment last part by 1, e.g. ``1.1 -> 1.2``, ``1.0.1 -> 1.0.2``. |  | ||||||
| #. If ``pkgrel`` is a number (e.g. ``1``), then append 1 to the end of the string, e.g. ``1 -> 1.1``. |  | ||||||
|  |  | ||||||
| Core functions reference | Core functions reference | ||||||
| ------------------------ | ------------------------ | ||||||
|  |  | ||||||
| @ -232,7 +206,7 @@ The package provides several authorization methods: disabled, based on configura | |||||||
|  |  | ||||||
| Disabled (default) authorization provider just allows everything for everyone and does not have any specific configuration (it uses some default configuration parameters though). It also provides generic interface for derived classes. | Disabled (default) authorization provider just allows everything for everyone and does not have any specific configuration (it uses some default configuration parameters though). It also provides generic interface for derived classes. | ||||||
|  |  | ||||||
| Mapping (aka configuration) provider uses hashed passwords with optional salt from the database in order to authenticate users. This provider also enables user permission checking (read/write) (authorization). Thus, it defines the following methods: | Mapping (aka configuration) provider uses hashed passwords with salt from the database in order to authenticate users. This provider also enables user permission checking (read/write) (authorization). Thus, it defines the following methods: | ||||||
|  |  | ||||||
| * ``check_credentials`` - user password validation (authentication). | * ``check_credentials`` - user password validation (authentication). | ||||||
| * ``verify_access`` - user permission validation (authorization). | * ``verify_access`` - user permission validation (authorization). | ||||||
| @ -250,7 +224,7 @@ OAuth provider uses library definitions (``aioauth-client``) in order *authentic | |||||||
|  |  | ||||||
| OAuth's implementation also allows authenticating users via username + password (in the same way as mapping does) though it is not recommended for end-users and password must be left blank. In particular this feature can be used by service reporting (aka robots). | OAuth's implementation also allows authenticating users via username + password (in the same way as mapping does) though it is not recommended for end-users and password must be left blank. In particular this feature can be used by service reporting (aka robots). | ||||||
|  |  | ||||||
| In addition, web service checks the source socket used. In case if it belongs to ``socket.AF_UNIX`` family, it will skip any further checks considering the request to be performed in safe environment (e.g. on the same physical machine). This feature, in particular is being used by the reporter instances in case if socket address is set in configuration. | In addition, web service checks the source socket used. In case if it belongs to ``socket.AF_UNIX`` family, it will skip any furher checks considering the request to be performed in safe environment (e.g. on the same physical machine). This feature, in particular is being used by the reporter instances in case if socket address is set in configuration. | ||||||
|  |  | ||||||
| In order to configure users there are special commands. | In order to configure users there are special commands. | ||||||
|  |  | ||||||
| @ -345,5 +319,3 @@ External calls | |||||||
| ^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^ | ||||||
|  |  | ||||||
| Web application provides external calls to control main service. It spawns child process with specific arguments and waits for its termination. This feature must be used either with authorization or in safe (i.e. when status page is not available world-wide) environment. | Web application provides external calls to control main service. It spawns child process with specific arguments and waits for its termination. This feature must be used either with authorization or in safe (i.e. when status page is not available world-wide) environment. | ||||||
|  |  | ||||||
| For most actions it also extracts user from authentication (if provided) and passes it to underlying process. |  | ||||||
|  | |||||||
| @ -1,18 +1,18 @@ | |||||||
| # AUTOMATICALLY GENERATED by `shtab` | # AUTOMATICALLY GENERATED by `shtab` | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_subparsers=('aur-search' 'search' 'help' 'help-commands-unsafe' 'help-updates' 'help-version' 'version' 'package-add' 'add' 'package-update' 'package-remove' 'remove' 'package-status' 'status' 'package-status-remove' 'package-status-update' 'status-update' 'patch-add' 'patch-list' 'patch-remove' 'patch-set-add' 'repo-backup' 'repo-check' 'check' 'repo-create-keyring' 'repo-create-mirrorlist' 'repo-daemon' 'daemon' 'repo-rebuild' 'rebuild' 'repo-remove-unknown' 'remove-unknown' 'repo-report' 'report' 'repo-restore' 'repo-sign' 'sign' 'repo-status-update' 'repo-sync' 'sync' 'repo-tree' 'repo-triggers' 'repo-update' 'update' 'service-clean' 'clean' 'repo-clean' 'service-config' 'config' 'repo-config' 'service-config-validate' 'config-validate' 'repo-config-validate' 'service-key-import' 'key-import' 'service-setup' 'init' 'repo-init' 'repo-setup' 'setup' 'service-shell' 'shell' 'user-add' 'user-list' 'user-remove' 'web') | _shtab_ahriman_subparsers=('aur-search' 'search' 'help' 'help-commands-unsafe' 'help-updates' 'help-version' 'version' 'package-add' 'add' 'package-update' 'package-remove' 'remove' 'package-status' 'status' 'package-status-remove' 'package-status-update' 'status-update' 'patch-add' 'patch-list' 'patch-remove' 'patch-set-add' 'repo-backup' 'repo-check' 'check' 'repo-daemon' 'daemon' 'repo-rebuild' 'rebuild' 'repo-remove-unknown' 'remove-unknown' 'repo-report' 'report' 'repo-restore' 'repo-sign' 'sign' 'repo-status-update' 'repo-sync' 'sync' 'repo-tree' 'repo-triggers' 'repo-update' 'update' 'service-clean' 'clean' 'repo-clean' 'service-config' 'config' 'repo-config' 'service-config-validate' 'config-validate' 'repo-config-validate' 'service-key-import' 'key-import' 'service-setup' 'init' 'repo-init' 'repo-setup' 'setup' 'service-shell' 'shell' 'user-add' 'user-list' 'user-remove' 'web') | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_option_strings=('-h' '--help' '-a' '--architecture' '-c' '--configuration' '--force' '-l' '--lock' '--log-handler' '--report' '--no-report' '-q' '--quiet' '--unsafe' '--wait-timeout' '-V' '--version') | _shtab_ahriman_option_strings=('-h' '--help' '-a' '--architecture' '-c' '--configuration' '--force' '-l' '--lock' '--report' '--no-report' '-q' '--quiet' '--unsafe' '-V' '--version') | ||||||
| _shtab_ahriman_aur_search_option_strings=('-h' '--help' '-e' '--exit-code' '--info' '--no-info' '--sort-by') | _shtab_ahriman_aur_search_option_strings=('-h' '--help' '-e' '--exit-code' '--info' '--no-info' '--sort-by') | ||||||
| _shtab_ahriman_search_option_strings=('-h' '--help' '-e' '--exit-code' '--info' '--no-info' '--sort-by') | _shtab_ahriman_search_option_strings=('-h' '--help' '-e' '--exit-code' '--info' '--no-info' '--sort-by') | ||||||
| _shtab_ahriman_help_option_strings=('-h' '--help') | _shtab_ahriman_help_option_strings=('-h' '--help') | ||||||
| _shtab_ahriman_help_commands_unsafe_option_strings=('-h' '--help') | _shtab_ahriman_help_commands_unsafe_option_strings=('-h' '--help' '--command') | ||||||
| _shtab_ahriman_help_updates_option_strings=('-h' '--help' '-e' '--exit-code') | _shtab_ahriman_help_updates_option_strings=('-h' '--help' '-e' '--exit-code') | ||||||
| _shtab_ahriman_help_version_option_strings=('-h' '--help') | _shtab_ahriman_help_version_option_strings=('-h' '--help') | ||||||
| _shtab_ahriman_version_option_strings=('-h' '--help') | _shtab_ahriman_version_option_strings=('-h' '--help') | ||||||
| _shtab_ahriman_package_add_option_strings=('-h' '--help' '--dependencies' '--no-dependencies' '-e' '--exit-code' '--increment' '--no-increment' '-n' '--now' '-y' '--refresh' '-s' '--source' '-u' '--username') | _shtab_ahriman_package_add_option_strings=('-h' '--help' '--dependencies' '--no-dependencies' '-e' '--exit-code' '-n' '--now' '-y' '--refresh' '-s' '--source') | ||||||
| _shtab_ahriman_add_option_strings=('-h' '--help' '--dependencies' '--no-dependencies' '-e' '--exit-code' '--increment' '--no-increment' '-n' '--now' '-y' '--refresh' '-s' '--source' '-u' '--username') | _shtab_ahriman_add_option_strings=('-h' '--help' '--dependencies' '--no-dependencies' '-e' '--exit-code' '-n' '--now' '-y' '--refresh' '-s' '--source') | ||||||
| _shtab_ahriman_package_update_option_strings=('-h' '--help' '--dependencies' '--no-dependencies' '-e' '--exit-code' '--increment' '--no-increment' '-n' '--now' '-y' '--refresh' '-s' '--source' '-u' '--username') | _shtab_ahriman_package_update_option_strings=('-h' '--help' '--dependencies' '--no-dependencies' '-e' '--exit-code' '-n' '--now' '-y' '--refresh' '-s' '--source') | ||||||
| _shtab_ahriman_package_remove_option_strings=('-h' '--help') | _shtab_ahriman_package_remove_option_strings=('-h' '--help') | ||||||
| _shtab_ahriman_remove_option_strings=('-h' '--help') | _shtab_ahriman_remove_option_strings=('-h' '--help') | ||||||
| _shtab_ahriman_package_status_option_strings=('-h' '--help' '--ahriman' '-e' '--exit-code' '--info' '--no-info' '-s' '--status') | _shtab_ahriman_package_status_option_strings=('-h' '--help' '--ahriman' '-e' '--exit-code' '--info' '--no-info' '-s' '--status') | ||||||
| @ -27,12 +27,10 @@ _shtab_ahriman_patch_set_add_option_strings=('-h' '--help' '-t' '--track') | |||||||
| _shtab_ahriman_repo_backup_option_strings=('-h' '--help') | _shtab_ahriman_repo_backup_option_strings=('-h' '--help') | ||||||
| _shtab_ahriman_repo_check_option_strings=('-h' '--help' '-e' '--exit-code' '--vcs' '--no-vcs' '-y' '--refresh') | _shtab_ahriman_repo_check_option_strings=('-h' '--help' '-e' '--exit-code' '--vcs' '--no-vcs' '-y' '--refresh') | ||||||
| _shtab_ahriman_check_option_strings=('-h' '--help' '-e' '--exit-code' '--vcs' '--no-vcs' '-y' '--refresh') | _shtab_ahriman_check_option_strings=('-h' '--help' '-e' '--exit-code' '--vcs' '--no-vcs' '-y' '--refresh') | ||||||
| _shtab_ahriman_repo_create_keyring_option_strings=('-h' '--help') |  | ||||||
| _shtab_ahriman_repo_create_mirrorlist_option_strings=('-h' '--help') |  | ||||||
| _shtab_ahriman_repo_daemon_option_strings=('-h' '--help' '-i' '--interval' '--aur' '--no-aur' '--dependencies' '--no-dependencies' '--local' '--no-local' '--manual' '--no-manual' '--vcs' '--no-vcs' '-y' '--refresh') | _shtab_ahriman_repo_daemon_option_strings=('-h' '--help' '-i' '--interval' '--aur' '--no-aur' '--dependencies' '--no-dependencies' '--local' '--no-local' '--manual' '--no-manual' '--vcs' '--no-vcs' '-y' '--refresh') | ||||||
| _shtab_ahriman_daemon_option_strings=('-h' '--help' '-i' '--interval' '--aur' '--no-aur' '--dependencies' '--no-dependencies' '--local' '--no-local' '--manual' '--no-manual' '--vcs' '--no-vcs' '-y' '--refresh') | _shtab_ahriman_daemon_option_strings=('-h' '--help' '-i' '--interval' '--aur' '--no-aur' '--dependencies' '--no-dependencies' '--local' '--no-local' '--manual' '--no-manual' '--vcs' '--no-vcs' '-y' '--refresh') | ||||||
| _shtab_ahriman_repo_rebuild_option_strings=('-h' '--help' '--depends-on' '--dry-run' '--from-database' '--increment' '--no-increment' '-e' '--exit-code' '-s' '--status' '-u' '--username') | _shtab_ahriman_repo_rebuild_option_strings=('-h' '--help' '--depends-on' '--dry-run' '--from-database' '-e' '--exit-code') | ||||||
| _shtab_ahriman_rebuild_option_strings=('-h' '--help' '--depends-on' '--dry-run' '--from-database' '--increment' '--no-increment' '-e' '--exit-code' '-s' '--status' '-u' '--username') | _shtab_ahriman_rebuild_option_strings=('-h' '--help' '--depends-on' '--dry-run' '--from-database' '-e' '--exit-code') | ||||||
| _shtab_ahriman_repo_remove_unknown_option_strings=('-h' '--help' '--dry-run') | _shtab_ahriman_repo_remove_unknown_option_strings=('-h' '--help' '--dry-run') | ||||||
| _shtab_ahriman_remove_unknown_option_strings=('-h' '--help' '--dry-run') | _shtab_ahriman_remove_unknown_option_strings=('-h' '--help' '--dry-run') | ||||||
| _shtab_ahriman_repo_report_option_strings=('-h' '--help') | _shtab_ahriman_repo_report_option_strings=('-h' '--help') | ||||||
| @ -43,10 +41,10 @@ _shtab_ahriman_sign_option_strings=('-h' '--help') | |||||||
| _shtab_ahriman_repo_status_update_option_strings=('-h' '--help' '-s' '--status') | _shtab_ahriman_repo_status_update_option_strings=('-h' '--help' '-s' '--status') | ||||||
| _shtab_ahriman_repo_sync_option_strings=('-h' '--help') | _shtab_ahriman_repo_sync_option_strings=('-h' '--help') | ||||||
| _shtab_ahriman_sync_option_strings=('-h' '--help') | _shtab_ahriman_sync_option_strings=('-h' '--help') | ||||||
| _shtab_ahriman_repo_tree_option_strings=('-h' '--help' '-p' '--partitions') | _shtab_ahriman_repo_tree_option_strings=('-h' '--help') | ||||||
| _shtab_ahriman_repo_triggers_option_strings=('-h' '--help') | _shtab_ahriman_repo_triggers_option_strings=('-h' '--help') | ||||||
| _shtab_ahriman_repo_update_option_strings=('-h' '--help' '--aur' '--no-aur' '--dependencies' '--no-dependencies' '--dry-run' '-e' '--exit-code' '--increment' '--no-increment' '--local' '--no-local' '--manual' '--no-manual' '-u' '--username' '--vcs' '--no-vcs' '-y' '--refresh') | _shtab_ahriman_repo_update_option_strings=('-h' '--help' '--aur' '--no-aur' '--dependencies' '--no-dependencies' '--dry-run' '-e' '--exit-code' '--local' '--no-local' '--manual' '--no-manual' '--vcs' '--no-vcs' '-y' '--refresh') | ||||||
| _shtab_ahriman_update_option_strings=('-h' '--help' '--aur' '--no-aur' '--dependencies' '--no-dependencies' '--dry-run' '-e' '--exit-code' '--increment' '--no-increment' '--local' '--no-local' '--manual' '--no-manual' '-u' '--username' '--vcs' '--no-vcs' '-y' '--refresh') | _shtab_ahriman_update_option_strings=('-h' '--help' '--aur' '--no-aur' '--dependencies' '--no-dependencies' '--dry-run' '-e' '--exit-code' '--local' '--no-local' '--manual' '--no-manual' '--vcs' '--no-vcs' '-y' '--refresh') | ||||||
| _shtab_ahriman_service_clean_option_strings=('-h' '--help' '--cache' '--no-cache' '--chroot' '--no-chroot' '--manual' '--no-manual' '--packages' '--no-packages' '--pacman' '--no-pacman') | _shtab_ahriman_service_clean_option_strings=('-h' '--help' '--cache' '--no-cache' '--chroot' '--no-chroot' '--manual' '--no-manual' '--packages' '--no-packages' '--pacman' '--no-pacman') | ||||||
| _shtab_ahriman_clean_option_strings=('-h' '--help' '--cache' '--no-cache' '--chroot' '--no-chroot' '--manual' '--no-manual' '--packages' '--no-packages' '--pacman' '--no-pacman') | _shtab_ahriman_clean_option_strings=('-h' '--help' '--cache' '--no-cache' '--chroot' '--no-chroot' '--manual' '--no-manual' '--packages' '--no-packages' '--pacman' '--no-pacman') | ||||||
| _shtab_ahriman_repo_clean_option_strings=('-h' '--help' '--cache' '--no-cache' '--chroot' '--no-chroot' '--manual' '--no-manual' '--packages' '--no-packages' '--pacman' '--no-pacman') | _shtab_ahriman_repo_clean_option_strings=('-h' '--help' '--cache' '--no-cache' '--chroot' '--no-chroot' '--manual' '--no-manual' '--packages' '--no-packages' '--pacman' '--no-pacman') | ||||||
| @ -58,22 +56,21 @@ _shtab_ahriman_config_validate_option_strings=('-h' '--help' '-e' '--exit-code') | |||||||
| _shtab_ahriman_repo_config_validate_option_strings=('-h' '--help' '-e' '--exit-code') | _shtab_ahriman_repo_config_validate_option_strings=('-h' '--help' '-e' '--exit-code') | ||||||
| _shtab_ahriman_service_key_import_option_strings=('-h' '--help' '--key-server') | _shtab_ahriman_service_key_import_option_strings=('-h' '--help' '--key-server') | ||||||
| _shtab_ahriman_key_import_option_strings=('-h' '--help' '--key-server') | _shtab_ahriman_key_import_option_strings=('-h' '--help' '--key-server') | ||||||
| _shtab_ahriman_service_setup_option_strings=('-h' '--help' '--build-as-user' '--build-command' '--from-configuration' '--generate-salt' '--no-generate-salt' '--makeflags-jobs' '--no-makeflags-jobs' '--mirror' '--multilib' '--no-multilib' '--packager' '--repository' '--server' '--sign-key' '--sign-target' '--web-port' '--web-unix-socket') | _shtab_ahriman_service_setup_option_strings=('-h' '--help' '--build-as-user' '--build-command' '--from-configuration' '--makeflags-jobs' '--no-makeflags-jobs' '--mirror' '--multilib' '--no-multilib' '--packager' '--repository' '--sign-key' '--sign-target' '--web-port' '--web-unix-socket') | ||||||
| _shtab_ahriman_init_option_strings=('-h' '--help' '--build-as-user' '--build-command' '--from-configuration' '--generate-salt' '--no-generate-salt' '--makeflags-jobs' '--no-makeflags-jobs' '--mirror' '--multilib' '--no-multilib' '--packager' '--repository' '--server' '--sign-key' '--sign-target' '--web-port' '--web-unix-socket') | _shtab_ahriman_init_option_strings=('-h' '--help' '--build-as-user' '--build-command' '--from-configuration' '--makeflags-jobs' '--no-makeflags-jobs' '--mirror' '--multilib' '--no-multilib' '--packager' '--repository' '--sign-key' '--sign-target' '--web-port' '--web-unix-socket') | ||||||
| _shtab_ahriman_repo_init_option_strings=('-h' '--help' '--build-as-user' '--build-command' '--from-configuration' '--generate-salt' '--no-generate-salt' '--makeflags-jobs' '--no-makeflags-jobs' '--mirror' '--multilib' '--no-multilib' '--packager' '--repository' '--server' '--sign-key' '--sign-target' '--web-port' '--web-unix-socket') | _shtab_ahriman_repo_init_option_strings=('-h' '--help' '--build-as-user' '--build-command' '--from-configuration' '--makeflags-jobs' '--no-makeflags-jobs' '--mirror' '--multilib' '--no-multilib' '--packager' '--repository' '--sign-key' '--sign-target' '--web-port' '--web-unix-socket') | ||||||
| _shtab_ahriman_repo_setup_option_strings=('-h' '--help' '--build-as-user' '--build-command' '--from-configuration' '--generate-salt' '--no-generate-salt' '--makeflags-jobs' '--no-makeflags-jobs' '--mirror' '--multilib' '--no-multilib' '--packager' '--repository' '--server' '--sign-key' '--sign-target' '--web-port' '--web-unix-socket') | _shtab_ahriman_repo_setup_option_strings=('-h' '--help' '--build-as-user' '--build-command' '--from-configuration' '--makeflags-jobs' '--no-makeflags-jobs' '--mirror' '--multilib' '--no-multilib' '--packager' '--repository' '--sign-key' '--sign-target' '--web-port' '--web-unix-socket') | ||||||
| _shtab_ahriman_setup_option_strings=('-h' '--help' '--build-as-user' '--build-command' '--from-configuration' '--generate-salt' '--no-generate-salt' '--makeflags-jobs' '--no-makeflags-jobs' '--mirror' '--multilib' '--no-multilib' '--packager' '--repository' '--server' '--sign-key' '--sign-target' '--web-port' '--web-unix-socket') | _shtab_ahriman_setup_option_strings=('-h' '--help' '--build-as-user' '--build-command' '--from-configuration' '--makeflags-jobs' '--no-makeflags-jobs' '--mirror' '--multilib' '--no-multilib' '--packager' '--repository' '--sign-key' '--sign-target' '--web-port' '--web-unix-socket') | ||||||
| _shtab_ahriman_service_shell_option_strings=('-h' '--help') | _shtab_ahriman_service_shell_option_strings=('-h' '--help') | ||||||
| _shtab_ahriman_shell_option_strings=('-h' '--help') | _shtab_ahriman_shell_option_strings=('-h' '--help') | ||||||
| _shtab_ahriman_user_add_option_strings=('-h' '--help' '--key' '--packager' '-p' '--password' '-r' '--role') | _shtab_ahriman_user_add_option_strings=('-h' '--help' '-p' '--password' '-r' '--role' '-s' '--secure') | ||||||
| _shtab_ahriman_user_list_option_strings=('-h' '--help' '-e' '--exit-code' '-r' '--role') | _shtab_ahriman_user_list_option_strings=('-h' '--help' '-e' '--exit-code' '-r' '--role') | ||||||
| _shtab_ahriman_user_remove_option_strings=('-h' '--help') | _shtab_ahriman_user_remove_option_strings=('-h' '--help') | ||||||
| _shtab_ahriman_web_option_strings=('-h' '--help') | _shtab_ahriman_web_option_strings=('-h' '--help') | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_pos_0_choices=('aur-search' 'search' 'help' 'help-commands-unsafe' 'help-updates' 'help-version' 'version' 'package-add' 'add' 'package-update' 'package-remove' 'remove' 'package-status' 'status' 'package-status-remove' 'package-status-update' 'status-update' 'patch-add' 'patch-list' 'patch-remove' 'patch-set-add' 'repo-backup' 'repo-check' 'check' 'repo-create-keyring' 'repo-create-mirrorlist' 'repo-daemon' 'daemon' 'repo-rebuild' 'rebuild' 'repo-remove-unknown' 'remove-unknown' 'repo-report' 'report' 'repo-restore' 'repo-sign' 'sign' 'repo-status-update' 'repo-sync' 'sync' 'repo-tree' 'repo-triggers' 'repo-update' 'update' 'service-clean' 'clean' 'repo-clean' 'service-config' 'config' 'repo-config' 'service-config-validate' 'config-validate' 'repo-config-validate' 'service-key-import' 'key-import' 'service-setup' 'init' 'repo-init' 'repo-setup' 'setup' 'service-shell' 'shell' 'user-add' 'user-list' 'user-remove' 'web') | _shtab_ahriman_pos_0_choices=('aur-search' 'search' 'help' 'help-commands-unsafe' 'help-updates' 'help-version' 'version' 'package-add' 'add' 'package-update' 'package-remove' 'remove' 'package-status' 'status' 'package-status-remove' 'package-status-update' 'status-update' 'patch-add' 'patch-list' 'patch-remove' 'patch-set-add' 'repo-backup' 'repo-check' 'check' 'repo-daemon' 'daemon' 'repo-rebuild' 'rebuild' 'repo-remove-unknown' 'remove-unknown' 'repo-report' 'report' 'repo-restore' 'repo-sign' 'sign' 'repo-status-update' 'repo-sync' 'sync' 'repo-tree' 'repo-triggers' 'repo-update' 'update' 'service-clean' 'clean' 'repo-clean' 'service-config' 'config' 'repo-config' 'service-config-validate' 'config-validate' 'repo-config-validate' 'service-key-import' 'key-import' 'service-setup' 'init' 'repo-init' 'repo-setup' 'setup' 'service-shell' 'shell' 'user-add' 'user-list' 'user-remove' 'web') | ||||||
| _shtab_ahriman___log_handler_choices=('console' 'syslog' 'journald') |  | ||||||
| _shtab_ahriman_aur_search___sort_by_choices=('description' 'first_submitted' 'id' 'last_modified' 'maintainer' 'name' 'num_votes' 'out_of_date' 'package_base' 'package_base_id' 'popularity' 'repository' 'submitter' 'url' 'url_path' 'version') | _shtab_ahriman_aur_search___sort_by_choices=('description' 'first_submitted' 'id' 'last_modified' 'maintainer' 'name' 'num_votes' 'out_of_date' 'package_base' 'package_base_id' 'popularity' 'repository' 'submitter' 'url' 'url_path' 'version') | ||||||
| _shtab_ahriman_search___sort_by_choices=('description' 'first_submitted' 'id' 'last_modified' 'maintainer' 'name' 'num_votes' 'out_of_date' 'package_base' 'package_base_id' 'popularity' 'repository' 'submitter' 'url' 'url_path' 'version') | _shtab_ahriman_search___sort_by_choices=('description' 'first_submitted' 'id' 'last_modified' 'maintainer' 'name' 'num_votes' 'out_of_date' 'package_base' 'package_base_id' 'popularity' 'repository' 'submitter' 'url' 'url_path' 'version') | ||||||
| _shtab_ahriman_package_add__s_choices=('auto' 'archive' 'aur' 'directory' 'local' 'remote' 'repository') | _shtab_ahriman_package_add__s_choices=('auto' 'archive' 'aur' 'directory' 'local' 'remote' 'repository') | ||||||
| @ -90,10 +87,6 @@ _shtab_ahriman_package_status_update__s_choices=('unknown' 'pending' 'building' | |||||||
| _shtab_ahriman_package_status_update___status_choices=('unknown' 'pending' 'building' 'failed' 'success') | _shtab_ahriman_package_status_update___status_choices=('unknown' 'pending' 'building' 'failed' 'success') | ||||||
| _shtab_ahriman_status_update__s_choices=('unknown' 'pending' 'building' 'failed' 'success') | _shtab_ahriman_status_update__s_choices=('unknown' 'pending' 'building' 'failed' 'success') | ||||||
| _shtab_ahriman_status_update___status_choices=('unknown' 'pending' 'building' 'failed' 'success') | _shtab_ahriman_status_update___status_choices=('unknown' 'pending' 'building' 'failed' 'success') | ||||||
| _shtab_ahriman_repo_rebuild__s_choices=('unknown' 'pending' 'building' 'failed' 'success') |  | ||||||
| _shtab_ahriman_repo_rebuild___status_choices=('unknown' 'pending' 'building' 'failed' 'success') |  | ||||||
| _shtab_ahriman_rebuild__s_choices=('unknown' 'pending' 'building' 'failed' 'success') |  | ||||||
| _shtab_ahriman_rebuild___status_choices=('unknown' 'pending' 'building' 'failed' 'success') |  | ||||||
| _shtab_ahriman_repo_status_update__s_choices=('unknown' 'pending' 'building' 'failed' 'success') | _shtab_ahriman_repo_status_update__s_choices=('unknown' 'pending' 'building' 'failed' 'success') | ||||||
| _shtab_ahriman_repo_status_update___status_choices=('unknown' 'pending' 'building' 'failed' 'success') | _shtab_ahriman_repo_status_update___status_choices=('unknown' 'pending' 'building' 'failed' 'success') | ||||||
| _shtab_ahriman_service_setup___sign_target_choices=('disabled' 'packages' 'repository') | _shtab_ahriman_service_setup___sign_target_choices=('disabled' 'packages' 'repository') | ||||||
| @ -133,7 +126,6 @@ _shtab_ahriman_search___info_nargs=0 | |||||||
| _shtab_ahriman_search___no_info_nargs=0 | _shtab_ahriman_search___no_info_nargs=0 | ||||||
| _shtab_ahriman_help__h_nargs=0 | _shtab_ahriman_help__h_nargs=0 | ||||||
| _shtab_ahriman_help___help_nargs=0 | _shtab_ahriman_help___help_nargs=0 | ||||||
| _shtab_ahriman_help_commands_unsafe_pos_0_nargs=* |  | ||||||
| _shtab_ahriman_help_commands_unsafe__h_nargs=0 | _shtab_ahriman_help_commands_unsafe__h_nargs=0 | ||||||
| _shtab_ahriman_help_commands_unsafe___help_nargs=0 | _shtab_ahriman_help_commands_unsafe___help_nargs=0 | ||||||
| _shtab_ahriman_help_updates__h_nargs=0 | _shtab_ahriman_help_updates__h_nargs=0 | ||||||
| @ -151,8 +143,6 @@ _shtab_ahriman_package_add___dependencies_nargs=0 | |||||||
| _shtab_ahriman_package_add___no_dependencies_nargs=0 | _shtab_ahriman_package_add___no_dependencies_nargs=0 | ||||||
| _shtab_ahriman_package_add__e_nargs=0 | _shtab_ahriman_package_add__e_nargs=0 | ||||||
| _shtab_ahriman_package_add___exit_code_nargs=0 | _shtab_ahriman_package_add___exit_code_nargs=0 | ||||||
| _shtab_ahriman_package_add___increment_nargs=0 |  | ||||||
| _shtab_ahriman_package_add___no_increment_nargs=0 |  | ||||||
| _shtab_ahriman_package_add__n_nargs=0 | _shtab_ahriman_package_add__n_nargs=0 | ||||||
| _shtab_ahriman_package_add___now_nargs=0 | _shtab_ahriman_package_add___now_nargs=0 | ||||||
| _shtab_ahriman_package_add__y_nargs=0 | _shtab_ahriman_package_add__y_nargs=0 | ||||||
| @ -164,8 +154,6 @@ _shtab_ahriman_add___dependencies_nargs=0 | |||||||
| _shtab_ahriman_add___no_dependencies_nargs=0 | _shtab_ahriman_add___no_dependencies_nargs=0 | ||||||
| _shtab_ahriman_add__e_nargs=0 | _shtab_ahriman_add__e_nargs=0 | ||||||
| _shtab_ahriman_add___exit_code_nargs=0 | _shtab_ahriman_add___exit_code_nargs=0 | ||||||
| _shtab_ahriman_add___increment_nargs=0 |  | ||||||
| _shtab_ahriman_add___no_increment_nargs=0 |  | ||||||
| _shtab_ahriman_add__n_nargs=0 | _shtab_ahriman_add__n_nargs=0 | ||||||
| _shtab_ahriman_add___now_nargs=0 | _shtab_ahriman_add___now_nargs=0 | ||||||
| _shtab_ahriman_add__y_nargs=0 | _shtab_ahriman_add__y_nargs=0 | ||||||
| @ -177,8 +165,6 @@ _shtab_ahriman_package_update___dependencies_nargs=0 | |||||||
| _shtab_ahriman_package_update___no_dependencies_nargs=0 | _shtab_ahriman_package_update___no_dependencies_nargs=0 | ||||||
| _shtab_ahriman_package_update__e_nargs=0 | _shtab_ahriman_package_update__e_nargs=0 | ||||||
| _shtab_ahriman_package_update___exit_code_nargs=0 | _shtab_ahriman_package_update___exit_code_nargs=0 | ||||||
| _shtab_ahriman_package_update___increment_nargs=0 |  | ||||||
| _shtab_ahriman_package_update___no_increment_nargs=0 |  | ||||||
| _shtab_ahriman_package_update__n_nargs=0 | _shtab_ahriman_package_update__n_nargs=0 | ||||||
| _shtab_ahriman_package_update___now_nargs=0 | _shtab_ahriman_package_update___now_nargs=0 | ||||||
| _shtab_ahriman_package_update__y_nargs=0 | _shtab_ahriman_package_update__y_nargs=0 | ||||||
| @ -244,10 +230,6 @@ _shtab_ahriman_check___vcs_nargs=0 | |||||||
| _shtab_ahriman_check___no_vcs_nargs=0 | _shtab_ahriman_check___no_vcs_nargs=0 | ||||||
| _shtab_ahriman_check__y_nargs=0 | _shtab_ahriman_check__y_nargs=0 | ||||||
| _shtab_ahriman_check___refresh_nargs=0 | _shtab_ahriman_check___refresh_nargs=0 | ||||||
| _shtab_ahriman_repo_create_keyring__h_nargs=0 |  | ||||||
| _shtab_ahriman_repo_create_keyring___help_nargs=0 |  | ||||||
| _shtab_ahriman_repo_create_mirrorlist__h_nargs=0 |  | ||||||
| _shtab_ahriman_repo_create_mirrorlist___help_nargs=0 |  | ||||||
| _shtab_ahriman_repo_daemon__h_nargs=0 | _shtab_ahriman_repo_daemon__h_nargs=0 | ||||||
| _shtab_ahriman_repo_daemon___help_nargs=0 | _shtab_ahriman_repo_daemon___help_nargs=0 | ||||||
| _shtab_ahriman_repo_daemon___aur_nargs=0 | _shtab_ahriman_repo_daemon___aur_nargs=0 | ||||||
| @ -280,16 +262,12 @@ _shtab_ahriman_repo_rebuild__h_nargs=0 | |||||||
| _shtab_ahriman_repo_rebuild___help_nargs=0 | _shtab_ahriman_repo_rebuild___help_nargs=0 | ||||||
| _shtab_ahriman_repo_rebuild___dry_run_nargs=0 | _shtab_ahriman_repo_rebuild___dry_run_nargs=0 | ||||||
| _shtab_ahriman_repo_rebuild___from_database_nargs=0 | _shtab_ahriman_repo_rebuild___from_database_nargs=0 | ||||||
| _shtab_ahriman_repo_rebuild___increment_nargs=0 |  | ||||||
| _shtab_ahriman_repo_rebuild___no_increment_nargs=0 |  | ||||||
| _shtab_ahriman_repo_rebuild__e_nargs=0 | _shtab_ahriman_repo_rebuild__e_nargs=0 | ||||||
| _shtab_ahriman_repo_rebuild___exit_code_nargs=0 | _shtab_ahriman_repo_rebuild___exit_code_nargs=0 | ||||||
| _shtab_ahriman_rebuild__h_nargs=0 | _shtab_ahriman_rebuild__h_nargs=0 | ||||||
| _shtab_ahriman_rebuild___help_nargs=0 | _shtab_ahriman_rebuild___help_nargs=0 | ||||||
| _shtab_ahriman_rebuild___dry_run_nargs=0 | _shtab_ahriman_rebuild___dry_run_nargs=0 | ||||||
| _shtab_ahriman_rebuild___from_database_nargs=0 | _shtab_ahriman_rebuild___from_database_nargs=0 | ||||||
| _shtab_ahriman_rebuild___increment_nargs=0 |  | ||||||
| _shtab_ahriman_rebuild___no_increment_nargs=0 |  | ||||||
| _shtab_ahriman_rebuild__e_nargs=0 | _shtab_ahriman_rebuild__e_nargs=0 | ||||||
| _shtab_ahriman_rebuild___exit_code_nargs=0 | _shtab_ahriman_rebuild___exit_code_nargs=0 | ||||||
| _shtab_ahriman_repo_remove_unknown__h_nargs=0 | _shtab_ahriman_repo_remove_unknown__h_nargs=0 | ||||||
| @ -331,8 +309,6 @@ _shtab_ahriman_repo_update___no_dependencies_nargs=0 | |||||||
| _shtab_ahriman_repo_update___dry_run_nargs=0 | _shtab_ahriman_repo_update___dry_run_nargs=0 | ||||||
| _shtab_ahriman_repo_update__e_nargs=0 | _shtab_ahriman_repo_update__e_nargs=0 | ||||||
| _shtab_ahriman_repo_update___exit_code_nargs=0 | _shtab_ahriman_repo_update___exit_code_nargs=0 | ||||||
| _shtab_ahriman_repo_update___increment_nargs=0 |  | ||||||
| _shtab_ahriman_repo_update___no_increment_nargs=0 |  | ||||||
| _shtab_ahriman_repo_update___local_nargs=0 | _shtab_ahriman_repo_update___local_nargs=0 | ||||||
| _shtab_ahriman_repo_update___no_local_nargs=0 | _shtab_ahriman_repo_update___no_local_nargs=0 | ||||||
| _shtab_ahriman_repo_update___manual_nargs=0 | _shtab_ahriman_repo_update___manual_nargs=0 | ||||||
| @ -351,8 +327,6 @@ _shtab_ahriman_update___no_dependencies_nargs=0 | |||||||
| _shtab_ahriman_update___dry_run_nargs=0 | _shtab_ahriman_update___dry_run_nargs=0 | ||||||
| _shtab_ahriman_update__e_nargs=0 | _shtab_ahriman_update__e_nargs=0 | ||||||
| _shtab_ahriman_update___exit_code_nargs=0 | _shtab_ahriman_update___exit_code_nargs=0 | ||||||
| _shtab_ahriman_update___increment_nargs=0 |  | ||||||
| _shtab_ahriman_update___no_increment_nargs=0 |  | ||||||
| _shtab_ahriman_update___local_nargs=0 | _shtab_ahriman_update___local_nargs=0 | ||||||
| _shtab_ahriman_update___no_local_nargs=0 | _shtab_ahriman_update___no_local_nargs=0 | ||||||
| _shtab_ahriman_update___manual_nargs=0 | _shtab_ahriman_update___manual_nargs=0 | ||||||
| @ -427,40 +401,30 @@ _shtab_ahriman_key_import__h_nargs=0 | |||||||
| _shtab_ahriman_key_import___help_nargs=0 | _shtab_ahriman_key_import___help_nargs=0 | ||||||
| _shtab_ahriman_service_setup__h_nargs=0 | _shtab_ahriman_service_setup__h_nargs=0 | ||||||
| _shtab_ahriman_service_setup___help_nargs=0 | _shtab_ahriman_service_setup___help_nargs=0 | ||||||
| _shtab_ahriman_service_setup___generate_salt_nargs=0 |  | ||||||
| _shtab_ahriman_service_setup___no_generate_salt_nargs=0 |  | ||||||
| _shtab_ahriman_service_setup___makeflags_jobs_nargs=0 | _shtab_ahriman_service_setup___makeflags_jobs_nargs=0 | ||||||
| _shtab_ahriman_service_setup___no_makeflags_jobs_nargs=0 | _shtab_ahriman_service_setup___no_makeflags_jobs_nargs=0 | ||||||
| _shtab_ahriman_service_setup___multilib_nargs=0 | _shtab_ahriman_service_setup___multilib_nargs=0 | ||||||
| _shtab_ahriman_service_setup___no_multilib_nargs=0 | _shtab_ahriman_service_setup___no_multilib_nargs=0 | ||||||
| _shtab_ahriman_init__h_nargs=0 | _shtab_ahriman_init__h_nargs=0 | ||||||
| _shtab_ahriman_init___help_nargs=0 | _shtab_ahriman_init___help_nargs=0 | ||||||
| _shtab_ahriman_init___generate_salt_nargs=0 |  | ||||||
| _shtab_ahriman_init___no_generate_salt_nargs=0 |  | ||||||
| _shtab_ahriman_init___makeflags_jobs_nargs=0 | _shtab_ahriman_init___makeflags_jobs_nargs=0 | ||||||
| _shtab_ahriman_init___no_makeflags_jobs_nargs=0 | _shtab_ahriman_init___no_makeflags_jobs_nargs=0 | ||||||
| _shtab_ahriman_init___multilib_nargs=0 | _shtab_ahriman_init___multilib_nargs=0 | ||||||
| _shtab_ahriman_init___no_multilib_nargs=0 | _shtab_ahriman_init___no_multilib_nargs=0 | ||||||
| _shtab_ahriman_repo_init__h_nargs=0 | _shtab_ahriman_repo_init__h_nargs=0 | ||||||
| _shtab_ahriman_repo_init___help_nargs=0 | _shtab_ahriman_repo_init___help_nargs=0 | ||||||
| _shtab_ahriman_repo_init___generate_salt_nargs=0 |  | ||||||
| _shtab_ahriman_repo_init___no_generate_salt_nargs=0 |  | ||||||
| _shtab_ahriman_repo_init___makeflags_jobs_nargs=0 | _shtab_ahriman_repo_init___makeflags_jobs_nargs=0 | ||||||
| _shtab_ahriman_repo_init___no_makeflags_jobs_nargs=0 | _shtab_ahriman_repo_init___no_makeflags_jobs_nargs=0 | ||||||
| _shtab_ahriman_repo_init___multilib_nargs=0 | _shtab_ahriman_repo_init___multilib_nargs=0 | ||||||
| _shtab_ahriman_repo_init___no_multilib_nargs=0 | _shtab_ahriman_repo_init___no_multilib_nargs=0 | ||||||
| _shtab_ahriman_repo_setup__h_nargs=0 | _shtab_ahriman_repo_setup__h_nargs=0 | ||||||
| _shtab_ahriman_repo_setup___help_nargs=0 | _shtab_ahriman_repo_setup___help_nargs=0 | ||||||
| _shtab_ahriman_repo_setup___generate_salt_nargs=0 |  | ||||||
| _shtab_ahriman_repo_setup___no_generate_salt_nargs=0 |  | ||||||
| _shtab_ahriman_repo_setup___makeflags_jobs_nargs=0 | _shtab_ahriman_repo_setup___makeflags_jobs_nargs=0 | ||||||
| _shtab_ahriman_repo_setup___no_makeflags_jobs_nargs=0 | _shtab_ahriman_repo_setup___no_makeflags_jobs_nargs=0 | ||||||
| _shtab_ahriman_repo_setup___multilib_nargs=0 | _shtab_ahriman_repo_setup___multilib_nargs=0 | ||||||
| _shtab_ahriman_repo_setup___no_multilib_nargs=0 | _shtab_ahriman_repo_setup___no_multilib_nargs=0 | ||||||
| _shtab_ahriman_setup__h_nargs=0 | _shtab_ahriman_setup__h_nargs=0 | ||||||
| _shtab_ahriman_setup___help_nargs=0 | _shtab_ahriman_setup___help_nargs=0 | ||||||
| _shtab_ahriman_setup___generate_salt_nargs=0 |  | ||||||
| _shtab_ahriman_setup___no_generate_salt_nargs=0 |  | ||||||
| _shtab_ahriman_setup___makeflags_jobs_nargs=0 | _shtab_ahriman_setup___makeflags_jobs_nargs=0 | ||||||
| _shtab_ahriman_setup___no_makeflags_jobs_nargs=0 | _shtab_ahriman_setup___no_makeflags_jobs_nargs=0 | ||||||
| _shtab_ahriman_setup___multilib_nargs=0 | _shtab_ahriman_setup___multilib_nargs=0 | ||||||
| @ -475,6 +439,8 @@ _shtab_ahriman_shell__v_nargs=0 | |||||||
| _shtab_ahriman_shell___verbose_nargs=0 | _shtab_ahriman_shell___verbose_nargs=0 | ||||||
| _shtab_ahriman_user_add__h_nargs=0 | _shtab_ahriman_user_add__h_nargs=0 | ||||||
| _shtab_ahriman_user_add___help_nargs=0 | _shtab_ahriman_user_add___help_nargs=0 | ||||||
|  | _shtab_ahriman_user_add__s_nargs=0 | ||||||
|  | _shtab_ahriman_user_add___secure_nargs=0 | ||||||
| _shtab_ahriman_user_list__h_nargs=0 | _shtab_ahriman_user_list__h_nargs=0 | ||||||
| _shtab_ahriman_user_list___help_nargs=0 | _shtab_ahriman_user_list___help_nargs=0 | ||||||
| _shtab_ahriman_user_list__e_nargs=0 | _shtab_ahriman_user_list__e_nargs=0 | ||||||
| @ -503,7 +469,7 @@ _shtab_replace_nonword() { | |||||||
| # set default values (called for the initial parser & any subparsers) | # set default values (called for the initial parser & any subparsers) | ||||||
| _set_parser_defaults() { | _set_parser_defaults() { | ||||||
|   local subparsers_var="${prefix}_subparsers[@]" |   local subparsers_var="${prefix}_subparsers[@]" | ||||||
|   sub_parsers=${!subparsers_var-} |   sub_parsers=${!subparsers_var} | ||||||
| 
 | 
 | ||||||
|   local current_option_strings_var="${prefix}_option_strings[@]" |   local current_option_strings_var="${prefix}_option_strings[@]" | ||||||
|   current_option_strings=${!current_option_strings_var} |   current_option_strings=${!current_option_strings_var} | ||||||
| @ -520,13 +486,13 @@ _set_new_action() { | |||||||
|   current_action="${prefix}_$(_shtab_replace_nonword $1)" |   current_action="${prefix}_$(_shtab_replace_nonword $1)" | ||||||
| 
 | 
 | ||||||
|   local current_action_compgen_var=${current_action}_COMPGEN |   local current_action_compgen_var=${current_action}_COMPGEN | ||||||
|   current_action_compgen="${!current_action_compgen_var-}" |   current_action_compgen="${!current_action_compgen_var}" | ||||||
| 
 | 
 | ||||||
|   local current_action_choices_var="${current_action}_choices[@]" |   local current_action_choices_var="${current_action}_choices[@]" | ||||||
|   current_action_choices="${!current_action_choices_var-}" |   current_action_choices="${!current_action_choices_var}" | ||||||
| 
 | 
 | ||||||
|   local current_action_nargs_var="${current_action}_nargs" |   local current_action_nargs_var="${current_action}_nargs" | ||||||
|   if [ -n "${!current_action_nargs_var-}" ]; then |   if [ -n "${!current_action_nargs_var}" ]; then | ||||||
|     current_action_nargs="${!current_action_nargs_var}" |     current_action_nargs="${!current_action_nargs_var}" | ||||||
|   else |   else | ||||||
|     current_action_nargs=1 |     current_action_nargs=1 | ||||||
| @ -548,8 +514,8 @@ _shtab_ahriman() { | |||||||
|   local completing_word="${COMP_WORDS[COMP_CWORD]}" |   local completing_word="${COMP_WORDS[COMP_CWORD]}" | ||||||
|   COMPREPLY=() |   COMPREPLY=() | ||||||
| 
 | 
 | ||||||
|   local prefix=_shtab_ahriman |   prefix=_shtab_ahriman | ||||||
|   local word_index=0 |   word_index=0 | ||||||
|   _set_parser_defaults |   _set_parser_defaults | ||||||
|   word_index=1 |   word_index=1 | ||||||
| 
 | 
 | ||||||
| @ -558,13 +524,13 @@ _shtab_ahriman() { | |||||||
|   while [ $word_index -ne $COMP_CWORD ]; do |   while [ $word_index -ne $COMP_CWORD ]; do | ||||||
|     local this_word="${COMP_WORDS[$word_index]}" |     local this_word="${COMP_WORDS[$word_index]}" | ||||||
| 
 | 
 | ||||||
|     if [[ -n $sub_parsers && " ${sub_parsers[@]} " == *" ${this_word} "* ]]; then |     if [[ -n $sub_parsers && " ${sub_parsers[@]} " =~ " ${this_word} " ]]; then | ||||||
|       # valid subcommand: add it to the prefix & reset the current action |       # valid subcommand: add it to the prefix & reset the current action | ||||||
|       prefix="${prefix}_$(_shtab_replace_nonword $this_word)" |       prefix="${prefix}_$(_shtab_replace_nonword $this_word)" | ||||||
|       _set_parser_defaults |       _set_parser_defaults | ||||||
|     fi |     fi | ||||||
| 
 | 
 | ||||||
|     if [[ " ${current_option_strings[@]} " == *" ${this_word} "* ]]; then |     if [[ " ${current_option_strings[@]} " =~ " ${this_word} " ]]; then | ||||||
|       # a new action should be acquired (due to recognised option string or |       # a new action should be acquired (due to recognised option string or | ||||||
|       # no more input expected from current action); |       # no more input expected from current action); | ||||||
|       # the next positional action can fill in here |       # the next positional action can fill in here | ||||||
| @ -36,8 +36,6 @@ _shtab_ahriman_commands() { | |||||||
|     "repo-clean:remove local caches" |     "repo-clean:remove local caches" | ||||||
|     "repo-config:dump configuration for the specified architecture" |     "repo-config:dump configuration for the specified architecture" | ||||||
|     "repo-config-validate:validate configuration and print found errors" |     "repo-config-validate:validate configuration and print found errors" | ||||||
|     "repo-create-keyring:create package which contains list of trusted keys as set by configuration. Note, that this action will only create package, the package itself has to be built manually" |  | ||||||
|     "repo-create-mirrorlist:create package which contains list of available mirrors as set by configuration. Note, that this action will only create package, the package itself has to be built manually" |  | ||||||
|     "repo-daemon:start process which periodically will run update process" |     "repo-daemon:start process which periodically will run update process" | ||||||
|     "repo-init:create initial service configuration, requires root" |     "repo-init:create initial service configuration, requires root" | ||||||
|     "repo-rebuild:force rebuild whole repository" |     "repo-rebuild:force rebuild whole repository" | ||||||
| @ -58,9 +56,9 @@ _shtab_ahriman_commands() { | |||||||
|     "service-config-validate:validate configuration and print found errors" |     "service-config-validate:validate configuration and print found errors" | ||||||
|     "service-key-import:import PGP key from public sources to the repository user" |     "service-key-import:import PGP key from public sources to the repository user" | ||||||
|     "service-setup:create initial service configuration, requires root" |     "service-setup:create initial service configuration, requires root" | ||||||
|     "service-shell:drop into python shell" |     "service-shell:drop into python shell while having created application" | ||||||
|     "setup:create initial service configuration, requires root" |     "setup:create initial service configuration, requires root" | ||||||
|     "shell:drop into python shell" |     "shell:drop into python shell while having created application" | ||||||
|     "sign:(re-)sign packages and repository database according to current settings" |     "sign:(re-)sign packages and repository database according to current settings" | ||||||
|     "status:request status of the package" |     "status:request status of the package" | ||||||
|     "status-update:update package status on the status page" |     "status-update:update package status on the status page" | ||||||
| @ -77,89 +75,85 @@ _shtab_ahriman_commands() { | |||||||
| 
 | 
 | ||||||
| _shtab_ahriman_options=( | _shtab_ahriman_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "*"{-a,--architecture}"[target architectures. For several subcommands it can be used multiple times (default\: None)]:architecture:" |   "*"{-a,--architecture}"[target architectures. For several subcommands it can be used multiple times]:architecture:" | ||||||
|   {-c,--configuration}"[configuration path (default\: \/etc\/ahriman.ini)]:configuration:" |   {-c,--configuration}"[configuration path]:configuration:" | ||||||
|   "--force[force run, remove file lock (default\: False)]" |   "--force[force run, remove file lock]" | ||||||
|   {-l,--lock}"[lock file (default\: \/tmp\/ahriman.lock)]:lock:" |   {-l,--lock}"[lock file]:lock:" | ||||||
|   "--log-handler[explicit log handler specification. If none set, the handler will be guessed from environment (default\: None)]:log_handler:(console syslog journald)" |   {--report,--no-report}"[force enable or disable reporting to web service (default\: \%(default)s)]:report:" | ||||||
|   {--report,--no-report}"[force enable or disable reporting to web service (default\: True)]:report:" |   {-q,--quiet}"[force disable any logging]" | ||||||
|   {-q,--quiet}"[force disable any logging (default\: False)]" |   "--unsafe[allow to run ahriman as non-ahriman user. Some actions might be unavailable]" | ||||||
|   "--unsafe[allow to run ahriman as non-ahriman user. Some actions might be unavailable (default\: False)]" |  | ||||||
|   "--wait-timeout[wait for lock to be free. Negative value will lead to immediate application run even if there is lock file. In case of zero value, the application will wait infinitely (default\: -1)]:wait_timeout:" |  | ||||||
|   "(- : *)"{-V,--version}"[show program\'s version number and exit]" |   "(- : *)"{-V,--version}"[show program\'s version number and exit]" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_add_options=( | _shtab_ahriman_add_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {--dependencies,--no-dependencies}"[process missing package dependencies (default\: True)]:dependencies:" |   {--dependencies,--no-dependencies}"[process missing package dependencies (default\: \%(default)s)]:dependencies:" | ||||||
|   {-e,--exit-code}"[return non-zero exit status if result is empty (default\: False)]" |   {-e,--exit-code}"[return non-zero exit status if result is empty]" | ||||||
|   {--increment,--no-increment}"[increment package release (pkgrel) version on duplicate (default\: True)]:increment:" |   {-n,--now}"[run update function after]" | ||||||
|   {-n,--now}"[run update function after (default\: False)]" |   "*"{-y,--refresh}"[download fresh package databases from the mirror before actions, -yy to force refresh even if up to date]" | ||||||
|   "*"{-y,--refresh}"[download fresh package databases from the mirror before actions, -yy to force refresh even if up to date (default\: False)]" |   {-s,--source}"[explicitly specify the package source for this command]:source:(auto archive aur directory local remote repository)" | ||||||
|   {-s,--source}"[explicitly specify the package source for this command (default\: PackageSource.Auto)]:source:(auto archive aur directory local remote repository)" |  | ||||||
|   {-u,--username}"[build as user (default\: None)]:username:" |  | ||||||
|   "(*):package source (base name, path to local files, remote URL):" |   "(*):package source (base name, path to local files, remote URL):" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_aur_search_options=( | _shtab_ahriman_aur_search_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {-e,--exit-code}"[return non-zero exit status if result is empty (default\: False)]" |   {-e,--exit-code}"[return non-zero exit status if result is empty]" | ||||||
|   {--info,--no-info}"[show additional package information (default\: False)]:info:" |   {--info,--no-info}"[show additional package information (default\: \%(default)s)]:info:" | ||||||
|   "--sort-by[sort field by this field. In case if two packages have the same value of the specified field, they will be always sorted by name (default\: name)]:sort_by:(description first_submitted id last_modified maintainer name num_votes out_of_date package_base package_base_id popularity repository submitter url url_path version)" |   "--sort-by[sort field by this field. In case if two packages have the same value of the specified field, they will be always sorted by name]:sort_by:(description first_submitted id last_modified maintainer name num_votes out_of_date package_base package_base_id popularity repository submitter url url_path version)" | ||||||
|   "(*):search terms, can be specified multiple times, the result will match all terms:" |   "(*):search terms, can be specified multiple times, the result will match all terms:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_check_options=( | _shtab_ahriman_check_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {-e,--exit-code}"[return non-zero exit status if result is empty (default\: False)]" |   {-e,--exit-code}"[return non-zero exit status if result is empty]" | ||||||
|   {--vcs,--no-vcs}"[fetch actual version of VCS packages (default\: True)]:vcs:" |   {--vcs,--no-vcs}"[fetch actual version of VCS packages (default\: \%(default)s)]:vcs:" | ||||||
|   "*"{-y,--refresh}"[download fresh package databases from the mirror before actions, -yy to force refresh even if up to date (default\: False)]" |   "*"{-y,--refresh}"[download fresh package databases from the mirror before actions, -yy to force refresh even if up to date]" | ||||||
|   "(*)::filter check by package base (default\: None):" |   "(*)::filter check by package base:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_clean_options=( | _shtab_ahriman_clean_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {--cache,--no-cache}"[clear directory with package caches (default\: False)]:cache:" |   {--cache,--no-cache}"[clear directory with package caches (default\: \%(default)s)]:cache:" | ||||||
|   {--chroot,--no-chroot}"[clear build chroot (default\: False)]:chroot:" |   {--chroot,--no-chroot}"[clear build chroot (default\: \%(default)s)]:chroot:" | ||||||
|   {--manual,--no-manual}"[clear manually added packages queue (default\: False)]:manual:" |   {--manual,--no-manual}"[clear manually added packages queue (default\: \%(default)s)]:manual:" | ||||||
|   {--packages,--no-packages}"[clear directory with built packages (default\: False)]:packages:" |   {--packages,--no-packages}"[clear directory with built packages (default\: \%(default)s)]:packages:" | ||||||
|   {--pacman,--no-pacman}"[clear directory with pacman local database cache (default\: False)]:pacman:" |   {--pacman,--no-pacman}"[clear directory with pacman local database cache (default\: \%(default)s)]:pacman:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_config_options=( | _shtab_ahriman_config_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {--secure,--no-secure}"[hide passwords and secrets from output (default\: True)]:secure:" |   {--secure,--no-secure}"[hide passwords and secrets from output (default\: \%(default)s)]:secure:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_config_validate_options=( | _shtab_ahriman_config_validate_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {-e,--exit-code}"[return non-zero exit status if configuration is invalid (default\: False)]" |   {-e,--exit-code}"[return non-zero exit status if configuration is invalid]" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_daemon_options=( | _shtab_ahriman_daemon_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {-i,--interval}"[interval between runs in seconds (default\: 43200)]:interval:" |   {-i,--interval}"[interval between runs in seconds]:interval:" | ||||||
|   {--aur,--no-aur}"[enable or disable checking for AUR updates (default\: True)]:aur:" |   {--aur,--no-aur}"[enable or disable checking for AUR updates (default\: \%(default)s)]:aur:" | ||||||
|   {--dependencies,--no-dependencies}"[process missing package dependencies (default\: True)]:dependencies:" |   {--dependencies,--no-dependencies}"[process missing package dependencies (default\: \%(default)s)]:dependencies:" | ||||||
|   {--local,--no-local}"[enable or disable checking of local packages for updates (default\: True)]:local:" |   {--local,--no-local}"[enable or disable checking of local packages for updates (default\: \%(default)s)]:local:" | ||||||
|   {--manual,--no-manual}"[include or exclude manual updates (default\: True)]:manual:" |   {--manual,--no-manual}"[include or exclude manual updates (default\: \%(default)s)]:manual:" | ||||||
|   {--vcs,--no-vcs}"[fetch actual version of VCS packages (default\: True)]:vcs:" |   {--vcs,--no-vcs}"[fetch actual version of VCS packages (default\: \%(default)s)]:vcs:" | ||||||
|   "*"{-y,--refresh}"[download fresh package databases from the mirror before actions, -yy to force refresh even if up to date (default\: False)]" |   "*"{-y,--refresh}"[download fresh package databases from the mirror before actions, -yy to force refresh even if up to date]" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_help_options=( | _shtab_ahriman_help_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   ":show help message for specific command (default\: None):" |   ":show help message for specific command:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_help_commands_unsafe_options=( | _shtab_ahriman_help_commands_unsafe_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "(*)::instead of showing commands, just test command line for unsafe subcommand and return 0 in case if command is safe and 1 otherwise (default\: None):" |   "--command[instead of showing commands, just test command line for unsafe subcommand and return 0 in case if command is safe and 1 otherwise]:command:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_help_updates_options=( | _shtab_ahriman_help_updates_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {-e,--exit-code}"[return non-zero exit code if updates available (default\: False)]" |   {-e,--exit-code}"[return non-zero exit code if updates available]" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_help_version_options=( | _shtab_ahriman_help_version_options=( | ||||||
| @ -168,37 +162,33 @@ _shtab_ahriman_help_version_options=( | |||||||
| 
 | 
 | ||||||
| _shtab_ahriman_init_options=( | _shtab_ahriman_init_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "--build-as-user[force makepkg user to the specific one (default\: None)]:build_as_user:" |   "--build-as-user[force makepkg user to the specific one]:build_as_user:" | ||||||
|   "--build-command[build command prefix (default\: ahriman)]:build_command:" |   "--build-command[build command prefix]:build_command:" | ||||||
|   "--from-configuration[path to default devtools pacman configuration (default\: \/usr\/share\/devtools\/pacman.conf.d\/extra.conf)]:from_configuration:" |   "--from-configuration[path to default devtools pacman configuration]:from_configuration:" | ||||||
|   {--generate-salt,--no-generate-salt}"[generate salt for user passwords (default\: False)]:generate_salt:" |   {--makeflags-jobs,--no-makeflags-jobs}"[append MAKEFLAGS variable with parallelism set to number of cores (default\: \%(default)s)]:makeflags_jobs:" | ||||||
|   {--makeflags-jobs,--no-makeflags-jobs}"[append MAKEFLAGS variable with parallelism set to number of cores (default\: True)]:makeflags_jobs:" |   "--mirror[use the specified explicitly mirror instead of including mirrorlist]:mirror:" | ||||||
|   "--mirror[use the specified explicitly mirror instead of including mirrorlist (default\: None)]:mirror:" |   {--multilib,--no-multilib}"[add or do not multilib repository (default\: \%(default)s)]:multilib:" | ||||||
|   {--multilib,--no-multilib}"[add or do not multilib repository (default\: True)]:multilib:" |   "--packager[packager name and email]:packager:" | ||||||
|   "--packager[packager name and email (default\: None)]:packager:" |   "--repository[repository name]:repository:" | ||||||
|   "--repository[repository name (default\: None)]:repository:" |   "--sign-key[sign key id]:sign_key:" | ||||||
|   "--server[server to be used for devtools. If none set, local files will be used (default\: None)]:server:" |   "*--sign-target[sign options]:sign_target:(disabled packages repository)" | ||||||
|   "--sign-key[sign key id (default\: None)]:sign_key:" |   "--web-port[port of the web service]:web_port:" | ||||||
|   "*--sign-target[sign options (default\: None)]:sign_target:(disabled packages repository)" |   "--web-unix-socket[path to unix socket used for interprocess communications]:web_unix_socket:" | ||||||
|   "--web-port[port of the web service (default\: None)]:web_port:" |  | ||||||
|   "--web-unix-socket[path to unix socket used for interprocess communications (default\: None)]:web_unix_socket:" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_key_import_options=( | _shtab_ahriman_key_import_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "--key-server[key server for key import (default\: keyserver.ubuntu.com)]:key_server:" |   "--key-server[key server for key import]:key_server:" | ||||||
|   ":PGP key to import from public server:" |   ":PGP key to import from public server:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_package_add_options=( | _shtab_ahriman_package_add_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {--dependencies,--no-dependencies}"[process missing package dependencies (default\: True)]:dependencies:" |   {--dependencies,--no-dependencies}"[process missing package dependencies (default\: \%(default)s)]:dependencies:" | ||||||
|   {-e,--exit-code}"[return non-zero exit status if result is empty (default\: False)]" |   {-e,--exit-code}"[return non-zero exit status if result is empty]" | ||||||
|   {--increment,--no-increment}"[increment package release (pkgrel) version on duplicate (default\: True)]:increment:" |   {-n,--now}"[run update function after]" | ||||||
|   {-n,--now}"[run update function after (default\: False)]" |   "*"{-y,--refresh}"[download fresh package databases from the mirror before actions, -yy to force refresh even if up to date]" | ||||||
|   "*"{-y,--refresh}"[download fresh package databases from the mirror before actions, -yy to force refresh even if up to date (default\: False)]" |   {-s,--source}"[explicitly specify the package source for this command]:source:(auto archive aur directory local remote repository)" | ||||||
|   {-s,--source}"[explicitly specify the package source for this command (default\: PackageSource.Auto)]:source:(auto archive aur directory local remote repository)" |  | ||||||
|   {-u,--username}"[build as user (default\: None)]:username:" |  | ||||||
|   "(*):package source (base name, path to local files, remote URL):" |   "(*):package source (base name, path to local files, remote URL):" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| @ -209,11 +199,11 @@ _shtab_ahriman_package_remove_options=( | |||||||
| 
 | 
 | ||||||
| _shtab_ahriman_package_status_options=( | _shtab_ahriman_package_status_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "--ahriman[get service status itself (default\: False)]" |   "--ahriman[get service status itself]" | ||||||
|   {-e,--exit-code}"[return non-zero exit status if result is empty (default\: False)]" |   {-e,--exit-code}"[return non-zero exit status if result is empty]" | ||||||
|   {--info,--no-info}"[show additional package information (default\: False)]:info:" |   {--info,--no-info}"[show additional package information (default\: \%(default)s)]:info:" | ||||||
|   {-s,--status}"[filter packages by status (default\: None)]:status:(unknown pending building failed success)" |   {-s,--status}"[filter packages by status]:status:(unknown pending building failed success)" | ||||||
|   "(*)::filter status by package base (default\: None):" |   "(*)::filter status by package base:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_package_status_remove_options=( | _shtab_ahriman_package_status_remove_options=( | ||||||
| @ -223,19 +213,17 @@ _shtab_ahriman_package_status_remove_options=( | |||||||
| 
 | 
 | ||||||
| _shtab_ahriman_package_status_update_options=( | _shtab_ahriman_package_status_update_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {-s,--status}"[new package build status (default\: BuildStatusEnum.Success)]:status:(unknown pending building failed success)" |   {-s,--status}"[new package build status]:status:(unknown pending building failed success)" | ||||||
|   "(*)::set status for specified packages. If no packages supplied, service status will be updated (default\: None):" |   "(*)::set status for specified packages. If no packages supplied, service status will be updated:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_package_update_options=( | _shtab_ahriman_package_update_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {--dependencies,--no-dependencies}"[process missing package dependencies (default\: True)]:dependencies:" |   {--dependencies,--no-dependencies}"[process missing package dependencies (default\: \%(default)s)]:dependencies:" | ||||||
|   {-e,--exit-code}"[return non-zero exit status if result is empty (default\: False)]" |   {-e,--exit-code}"[return non-zero exit status if result is empty]" | ||||||
|   {--increment,--no-increment}"[increment package release (pkgrel) version on duplicate (default\: True)]:increment:" |   {-n,--now}"[run update function after]" | ||||||
|   {-n,--now}"[run update function after (default\: False)]" |   "*"{-y,--refresh}"[download fresh package databases from the mirror before actions, -yy to force refresh even if up to date]" | ||||||
|   "*"{-y,--refresh}"[download fresh package databases from the mirror before actions, -yy to force refresh even if up to date (default\: False)]" |   {-s,--source}"[explicitly specify the package source for this command]:source:(auto archive aur directory local remote repository)" | ||||||
|   {-s,--source}"[explicitly specify the package source for this command (default\: PackageSource.Auto)]:source:(auto archive aur directory local remote repository)" |  | ||||||
|   {-u,--username}"[build as user (default\: None)]:username:" |  | ||||||
|   "(*):package source (base name, path to local files, remote URL):" |   "(*):package source (base name, path to local files, remote URL):" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| @ -243,37 +231,34 @@ _shtab_ahriman_patch_add_options=( | |||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   ":package base:" |   ":package base:" | ||||||
|   ":PKGBUILD variable or function name. If variable is a function, it must end with ():" |   ":PKGBUILD variable or function name. If variable is a function, it must end with ():" | ||||||
|   ":path to file which contains function or variable value. If not set, the value will be read from stdin (default\: None):" |   ":path to file which contains function or variable value. If not set, the value will be read from stdin:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_patch_list_options=( | _shtab_ahriman_patch_list_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {-e,--exit-code}"[return non-zero exit status if result is empty (default\: False)]" |   {-e,--exit-code}"[return non-zero exit status if result is empty]" | ||||||
|   "*"{-v,--variable}"[if set, show only patches for specified PKGBUILD variables (default\: None)]:variable:" |   "*"{-v,--variable}"[if set, show only patches for specified PKGBUILD variables]:variable:" | ||||||
|   ":package base (default\: None):" |   ":package base:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_patch_remove_options=( | _shtab_ahriman_patch_remove_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "*"{-v,--variable}"[should be used for single-function patches in case if you wold like to remove only specified PKGBUILD variables. In case if not set, it will remove all patches related to the package (default\: None)]:variable:" |   "*"{-v,--variable}"[should be used for single-function patches in case if you wold like to remove only specified PKGBUILD variables. In case if not set, it will remove all patches related to the package]:variable:" | ||||||
|   ":package base:" |   ":package base:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_patch_set_add_options=( | _shtab_ahriman_patch_set_add_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "*"{-t,--track}"[files which has to be tracked (default\: \[\'\*.diff\', \'\*.patch\'\])]:track:" |   "*"{-t,--track}"[files which has to be tracked]:track:" | ||||||
|   ":path to directory with changed files for patch addition\/update:" |   ":path to directory with changed files for patch addition\/update:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_rebuild_options=( | _shtab_ahriman_rebuild_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "*--depends-on[only rebuild packages that depend on specified packages (default\: None)]:depends_on:" |   "*--depends-on[only rebuild packages that depend on specified packages]:depends_on:" | ||||||
|   "--dry-run[just perform check for packages without rebuild process itself (default\: False)]" |   "--dry-run[just perform check for packages without rebuild process itself]" | ||||||
|   "--from-database[read packages from database instead of filesystem. This feature in particular is required in case if you would like to restore repository from another repository instance. Note, however, that in order to restore packages you need to have original ahriman instance run with web service and have run repo-update at least once. (default\: False)]" |   "--from-database[read packages from database instead of filesystem. This feature in particular is required in case if you would like to restore repository from another repository instance. Note, however, that in order to restore packages you need to have original ahriman instance run with web service and have run repo-update at least once.]" | ||||||
|   {--increment,--no-increment}"[increment package release (pkgrel) on duplicate (default\: True)]:increment:" |   {-e,--exit-code}"[return non-zero exit status if result is empty]" | ||||||
|   {-e,--exit-code}"[return non-zero exit status if result is empty (default\: False)]" |  | ||||||
|   {-s,--status}"[filter packages by status. Requires --from-database to be set (default\: None)]:status:(unknown pending building failed success)" |  | ||||||
|   {-u,--username}"[build as user (default\: None)]:username:" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_remove_options=( | _shtab_ahriman_remove_options=( | ||||||
| @ -283,7 +268,7 @@ _shtab_ahriman_remove_options=( | |||||||
| 
 | 
 | ||||||
| _shtab_ahriman_remove_unknown_options=( | _shtab_ahriman_remove_unknown_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "--dry-run[just perform check for packages without removal (default\: False)]" |   "--dry-run[just perform check for packages without removal]" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_repo_backup_options=( | _shtab_ahriman_repo_backup_options=( | ||||||
| @ -293,82 +278,69 @@ _shtab_ahriman_repo_backup_options=( | |||||||
| 
 | 
 | ||||||
| _shtab_ahriman_repo_check_options=( | _shtab_ahriman_repo_check_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {-e,--exit-code}"[return non-zero exit status if result is empty (default\: False)]" |   {-e,--exit-code}"[return non-zero exit status if result is empty]" | ||||||
|   {--vcs,--no-vcs}"[fetch actual version of VCS packages (default\: True)]:vcs:" |   {--vcs,--no-vcs}"[fetch actual version of VCS packages (default\: \%(default)s)]:vcs:" | ||||||
|   "*"{-y,--refresh}"[download fresh package databases from the mirror before actions, -yy to force refresh even if up to date (default\: False)]" |   "*"{-y,--refresh}"[download fresh package databases from the mirror before actions, -yy to force refresh even if up to date]" | ||||||
|   "(*)::filter check by package base (default\: None):" |   "(*)::filter check by package base:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_repo_clean_options=( | _shtab_ahriman_repo_clean_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {--cache,--no-cache}"[clear directory with package caches (default\: False)]:cache:" |   {--cache,--no-cache}"[clear directory with package caches (default\: \%(default)s)]:cache:" | ||||||
|   {--chroot,--no-chroot}"[clear build chroot (default\: False)]:chroot:" |   {--chroot,--no-chroot}"[clear build chroot (default\: \%(default)s)]:chroot:" | ||||||
|   {--manual,--no-manual}"[clear manually added packages queue (default\: False)]:manual:" |   {--manual,--no-manual}"[clear manually added packages queue (default\: \%(default)s)]:manual:" | ||||||
|   {--packages,--no-packages}"[clear directory with built packages (default\: False)]:packages:" |   {--packages,--no-packages}"[clear directory with built packages (default\: \%(default)s)]:packages:" | ||||||
|   {--pacman,--no-pacman}"[clear directory with pacman local database cache (default\: False)]:pacman:" |   {--pacman,--no-pacman}"[clear directory with pacman local database cache (default\: \%(default)s)]:pacman:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_repo_config_options=( | _shtab_ahriman_repo_config_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {--secure,--no-secure}"[hide passwords and secrets from output (default\: True)]:secure:" |   {--secure,--no-secure}"[hide passwords and secrets from output (default\: \%(default)s)]:secure:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_repo_config_validate_options=( | _shtab_ahriman_repo_config_validate_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {-e,--exit-code}"[return non-zero exit status if configuration is invalid (default\: False)]" |   {-e,--exit-code}"[return non-zero exit status if configuration is invalid]" | ||||||
| ) |  | ||||||
| 
 |  | ||||||
| _shtab_ahriman_repo_create_keyring_options=( |  | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |  | ||||||
| ) |  | ||||||
| 
 |  | ||||||
| _shtab_ahriman_repo_create_mirrorlist_options=( |  | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_repo_daemon_options=( | _shtab_ahriman_repo_daemon_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {-i,--interval}"[interval between runs in seconds (default\: 43200)]:interval:" |   {-i,--interval}"[interval between runs in seconds]:interval:" | ||||||
|   {--aur,--no-aur}"[enable or disable checking for AUR updates (default\: True)]:aur:" |   {--aur,--no-aur}"[enable or disable checking for AUR updates (default\: \%(default)s)]:aur:" | ||||||
|   {--dependencies,--no-dependencies}"[process missing package dependencies (default\: True)]:dependencies:" |   {--dependencies,--no-dependencies}"[process missing package dependencies (default\: \%(default)s)]:dependencies:" | ||||||
|   {--local,--no-local}"[enable or disable checking of local packages for updates (default\: True)]:local:" |   {--local,--no-local}"[enable or disable checking of local packages for updates (default\: \%(default)s)]:local:" | ||||||
|   {--manual,--no-manual}"[include or exclude manual updates (default\: True)]:manual:" |   {--manual,--no-manual}"[include or exclude manual updates (default\: \%(default)s)]:manual:" | ||||||
|   {--vcs,--no-vcs}"[fetch actual version of VCS packages (default\: True)]:vcs:" |   {--vcs,--no-vcs}"[fetch actual version of VCS packages (default\: \%(default)s)]:vcs:" | ||||||
|   "*"{-y,--refresh}"[download fresh package databases from the mirror before actions, -yy to force refresh even if up to date (default\: False)]" |   "*"{-y,--refresh}"[download fresh package databases from the mirror before actions, -yy to force refresh even if up to date]" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_repo_init_options=( | _shtab_ahriman_repo_init_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "--build-as-user[force makepkg user to the specific one (default\: None)]:build_as_user:" |   "--build-as-user[force makepkg user to the specific one]:build_as_user:" | ||||||
|   "--build-command[build command prefix (default\: ahriman)]:build_command:" |   "--build-command[build command prefix]:build_command:" | ||||||
|   "--from-configuration[path to default devtools pacman configuration (default\: \/usr\/share\/devtools\/pacman.conf.d\/extra.conf)]:from_configuration:" |   "--from-configuration[path to default devtools pacman configuration]:from_configuration:" | ||||||
|   {--generate-salt,--no-generate-salt}"[generate salt for user passwords (default\: False)]:generate_salt:" |   {--makeflags-jobs,--no-makeflags-jobs}"[append MAKEFLAGS variable with parallelism set to number of cores (default\: \%(default)s)]:makeflags_jobs:" | ||||||
|   {--makeflags-jobs,--no-makeflags-jobs}"[append MAKEFLAGS variable with parallelism set to number of cores (default\: True)]:makeflags_jobs:" |   "--mirror[use the specified explicitly mirror instead of including mirrorlist]:mirror:" | ||||||
|   "--mirror[use the specified explicitly mirror instead of including mirrorlist (default\: None)]:mirror:" |   {--multilib,--no-multilib}"[add or do not multilib repository (default\: \%(default)s)]:multilib:" | ||||||
|   {--multilib,--no-multilib}"[add or do not multilib repository (default\: True)]:multilib:" |   "--packager[packager name and email]:packager:" | ||||||
|   "--packager[packager name and email (default\: None)]:packager:" |   "--repository[repository name]:repository:" | ||||||
|   "--repository[repository name (default\: None)]:repository:" |   "--sign-key[sign key id]:sign_key:" | ||||||
|   "--server[server to be used for devtools. If none set, local files will be used (default\: None)]:server:" |   "*--sign-target[sign options]:sign_target:(disabled packages repository)" | ||||||
|   "--sign-key[sign key id (default\: None)]:sign_key:" |   "--web-port[port of the web service]:web_port:" | ||||||
|   "*--sign-target[sign options (default\: None)]:sign_target:(disabled packages repository)" |   "--web-unix-socket[path to unix socket used for interprocess communications]:web_unix_socket:" | ||||||
|   "--web-port[port of the web service (default\: None)]:web_port:" |  | ||||||
|   "--web-unix-socket[path to unix socket used for interprocess communications (default\: None)]:web_unix_socket:" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_repo_rebuild_options=( | _shtab_ahriman_repo_rebuild_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "*--depends-on[only rebuild packages that depend on specified packages (default\: None)]:depends_on:" |   "*--depends-on[only rebuild packages that depend on specified packages]:depends_on:" | ||||||
|   "--dry-run[just perform check for packages without rebuild process itself (default\: False)]" |   "--dry-run[just perform check for packages without rebuild process itself]" | ||||||
|   "--from-database[read packages from database instead of filesystem. This feature in particular is required in case if you would like to restore repository from another repository instance. Note, however, that in order to restore packages you need to have original ahriman instance run with web service and have run repo-update at least once. (default\: False)]" |   "--from-database[read packages from database instead of filesystem. This feature in particular is required in case if you would like to restore repository from another repository instance. Note, however, that in order to restore packages you need to have original ahriman instance run with web service and have run repo-update at least once.]" | ||||||
|   {--increment,--no-increment}"[increment package release (pkgrel) on duplicate (default\: True)]:increment:" |   {-e,--exit-code}"[return non-zero exit status if result is empty]" | ||||||
|   {-e,--exit-code}"[return non-zero exit status if result is empty (default\: False)]" |  | ||||||
|   {-s,--status}"[filter packages by status. Requires --from-database to be set (default\: None)]:status:(unknown pending building failed success)" |  | ||||||
|   {-u,--username}"[build as user (default\: None)]:username:" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_repo_remove_unknown_options=( | _shtab_ahriman_repo_remove_unknown_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "--dry-run[just perform check for packages without removal (default\: False)]" |   "--dry-run[just perform check for packages without removal]" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_repo_report_options=( | _shtab_ahriman_repo_report_options=( | ||||||
| @ -377,36 +349,34 @@ _shtab_ahriman_repo_report_options=( | |||||||
| 
 | 
 | ||||||
| _shtab_ahriman_repo_restore_options=( | _shtab_ahriman_repo_restore_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {-o,--output}"[root path of the extracted files (default\: \/)]:output:" |   {-o,--output}"[root path of the extracted files]:output:" | ||||||
|   ":path of the input archive:" |   ":path of the input archive:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_repo_setup_options=( | _shtab_ahriman_repo_setup_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "--build-as-user[force makepkg user to the specific one (default\: None)]:build_as_user:" |   "--build-as-user[force makepkg user to the specific one]:build_as_user:" | ||||||
|   "--build-command[build command prefix (default\: ahriman)]:build_command:" |   "--build-command[build command prefix]:build_command:" | ||||||
|   "--from-configuration[path to default devtools pacman configuration (default\: \/usr\/share\/devtools\/pacman.conf.d\/extra.conf)]:from_configuration:" |   "--from-configuration[path to default devtools pacman configuration]:from_configuration:" | ||||||
|   {--generate-salt,--no-generate-salt}"[generate salt for user passwords (default\: False)]:generate_salt:" |   {--makeflags-jobs,--no-makeflags-jobs}"[append MAKEFLAGS variable with parallelism set to number of cores (default\: \%(default)s)]:makeflags_jobs:" | ||||||
|   {--makeflags-jobs,--no-makeflags-jobs}"[append MAKEFLAGS variable with parallelism set to number of cores (default\: True)]:makeflags_jobs:" |   "--mirror[use the specified explicitly mirror instead of including mirrorlist]:mirror:" | ||||||
|   "--mirror[use the specified explicitly mirror instead of including mirrorlist (default\: None)]:mirror:" |   {--multilib,--no-multilib}"[add or do not multilib repository (default\: \%(default)s)]:multilib:" | ||||||
|   {--multilib,--no-multilib}"[add or do not multilib repository (default\: True)]:multilib:" |   "--packager[packager name and email]:packager:" | ||||||
|   "--packager[packager name and email (default\: None)]:packager:" |   "--repository[repository name]:repository:" | ||||||
|   "--repository[repository name (default\: None)]:repository:" |   "--sign-key[sign key id]:sign_key:" | ||||||
|   "--server[server to be used for devtools. If none set, local files will be used (default\: None)]:server:" |   "*--sign-target[sign options]:sign_target:(disabled packages repository)" | ||||||
|   "--sign-key[sign key id (default\: None)]:sign_key:" |   "--web-port[port of the web service]:web_port:" | ||||||
|   "*--sign-target[sign options (default\: None)]:sign_target:(disabled packages repository)" |   "--web-unix-socket[path to unix socket used for interprocess communications]:web_unix_socket:" | ||||||
|   "--web-port[port of the web service (default\: None)]:web_port:" |  | ||||||
|   "--web-unix-socket[path to unix socket used for interprocess communications (default\: None)]:web_unix_socket:" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_repo_sign_options=( | _shtab_ahriman_repo_sign_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "(*)::sign only specified packages (default\: None):" |   "(*)::sign only specified packages:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_repo_status_update_options=( | _shtab_ahriman_repo_status_update_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {-s,--status}"[new status (default\: BuildStatusEnum.Success)]:status:(unknown pending building failed success)" |   {-s,--status}"[new status]:status:(unknown pending building failed success)" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_repo_sync_options=( | _shtab_ahriman_repo_sync_options=( | ||||||
| @ -415,27 +385,24 @@ _shtab_ahriman_repo_sync_options=( | |||||||
| 
 | 
 | ||||||
| _shtab_ahriman_repo_tree_options=( | _shtab_ahriman_repo_tree_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {-p,--partitions}"[also divide packages by independent partitions (default\: 1)]:partitions:" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_repo_triggers_options=( | _shtab_ahriman_repo_triggers_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "(*)::instead of running all triggers as set by configuration, just process specified ones in order of mention (default\: None):" |   "(*)::instead of running all triggers as set by configuration, just process specified ones in order of mention:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_repo_update_options=( | _shtab_ahriman_repo_update_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {--aur,--no-aur}"[enable or disable checking for AUR updates (default\: True)]:aur:" |   {--aur,--no-aur}"[enable or disable checking for AUR updates (default\: \%(default)s)]:aur:" | ||||||
|   {--dependencies,--no-dependencies}"[process missing package dependencies (default\: True)]:dependencies:" |   {--dependencies,--no-dependencies}"[process missing package dependencies (default\: \%(default)s)]:dependencies:" | ||||||
|   "--dry-run[just perform check for updates, same as check command (default\: False)]" |   "--dry-run[just perform check for updates, same as check command]" | ||||||
|   {-e,--exit-code}"[return non-zero exit status if result is empty (default\: False)]" |   {-e,--exit-code}"[return non-zero exit status if result is empty]" | ||||||
|   {--increment,--no-increment}"[increment package release (pkgrel) on duplicate (default\: True)]:increment:" |   {--local,--no-local}"[enable or disable checking of local packages for updates (default\: \%(default)s)]:local:" | ||||||
|   {--local,--no-local}"[enable or disable checking of local packages for updates (default\: True)]:local:" |   {--manual,--no-manual}"[include or exclude manual updates (default\: \%(default)s)]:manual:" | ||||||
|   {--manual,--no-manual}"[include or exclude manual updates (default\: True)]:manual:" |   {--vcs,--no-vcs}"[fetch actual version of VCS packages (default\: \%(default)s)]:vcs:" | ||||||
|   {-u,--username}"[build as user (default\: None)]:username:" |   "*"{-y,--refresh}"[download fresh package databases from the mirror before actions, -yy to force refresh even if up to date]" | ||||||
|   {--vcs,--no-vcs}"[fetch actual version of VCS packages (default\: True)]:vcs:" |   "(*)::filter check by package base:" | ||||||
|   "*"{-y,--refresh}"[download fresh package databases from the mirror before actions, -yy to force refresh even if up to date (default\: False)]" |  | ||||||
|   "(*)::filter check by package base (default\: None):" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_report_options=( | _shtab_ahriman_report_options=( | ||||||
| @ -444,101 +411,97 @@ _shtab_ahriman_report_options=( | |||||||
| 
 | 
 | ||||||
| _shtab_ahriman_search_options=( | _shtab_ahriman_search_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {-e,--exit-code}"[return non-zero exit status if result is empty (default\: False)]" |   {-e,--exit-code}"[return non-zero exit status if result is empty]" | ||||||
|   {--info,--no-info}"[show additional package information (default\: False)]:info:" |   {--info,--no-info}"[show additional package information (default\: \%(default)s)]:info:" | ||||||
|   "--sort-by[sort field by this field. In case if two packages have the same value of the specified field, they will be always sorted by name (default\: name)]:sort_by:(description first_submitted id last_modified maintainer name num_votes out_of_date package_base package_base_id popularity repository submitter url url_path version)" |   "--sort-by[sort field by this field. In case if two packages have the same value of the specified field, they will be always sorted by name]:sort_by:(description first_submitted id last_modified maintainer name num_votes out_of_date package_base package_base_id popularity repository submitter url url_path version)" | ||||||
|   "(*):search terms, can be specified multiple times, the result will match all terms:" |   "(*):search terms, can be specified multiple times, the result will match all terms:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_service_clean_options=( | _shtab_ahriman_service_clean_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {--cache,--no-cache}"[clear directory with package caches (default\: False)]:cache:" |   {--cache,--no-cache}"[clear directory with package caches (default\: \%(default)s)]:cache:" | ||||||
|   {--chroot,--no-chroot}"[clear build chroot (default\: False)]:chroot:" |   {--chroot,--no-chroot}"[clear build chroot (default\: \%(default)s)]:chroot:" | ||||||
|   {--manual,--no-manual}"[clear manually added packages queue (default\: False)]:manual:" |   {--manual,--no-manual}"[clear manually added packages queue (default\: \%(default)s)]:manual:" | ||||||
|   {--packages,--no-packages}"[clear directory with built packages (default\: False)]:packages:" |   {--packages,--no-packages}"[clear directory with built packages (default\: \%(default)s)]:packages:" | ||||||
|   {--pacman,--no-pacman}"[clear directory with pacman local database cache (default\: False)]:pacman:" |   {--pacman,--no-pacman}"[clear directory with pacman local database cache (default\: \%(default)s)]:pacman:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_service_config_options=( | _shtab_ahriman_service_config_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {--secure,--no-secure}"[hide passwords and secrets from output (default\: True)]:secure:" |   {--secure,--no-secure}"[hide passwords and secrets from output (default\: \%(default)s)]:secure:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_service_config_validate_options=( | _shtab_ahriman_service_config_validate_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {-e,--exit-code}"[return non-zero exit status if configuration is invalid (default\: False)]" |   {-e,--exit-code}"[return non-zero exit status if configuration is invalid]" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_service_key_import_options=( | _shtab_ahriman_service_key_import_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "--key-server[key server for key import (default\: keyserver.ubuntu.com)]:key_server:" |   "--key-server[key server for key import]:key_server:" | ||||||
|   ":PGP key to import from public server:" |   ":PGP key to import from public server:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_service_setup_options=( | _shtab_ahriman_service_setup_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "--build-as-user[force makepkg user to the specific one (default\: None)]:build_as_user:" |   "--build-as-user[force makepkg user to the specific one]:build_as_user:" | ||||||
|   "--build-command[build command prefix (default\: ahriman)]:build_command:" |   "--build-command[build command prefix]:build_command:" | ||||||
|   "--from-configuration[path to default devtools pacman configuration (default\: \/usr\/share\/devtools\/pacman.conf.d\/extra.conf)]:from_configuration:" |   "--from-configuration[path to default devtools pacman configuration]:from_configuration:" | ||||||
|   {--generate-salt,--no-generate-salt}"[generate salt for user passwords (default\: False)]:generate_salt:" |   {--makeflags-jobs,--no-makeflags-jobs}"[append MAKEFLAGS variable with parallelism set to number of cores (default\: \%(default)s)]:makeflags_jobs:" | ||||||
|   {--makeflags-jobs,--no-makeflags-jobs}"[append MAKEFLAGS variable with parallelism set to number of cores (default\: True)]:makeflags_jobs:" |   "--mirror[use the specified explicitly mirror instead of including mirrorlist]:mirror:" | ||||||
|   "--mirror[use the specified explicitly mirror instead of including mirrorlist (default\: None)]:mirror:" |   {--multilib,--no-multilib}"[add or do not multilib repository (default\: \%(default)s)]:multilib:" | ||||||
|   {--multilib,--no-multilib}"[add or do not multilib repository (default\: True)]:multilib:" |   "--packager[packager name and email]:packager:" | ||||||
|   "--packager[packager name and email (default\: None)]:packager:" |   "--repository[repository name]:repository:" | ||||||
|   "--repository[repository name (default\: None)]:repository:" |   "--sign-key[sign key id]:sign_key:" | ||||||
|   "--server[server to be used for devtools. If none set, local files will be used (default\: None)]:server:" |   "*--sign-target[sign options]:sign_target:(disabled packages repository)" | ||||||
|   "--sign-key[sign key id (default\: None)]:sign_key:" |   "--web-port[port of the web service]:web_port:" | ||||||
|   "*--sign-target[sign options (default\: None)]:sign_target:(disabled packages repository)" |   "--web-unix-socket[path to unix socket used for interprocess communications]:web_unix_socket:" | ||||||
|   "--web-port[port of the web service (default\: None)]:web_port:" |  | ||||||
|   "--web-unix-socket[path to unix socket used for interprocess communications (default\: None)]:web_unix_socket:" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_service_shell_options=( | _shtab_ahriman_service_shell_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   ":instead of dropping into shell, just execute the specified code (default\: None):" |   ":instead of dropping into shell, just execute the specified code:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_setup_options=( | _shtab_ahriman_setup_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "--build-as-user[force makepkg user to the specific one (default\: None)]:build_as_user:" |   "--build-as-user[force makepkg user to the specific one]:build_as_user:" | ||||||
|   "--build-command[build command prefix (default\: ahriman)]:build_command:" |   "--build-command[build command prefix]:build_command:" | ||||||
|   "--from-configuration[path to default devtools pacman configuration (default\: \/usr\/share\/devtools\/pacman.conf.d\/extra.conf)]:from_configuration:" |   "--from-configuration[path to default devtools pacman configuration]:from_configuration:" | ||||||
|   {--generate-salt,--no-generate-salt}"[generate salt for user passwords (default\: False)]:generate_salt:" |   {--makeflags-jobs,--no-makeflags-jobs}"[append MAKEFLAGS variable with parallelism set to number of cores (default\: \%(default)s)]:makeflags_jobs:" | ||||||
|   {--makeflags-jobs,--no-makeflags-jobs}"[append MAKEFLAGS variable with parallelism set to number of cores (default\: True)]:makeflags_jobs:" |   "--mirror[use the specified explicitly mirror instead of including mirrorlist]:mirror:" | ||||||
|   "--mirror[use the specified explicitly mirror instead of including mirrorlist (default\: None)]:mirror:" |   {--multilib,--no-multilib}"[add or do not multilib repository (default\: \%(default)s)]:multilib:" | ||||||
|   {--multilib,--no-multilib}"[add or do not multilib repository (default\: True)]:multilib:" |   "--packager[packager name and email]:packager:" | ||||||
|   "--packager[packager name and email (default\: None)]:packager:" |   "--repository[repository name]:repository:" | ||||||
|   "--repository[repository name (default\: None)]:repository:" |   "--sign-key[sign key id]:sign_key:" | ||||||
|   "--server[server to be used for devtools. If none set, local files will be used (default\: None)]:server:" |   "*--sign-target[sign options]:sign_target:(disabled packages repository)" | ||||||
|   "--sign-key[sign key id (default\: None)]:sign_key:" |   "--web-port[port of the web service]:web_port:" | ||||||
|   "*--sign-target[sign options (default\: None)]:sign_target:(disabled packages repository)" |   "--web-unix-socket[path to unix socket used for interprocess communications]:web_unix_socket:" | ||||||
|   "--web-port[port of the web service (default\: None)]:web_port:" |  | ||||||
|   "--web-unix-socket[path to unix socket used for interprocess communications (default\: None)]:web_unix_socket:" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_shell_options=( | _shtab_ahriman_shell_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   ":instead of dropping into shell, just execute the specified code (default\: None):" |   ":instead of dropping into shell, just execute the specified code:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_sign_options=( | _shtab_ahriman_sign_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "(*)::sign only specified packages (default\: None):" |   "(*)::sign only specified packages:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_status_options=( | _shtab_ahriman_status_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "--ahriman[get service status itself (default\: False)]" |   "--ahriman[get service status itself]" | ||||||
|   {-e,--exit-code}"[return non-zero exit status if result is empty (default\: False)]" |   {-e,--exit-code}"[return non-zero exit status if result is empty]" | ||||||
|   {--info,--no-info}"[show additional package information (default\: False)]:info:" |   {--info,--no-info}"[show additional package information (default\: \%(default)s)]:info:" | ||||||
|   {-s,--status}"[filter packages by status (default\: None)]:status:(unknown pending building failed success)" |   {-s,--status}"[filter packages by status]:status:(unknown pending building failed success)" | ||||||
|   "(*)::filter status by package base (default\: None):" |   "(*)::filter status by package base:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_status_update_options=( | _shtab_ahriman_status_update_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {-s,--status}"[new package build status (default\: BuildStatusEnum.Success)]:status:(unknown pending building failed success)" |   {-s,--status}"[new package build status]:status:(unknown pending building failed success)" | ||||||
|   "(*)::set status for specified packages. If no packages supplied, service status will be updated (default\: None):" |   "(*)::set status for specified packages. If no packages supplied, service status will be updated:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_sync_options=( | _shtab_ahriman_sync_options=( | ||||||
| @ -547,33 +510,30 @@ _shtab_ahriman_sync_options=( | |||||||
| 
 | 
 | ||||||
| _shtab_ahriman_update_options=( | _shtab_ahriman_update_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {--aur,--no-aur}"[enable or disable checking for AUR updates (default\: True)]:aur:" |   {--aur,--no-aur}"[enable or disable checking for AUR updates (default\: \%(default)s)]:aur:" | ||||||
|   {--dependencies,--no-dependencies}"[process missing package dependencies (default\: True)]:dependencies:" |   {--dependencies,--no-dependencies}"[process missing package dependencies (default\: \%(default)s)]:dependencies:" | ||||||
|   "--dry-run[just perform check for updates, same as check command (default\: False)]" |   "--dry-run[just perform check for updates, same as check command]" | ||||||
|   {-e,--exit-code}"[return non-zero exit status if result is empty (default\: False)]" |   {-e,--exit-code}"[return non-zero exit status if result is empty]" | ||||||
|   {--increment,--no-increment}"[increment package release (pkgrel) on duplicate (default\: True)]:increment:" |   {--local,--no-local}"[enable or disable checking of local packages for updates (default\: \%(default)s)]:local:" | ||||||
|   {--local,--no-local}"[enable or disable checking of local packages for updates (default\: True)]:local:" |   {--manual,--no-manual}"[include or exclude manual updates (default\: \%(default)s)]:manual:" | ||||||
|   {--manual,--no-manual}"[include or exclude manual updates (default\: True)]:manual:" |   {--vcs,--no-vcs}"[fetch actual version of VCS packages (default\: \%(default)s)]:vcs:" | ||||||
|   {-u,--username}"[build as user (default\: None)]:username:" |   "*"{-y,--refresh}"[download fresh package databases from the mirror before actions, -yy to force refresh even if up to date]" | ||||||
|   {--vcs,--no-vcs}"[fetch actual version of VCS packages (default\: True)]:vcs:" |   "(*)::filter check by package base:" | ||||||
|   "*"{-y,--refresh}"[download fresh package databases from the mirror before actions, -yy to force refresh even if up to date (default\: False)]" |  | ||||||
|   "(*)::filter check by package base (default\: None):" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_user_add_options=( | _shtab_ahriman_user_add_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   "--key[optional PGP key used by this user. The private key must be imported (default\: None)]:key:" |   {-p,--password}"[user password. Blank password will be treated as empty password, which is in particular must be used for OAuth2 authorization type.]:password:" | ||||||
|   "--packager[optional packager id used for build process in form of \`Name Surname \<mail\@example.com\>\` (default\: None)]:packager:" |   {-r,--role}"[user access level]:role:(unauthorized read reporter full)" | ||||||
|   {-p,--password}"[user password. Blank password will be treated as empty password, which is in particular must be used for OAuth2 authorization type. (default\: None)]:password:" |   {-s,--secure}"[set file permissions to user-only]" | ||||||
|   {-r,--role}"[user access level (default\: UserAccess.Read)]:role:(unauthorized read reporter full)" |  | ||||||
|   ":username for web service:" |   ":username for web service:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_user_list_options=( | _shtab_ahriman_user_list_options=( | ||||||
|   "(- : *)"{-h,--help}"[show this help message and exit]" |   "(- : *)"{-h,--help}"[show this help message and exit]" | ||||||
|   {-e,--exit-code}"[return non-zero exit status if result is empty (default\: False)]" |   {-e,--exit-code}"[return non-zero exit status if result is empty]" | ||||||
|   {-r,--role}"[filter users by role (default\: None)]:role:(unauthorized read reporter full)" |   {-r,--role}"[filter users by role]:role:(unauthorized read reporter full)" | ||||||
|   ":filter users by username (default\: None):" |   ":filter users by username:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| _shtab_ahriman_user_remove_options=( | _shtab_ahriman_user_remove_options=( | ||||||
| @ -596,7 +556,7 @@ _shtab_ahriman() { | |||||||
|   if ((${_shtab_ahriman_options[(I)${(q)one_or_more}*]} + ${_shtab_ahriman_options[(I)${(q)remainder}*]} == 0)); then  # noqa: E501 |   if ((${_shtab_ahriman_options[(I)${(q)one_or_more}*]} + ${_shtab_ahriman_options[(I)${(q)remainder}*]} == 0)); then  # noqa: E501 | ||||||
|     _shtab_ahriman_options+=(': :_shtab_ahriman_commands' '*::: :->ahriman') |     _shtab_ahriman_options+=(': :_shtab_ahriman_commands' '*::: :->ahriman') | ||||||
|   fi |   fi | ||||||
|   _arguments -C -s $_shtab_ahriman_options |   _arguments -C $_shtab_ahriman_options | ||||||
| 
 | 
 | ||||||
|   case $state in |   case $state in | ||||||
|     ahriman) |     ahriman) | ||||||
| @ -604,72 +564,70 @@ _shtab_ahriman() { | |||||||
|       (( CURRENT += 1 )) |       (( CURRENT += 1 )) | ||||||
|       curcontext="${curcontext%:*:*}:_shtab_ahriman-$line[1]:" |       curcontext="${curcontext%:*:*}:_shtab_ahriman-$line[1]:" | ||||||
|       case $line[1] in |       case $line[1] in | ||||||
|         add) _arguments -C -s $_shtab_ahriman_add_options ;; |         add) _arguments -C $_shtab_ahriman_add_options ;; | ||||||
|         aur-search) _arguments -C -s $_shtab_ahriman_aur_search_options ;; |         aur-search) _arguments -C $_shtab_ahriman_aur_search_options ;; | ||||||
|         check) _arguments -C -s $_shtab_ahriman_check_options ;; |         check) _arguments -C $_shtab_ahriman_check_options ;; | ||||||
|         clean) _arguments -C -s $_shtab_ahriman_clean_options ;; |         clean) _arguments -C $_shtab_ahriman_clean_options ;; | ||||||
|         config) _arguments -C -s $_shtab_ahriman_config_options ;; |         config) _arguments -C $_shtab_ahriman_config_options ;; | ||||||
|         config-validate) _arguments -C -s $_shtab_ahriman_config_validate_options ;; |         config-validate) _arguments -C $_shtab_ahriman_config_validate_options ;; | ||||||
|         daemon) _arguments -C -s $_shtab_ahriman_daemon_options ;; |         daemon) _arguments -C $_shtab_ahriman_daemon_options ;; | ||||||
|         help) _arguments -C -s $_shtab_ahriman_help_options ;; |         help) _arguments -C $_shtab_ahriman_help_options ;; | ||||||
|         help-commands-unsafe) _arguments -C -s $_shtab_ahriman_help_commands_unsafe_options ;; |         help-commands-unsafe) _arguments -C $_shtab_ahriman_help_commands_unsafe_options ;; | ||||||
|         help-updates) _arguments -C -s $_shtab_ahriman_help_updates_options ;; |         help-updates) _arguments -C $_shtab_ahriman_help_updates_options ;; | ||||||
|         help-version) _arguments -C -s $_shtab_ahriman_help_version_options ;; |         help-version) _arguments -C $_shtab_ahriman_help_version_options ;; | ||||||
|         init) _arguments -C -s $_shtab_ahriman_init_options ;; |         init) _arguments -C $_shtab_ahriman_init_options ;; | ||||||
|         key-import) _arguments -C -s $_shtab_ahriman_key_import_options ;; |         key-import) _arguments -C $_shtab_ahriman_key_import_options ;; | ||||||
|         package-add) _arguments -C -s $_shtab_ahriman_package_add_options ;; |         package-add) _arguments -C $_shtab_ahriman_package_add_options ;; | ||||||
|         package-remove) _arguments -C -s $_shtab_ahriman_package_remove_options ;; |         package-remove) _arguments -C $_shtab_ahriman_package_remove_options ;; | ||||||
|         package-status) _arguments -C -s $_shtab_ahriman_package_status_options ;; |         package-status) _arguments -C $_shtab_ahriman_package_status_options ;; | ||||||
|         package-status-remove) _arguments -C -s $_shtab_ahriman_package_status_remove_options ;; |         package-status-remove) _arguments -C $_shtab_ahriman_package_status_remove_options ;; | ||||||
|         package-status-update) _arguments -C -s $_shtab_ahriman_package_status_update_options ;; |         package-status-update) _arguments -C $_shtab_ahriman_package_status_update_options ;; | ||||||
|         package-update) _arguments -C -s $_shtab_ahriman_package_update_options ;; |         package-update) _arguments -C $_shtab_ahriman_package_update_options ;; | ||||||
|         patch-add) _arguments -C -s $_shtab_ahriman_patch_add_options ;; |         patch-add) _arguments -C $_shtab_ahriman_patch_add_options ;; | ||||||
|         patch-list) _arguments -C -s $_shtab_ahriman_patch_list_options ;; |         patch-list) _arguments -C $_shtab_ahriman_patch_list_options ;; | ||||||
|         patch-remove) _arguments -C -s $_shtab_ahriman_patch_remove_options ;; |         patch-remove) _arguments -C $_shtab_ahriman_patch_remove_options ;; | ||||||
|         patch-set-add) _arguments -C -s $_shtab_ahriman_patch_set_add_options ;; |         patch-set-add) _arguments -C $_shtab_ahriman_patch_set_add_options ;; | ||||||
|         rebuild) _arguments -C -s $_shtab_ahriman_rebuild_options ;; |         rebuild) _arguments -C $_shtab_ahriman_rebuild_options ;; | ||||||
|         remove) _arguments -C -s $_shtab_ahriman_remove_options ;; |         remove) _arguments -C $_shtab_ahriman_remove_options ;; | ||||||
|         remove-unknown) _arguments -C -s $_shtab_ahriman_remove_unknown_options ;; |         remove-unknown) _arguments -C $_shtab_ahriman_remove_unknown_options ;; | ||||||
|         repo-backup) _arguments -C -s $_shtab_ahriman_repo_backup_options ;; |         repo-backup) _arguments -C $_shtab_ahriman_repo_backup_options ;; | ||||||
|         repo-check) _arguments -C -s $_shtab_ahriman_repo_check_options ;; |         repo-check) _arguments -C $_shtab_ahriman_repo_check_options ;; | ||||||
|         repo-clean) _arguments -C -s $_shtab_ahriman_repo_clean_options ;; |         repo-clean) _arguments -C $_shtab_ahriman_repo_clean_options ;; | ||||||
|         repo-config) _arguments -C -s $_shtab_ahriman_repo_config_options ;; |         repo-config) _arguments -C $_shtab_ahriman_repo_config_options ;; | ||||||
|         repo-config-validate) _arguments -C -s $_shtab_ahriman_repo_config_validate_options ;; |         repo-config-validate) _arguments -C $_shtab_ahriman_repo_config_validate_options ;; | ||||||
|         repo-create-keyring) _arguments -C -s $_shtab_ahriman_repo_create_keyring_options ;; |         repo-daemon) _arguments -C $_shtab_ahriman_repo_daemon_options ;; | ||||||
|         repo-create-mirrorlist) _arguments -C -s $_shtab_ahriman_repo_create_mirrorlist_options ;; |         repo-init) _arguments -C $_shtab_ahriman_repo_init_options ;; | ||||||
|         repo-daemon) _arguments -C -s $_shtab_ahriman_repo_daemon_options ;; |         repo-rebuild) _arguments -C $_shtab_ahriman_repo_rebuild_options ;; | ||||||
|         repo-init) _arguments -C -s $_shtab_ahriman_repo_init_options ;; |         repo-remove-unknown) _arguments -C $_shtab_ahriman_repo_remove_unknown_options ;; | ||||||
|         repo-rebuild) _arguments -C -s $_shtab_ahriman_repo_rebuild_options ;; |         repo-report) _arguments -C $_shtab_ahriman_repo_report_options ;; | ||||||
|         repo-remove-unknown) _arguments -C -s $_shtab_ahriman_repo_remove_unknown_options ;; |         repo-restore) _arguments -C $_shtab_ahriman_repo_restore_options ;; | ||||||
|         repo-report) _arguments -C -s $_shtab_ahriman_repo_report_options ;; |         repo-setup) _arguments -C $_shtab_ahriman_repo_setup_options ;; | ||||||
|         repo-restore) _arguments -C -s $_shtab_ahriman_repo_restore_options ;; |         repo-sign) _arguments -C $_shtab_ahriman_repo_sign_options ;; | ||||||
|         repo-setup) _arguments -C -s $_shtab_ahriman_repo_setup_options ;; |         repo-status-update) _arguments -C $_shtab_ahriman_repo_status_update_options ;; | ||||||
|         repo-sign) _arguments -C -s $_shtab_ahriman_repo_sign_options ;; |         repo-sync) _arguments -C $_shtab_ahriman_repo_sync_options ;; | ||||||
|         repo-status-update) _arguments -C -s $_shtab_ahriman_repo_status_update_options ;; |         repo-tree) _arguments -C $_shtab_ahriman_repo_tree_options ;; | ||||||
|         repo-sync) _arguments -C -s $_shtab_ahriman_repo_sync_options ;; |         repo-triggers) _arguments -C $_shtab_ahriman_repo_triggers_options ;; | ||||||
|         repo-tree) _arguments -C -s $_shtab_ahriman_repo_tree_options ;; |         repo-update) _arguments -C $_shtab_ahriman_repo_update_options ;; | ||||||
|         repo-triggers) _arguments -C -s $_shtab_ahriman_repo_triggers_options ;; |         report) _arguments -C $_shtab_ahriman_report_options ;; | ||||||
|         repo-update) _arguments -C -s $_shtab_ahriman_repo_update_options ;; |         search) _arguments -C $_shtab_ahriman_search_options ;; | ||||||
|         report) _arguments -C -s $_shtab_ahriman_report_options ;; |         service-clean) _arguments -C $_shtab_ahriman_service_clean_options ;; | ||||||
|         search) _arguments -C -s $_shtab_ahriman_search_options ;; |         service-config) _arguments -C $_shtab_ahriman_service_config_options ;; | ||||||
|         service-clean) _arguments -C -s $_shtab_ahriman_service_clean_options ;; |         service-config-validate) _arguments -C $_shtab_ahriman_service_config_validate_options ;; | ||||||
|         service-config) _arguments -C -s $_shtab_ahriman_service_config_options ;; |         service-key-import) _arguments -C $_shtab_ahriman_service_key_import_options ;; | ||||||
|         service-config-validate) _arguments -C -s $_shtab_ahriman_service_config_validate_options ;; |         service-setup) _arguments -C $_shtab_ahriman_service_setup_options ;; | ||||||
|         service-key-import) _arguments -C -s $_shtab_ahriman_service_key_import_options ;; |         service-shell) _arguments -C $_shtab_ahriman_service_shell_options ;; | ||||||
|         service-setup) _arguments -C -s $_shtab_ahriman_service_setup_options ;; |         setup) _arguments -C $_shtab_ahriman_setup_options ;; | ||||||
|         service-shell) _arguments -C -s $_shtab_ahriman_service_shell_options ;; |         shell) _arguments -C $_shtab_ahriman_shell_options ;; | ||||||
|         setup) _arguments -C -s $_shtab_ahriman_setup_options ;; |         sign) _arguments -C $_shtab_ahriman_sign_options ;; | ||||||
|         shell) _arguments -C -s $_shtab_ahriman_shell_options ;; |         status) _arguments -C $_shtab_ahriman_status_options ;; | ||||||
|         sign) _arguments -C -s $_shtab_ahriman_sign_options ;; |         status-update) _arguments -C $_shtab_ahriman_status_update_options ;; | ||||||
|         status) _arguments -C -s $_shtab_ahriman_status_options ;; |         sync) _arguments -C $_shtab_ahriman_sync_options ;; | ||||||
|         status-update) _arguments -C -s $_shtab_ahriman_status_update_options ;; |         update) _arguments -C $_shtab_ahriman_update_options ;; | ||||||
|         sync) _arguments -C -s $_shtab_ahriman_sync_options ;; |         user-add) _arguments -C $_shtab_ahriman_user_add_options ;; | ||||||
|         update) _arguments -C -s $_shtab_ahriman_update_options ;; |         user-list) _arguments -C $_shtab_ahriman_user_list_options ;; | ||||||
|         user-add) _arguments -C -s $_shtab_ahriman_user_add_options ;; |         user-remove) _arguments -C $_shtab_ahriman_user_remove_options ;; | ||||||
|         user-list) _arguments -C -s $_shtab_ahriman_user_list_options ;; |         version) _arguments -C $_shtab_ahriman_version_options ;; | ||||||
|         user-remove) _arguments -C -s $_shtab_ahriman_user_remove_options ;; |         web) _arguments -C $_shtab_ahriman_web_options ;; | ||||||
|         version) _arguments -C -s $_shtab_ahriman_version_options ;; |  | ||||||
|         web) _arguments -C -s $_shtab_ahriman_web_options ;; |  | ||||||
|       esac |       esac | ||||||
|   esac |   esac | ||||||
| } | } | ||||||
							
								
								
									
										15
									
								
								docs/conf.py
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								docs/conf.py
									
									
									
									
									
								
							| @ -14,8 +14,9 @@ import os | |||||||
| import sys | import sys | ||||||
|  |  | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
|  | from unittest import mock | ||||||
|  |  | ||||||
| from ahriman import __version__ | from ahriman.version import __version__ | ||||||
|  |  | ||||||
|  |  | ||||||
| basedir = Path(__file__).resolve().parent.parent / "src" | basedir = Path(__file__).resolve().parent.parent / "src" | ||||||
| @ -23,6 +24,13 @@ sys.path.insert(0, str(basedir)) | |||||||
|  |  | ||||||
| on_rtd = os.environ.get("READTHEDOCS", None) == "True" | on_rtd = os.environ.get("READTHEDOCS", None) == "True" | ||||||
|  |  | ||||||
|  | for module in ( | ||||||
|  |         "pyalpm", | ||||||
|  | ): | ||||||
|  |     if module in sys.modules: | ||||||
|  |         continue | ||||||
|  |     sys.modules[module] = mock.Mock() | ||||||
|  |  | ||||||
|  |  | ||||||
| # -- Project information ----------------------------------------------------- | # -- Project information ----------------------------------------------------- | ||||||
|  |  | ||||||
| @ -42,7 +50,6 @@ release = __version__ | |||||||
| extensions = [ | extensions = [ | ||||||
|     "sphinx.ext.autodoc", |     "sphinx.ext.autodoc", | ||||||
|     "sphinx.ext.napoleon", |     "sphinx.ext.napoleon", | ||||||
|     "sphinx_rtd_theme", |  | ||||||
|     "sphinxarg.ext", |     "sphinxarg.ext", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| @ -67,7 +74,7 @@ exclude_patterns = [] | |||||||
| # The theme to use for HTML and HTML Help pages.  See the documentation for | # The theme to use for HTML and HTML Help pages.  See the documentation for | ||||||
| # a list of builtin themes. | # a list of builtin themes. | ||||||
| # | # | ||||||
| html_theme = "sphinx_rtd_theme" | html_theme = "default" if on_rtd else "alabaster" | ||||||
|  |  | ||||||
| # Add any paths that contain custom static files (such as style sheets) here, | # Add any paths that contain custom static files (such as style sheets) here, | ||||||
| # relative to this directory. They are copied after the builtin static files, | # relative to this directory. They are copied after the builtin static files, | ||||||
| @ -85,8 +92,6 @@ autoclass_content = "both" | |||||||
|  |  | ||||||
| autodoc_member_order = "groupwise" | autodoc_member_order = "groupwise" | ||||||
|  |  | ||||||
| autodoc_mock_imports = ["pyalpm"] |  | ||||||
|  |  | ||||||
| autodoc_default_options = { | autodoc_default_options = { | ||||||
|     "no-undoc-members": True, |     "no-undoc-members": True, | ||||||
| } | } | ||||||
|  | |||||||
| @ -12,15 +12,6 @@ There are two variable types which have been added to default ones, they are pat | |||||||
|  |  | ||||||
| Path values, except for casting to ``pathlib.Path`` type, will be also expanded to absolute paths relative to the configuration path. E.g. if path is set to ``ahriman.ini.d/logging.ini`` and root configuration path is ``/etc/ahriman.ini``, the value will be expanded to ``/etc/ahriman.ini.d/logging.ini``. In order to disable path expand, use the full path, e.g. ``/etc/ahriman.ini.d/logging.ini``. | Path values, except for casting to ``pathlib.Path`` type, will be also expanded to absolute paths relative to the configuration path. E.g. if path is set to ``ahriman.ini.d/logging.ini`` and root configuration path is ``/etc/ahriman.ini``, the value will be expanded to ``/etc/ahriman.ini.d/logging.ini``. In order to disable path expand, use the full path, e.g. ``/etc/ahriman.ini.d/logging.ini``. | ||||||
|  |  | ||||||
| Configuration allows string interpolation from environment variables, e.g.: |  | ||||||
|  |  | ||||||
| .. code-block:: ini |  | ||||||
|  |  | ||||||
|    [section] |  | ||||||
|    key = $SECRET |  | ||||||
|  |  | ||||||
| will try to read value from ``SECRET`` environment variable. In case if the required environment variable wasn't found, it will keep original value (i.e. ``$SECRET`` in the example). Dollar sign can be set as ``$$``. |  | ||||||
|  |  | ||||||
| There is also additional subcommand which will allow to validate configuration and print found errors. In order to do so, run ``service-config-validate`` subcommand, e.g.: | There is also additional subcommand which will allow to validate configuration and print found errors. In order to do so, run ``service-config-validate`` subcommand, e.g.: | ||||||
|  |  | ||||||
| .. code-block:: shell | .. code-block:: shell | ||||||
| @ -39,8 +30,8 @@ Base configuration settings. | |||||||
| * ``logging`` - path to logging configuration, string, required. Check ``logging.ini`` for reference. | * ``logging`` - path to logging configuration, string, required. Check ``logging.ini`` for reference. | ||||||
| * ``suppress_http_log_errors`` - suppress http log errors, boolean, optional, default ``no``. If set to ``yes``, any http log errors (e.g. if web server is not available, but http logging is enabled) will be suppressed. | * ``suppress_http_log_errors`` - suppress http log errors, boolean, optional, default ``no``. If set to ``yes``, any http log errors (e.g. if web server is not available, but http logging is enabled) will be suppressed. | ||||||
|  |  | ||||||
| ``alpm:*`` groups | ``alpm`` group | ||||||
| ----------------- | -------------- | ||||||
|  |  | ||||||
| libalpm and AUR related configuration. Group name can refer to architecture, e.g. ``alpm:x86_64`` can be used for x86_64 architecture specific settings. | libalpm and AUR related configuration. Group name can refer to architecture, e.g. ``alpm:x86_64`` can be used for x86_64 architecture specific settings. | ||||||
|  |  | ||||||
| @ -63,7 +54,7 @@ Base authorization settings. ``OAuth`` provider requires ``aioauth-client`` libr | |||||||
| * ``max_age`` - parameter which controls both cookie expiration and token expiration inside the service, integer, optional, default is 7 days. | * ``max_age`` - parameter which controls both cookie expiration and token expiration inside the service, integer, optional, default is 7 days. | ||||||
| * ``oauth_provider`` - OAuth2 provider class name as is in ``aioauth-client`` (e.g. ``GoogleClient``, ``GithubClient`` etc), string, required in case if ``oauth`` is used. | * ``oauth_provider`` - OAuth2 provider class name as is in ``aioauth-client`` (e.g. ``GoogleClient``, ``GithubClient`` etc), string, required in case if ``oauth`` is used. | ||||||
| * ``oauth_scopes`` - scopes list for OAuth2 provider, which will allow retrieving user email (which is used for checking user permissions), e.g. ``https://www.googleapis.com/auth/userinfo.email`` for ``GoogleClient`` or ``user:email`` for ``GithubClient``, space separated list of strings, required in case if ``oauth`` is used. | * ``oauth_scopes`` - scopes list for OAuth2 provider, which will allow retrieving user email (which is used for checking user permissions), e.g. ``https://www.googleapis.com/auth/userinfo.email`` for ``GoogleClient`` or ``user:email`` for ``GithubClient``, space separated list of strings, required in case if ``oauth`` is used. | ||||||
| * ``salt`` - additional password hash salt, string, optional. | * ``salt`` - password hash salt, string, required in case if authorization enabled (automatically generated by ``user-add`` subcommand). | ||||||
|  |  | ||||||
| Authorized users are stored inside internal database, if any of external provides are used the password field for non-service users must be empty.  | Authorized users are stored inside internal database, if any of external provides are used the password field for non-service users must be empty.  | ||||||
|  |  | ||||||
| @ -72,13 +63,10 @@ Authorized users are stored inside internal database, if any of external provide | |||||||
|  |  | ||||||
| Build related configuration. Group name can refer to architecture, e.g. ``build:x86_64`` can be used for x86_64 architecture specific settings. | Build related configuration. Group name can refer to architecture, e.g. ``build:x86_64`` can be used for x86_64 architecture specific settings. | ||||||
|  |  | ||||||
| * ``archbuild_flags`` - additional flags passed to ``archbuild`` command, space separated list of strings, optional. |  | ||||||
| * ``build_command`` - default build command, string, required. | * ``build_command`` - default build command, string, required. | ||||||
|  | * ``build_flags`` - additional ``pkgctl`` flags, space separated list of strings, optional. | ||||||
| * ``ignore_packages`` - list packages to ignore during a regular update (manual update will still work), space separated list of strings, optional. | * ``ignore_packages`` - list packages to ignore during a regular update (manual update will still work), space separated list of strings, optional. | ||||||
| * ``makepkg_flags`` - additional flags passed to ``makepkg`` command, space separated list of strings, optional. |  | ||||||
| * ``makechrootpkg_flags`` - additional flags passed to ``makechrootpkg`` command, space separated list of strings, optional. |  | ||||||
| * ``triggers`` - list of ``ahriman.core.triggers.Trigger`` class implementation (e.g. ``ahriman.core.report.ReportTrigger ahriman.core.upload.UploadTrigger``) which will be loaded and run at the end of processing, space separated list of strings, optional. You can also specify triggers by their paths, e.g. ``/usr/lib/python3.10/site-packages/ahriman/core/report/report.py.ReportTrigger``. Triggers are run in the order of mention. | * ``triggers`` - list of ``ahriman.core.triggers.Trigger`` class implementation (e.g. ``ahriman.core.report.ReportTrigger ahriman.core.upload.UploadTrigger``) which will be loaded and run at the end of processing, space separated list of strings, optional. You can also specify triggers by their paths, e.g. ``/usr/lib/python3.10/site-packages/ahriman/core/report/report.py.ReportTrigger``. Triggers are run in the order of mention. | ||||||
| * ``triggers_known`` - optional list of ``ahriman.core.triggers.Trigger`` class implementations which are not run automatically and used only for trigger discovery and configuration validation. |  | ||||||
| * ``vcs_allowed_age`` - maximal age in seconds of the VCS packages before their version will be updated with its remote source, int, optional, default ``604800``. | * ``vcs_allowed_age`` - maximal age in seconds of the VCS packages before their version will be updated with its remote source, int, optional, default ``604800``. | ||||||
|  |  | ||||||
| ``repository`` group | ``repository`` group | ||||||
| @ -96,6 +84,7 @@ Settings for signing packages or repository. Group name can refer to architectur | |||||||
|  |  | ||||||
| * ``target`` - configuration flag to enable signing, space separated list of strings, required. Allowed values are ``package`` (sign each package separately), ``repository`` (sign repository database file). | * ``target`` - configuration flag to enable signing, space separated list of strings, required. Allowed values are ``package`` (sign each package separately), ``repository`` (sign repository database file). | ||||||
| * ``key`` - default PGP key, string, required. This key will also be used for database signing if enabled. | * ``key`` - default PGP key, string, required. This key will also be used for database signing if enabled. | ||||||
|  | * ``key_*`` settings - PGP key which will be used for specific packages, string, optional. For example, if there is ``key_yay`` option the specified key will be used for yay package and default key for others. | ||||||
|  |  | ||||||
| ``web:*`` groups | ``web:*`` groups | ||||||
| ---------------- | ---------------- | ||||||
| @ -106,56 +95,15 @@ Web server settings. If any of ``host``/``port`` is not set, web integration wil | |||||||
| * ``debug`` - enable debug toolbar, boolean, optional, default ``no``. | * ``debug`` - enable debug toolbar, boolean, optional, default ``no``. | ||||||
| * ``debug_check_host`` - check hosts to access debug toolbar, boolean, optional, default ``no``. | * ``debug_check_host`` - check hosts to access debug toolbar, boolean, optional, default ``no``. | ||||||
| * ``debug_allowed_hosts`` - allowed hosts to get access to debug toolbar, space separated list of string, optional. | * ``debug_allowed_hosts`` - allowed hosts to get access to debug toolbar, space separated list of string, optional. | ||||||
| * ``enable_archive_upload`` - allow to upload packages via HTTP (i.e. call of ``/api/v1/service/upload`` uri), boolean, optional, default ``no``. |  | ||||||
| * ``host`` - host to bind, string, optional. | * ``host`` - host to bind, string, optional. | ||||||
| * ``index_url`` - full url of the repository index page, string, optional. | * ``index_url`` - full url of the repository index page, string, optional. | ||||||
| * ``max_body_size`` - max body size in bytes to be validated for archive upload, integer, optional. If not set, validation will be disabled. |  | ||||||
| * ``password`` - password to authorize in web service in order to update service status, string, required in case if authorization enabled. | * ``password`` - password to authorize in web service in order to update service status, string, required in case if authorization enabled. | ||||||
| * ``port`` - port to bind, int, optional. | * ``port`` - port to bind, int, optional. | ||||||
| * ``static_path`` - path to directory with static files, string, required. | * ``static_path`` - path to directory with static files, string, required. | ||||||
| * ``templates`` - path to templates directory, string, required. | * ``templates`` - path to templates directory, string, required. | ||||||
| * ``timeout`` - HTTP request timeout in seconds, int, optional, default is ``30``. |  | ||||||
| * ``unix_socket`` - path to the listening unix socket, string, optional. If set, server will create the socket on the specified address which can (and will) be used by application. Note, that unlike usual host/port configuration, unix socket allows to perform requests without authorization. | * ``unix_socket`` - path to the listening unix socket, string, optional. If set, server will create the socket on the specified address which can (and will) be used by application. Note, that unlike usual host/port configuration, unix socket allows to perform requests without authorization. | ||||||
| * ``unix_socket_unsafe`` - set unsafe (o+w) permissions to unix socket, boolean, optional, default ``yes``. This option is enabled by default, because it is supposed that unix socket is created in safe environment (only web service is supposed to be used in unsafe), but it can be disabled by configuration. | * ``unix_socket_unsafe`` - set unsafe (o+w) permissions to unix socket, boolean, optional, default ``yes``. This option is enabled by default, because it is supposed that unix socket is created in safe environment (only web service is supposed to be used in unsafe), but it can be disabled by configuration. | ||||||
| * ``username`` - username to authorize in web service in order to update service status, string, required in case if authorization enabled. | * ``username`` - username to authorize in web service in order to update service status, string, required in case if authorization enabled. | ||||||
| * ``wait_timeout`` - wait timeout in seconds, maximum amount of time to be waited before lock will be free, int, optional. |  | ||||||
|  |  | ||||||
| ``keyring`` group |  | ||||||
| -------------------- |  | ||||||
|  |  | ||||||
| Keyring package generator plugin. |  | ||||||
|  |  | ||||||
| * ``target`` - list of generator settings sections, space separated list of strings, required. It must point to valid section name. |  | ||||||
|  |  | ||||||
| Keyring generator plugin |  | ||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^ |  | ||||||
|  |  | ||||||
| * ``type`` - type of the generator, string, optional, must be set to ``keyring-generator`` if exists. |  | ||||||
| * ``description`` - keyring package description, string, optional, default is ``repo PGP keyring``, where ``repo`` is the repository name. |  | ||||||
| * ``homepage`` - url to homepage location if any, string, optional. |  | ||||||
| * ``license`` - list of licenses which are applied to this package, space separated list of strings, optional, default is ``Unlicense``. |  | ||||||
| * ``package`` - keyring package name, string, optional, default is ``repo-keyring``, where ``repo`` is the repository name. |  | ||||||
| * ``packagers`` - list of packagers keys, space separated list of strings, optional, if not set, the ``key_*`` options from ``sign`` group will be used. |  | ||||||
| * ``revoked`` - list of revoked packagers keys, space separated list of strings, optional. |  | ||||||
| * ``trusted`` - list of master keys, space separated list of strings, optional, if not set, the ``key`` option from ``sign`` group will be used. |  | ||||||
|  |  | ||||||
| ``mirrorlist`` group |  | ||||||
| -------------------- |  | ||||||
|  |  | ||||||
| Mirrorlist package generator plugin. |  | ||||||
|  |  | ||||||
| * ``target`` - list of generator settings sections, space separated list of strings, required. It must point to valid section name. |  | ||||||
|  |  | ||||||
| Mirrorlist generator plugin |  | ||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^ |  | ||||||
|  |  | ||||||
| * ``type`` - type of the generator, string, optional, must be set to ``mirrorlist-generator`` if exists. |  | ||||||
| * ``description`` - mirrorlist package description, string, optional, default is ``repo mirror list for use by pacman``, where ``repo`` is the repository name. |  | ||||||
| * ``homepage`` - url to homepage location if any, string, optional. |  | ||||||
| * ``license`` - list of licenses which are applied to this package, space separated list of strings, optional, default is ``Unlicense``. |  | ||||||
| * ``package`` - mirrorlist package name, string, optional, default is ``repo-mirrorlist``, where ``repo`` is the repository name. |  | ||||||
| * ``path`` - absolute path to generated mirrorlist file, string, optional, default is ``/etc/pacman.d/repo-mirrorlist``, where ``repo`` is the repository name. |  | ||||||
| * ``servers`` - list of repository mirrors, space separated list of strings, required. |  | ||||||
|  |  | ||||||
| ``remote-pull`` group | ``remote-pull`` group | ||||||
| --------------------- | --------------------- | ||||||
| @ -194,8 +142,7 @@ Available options are: | |||||||
| Remote push trigger | Remote push trigger | ||||||
| ^^^^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^^^^ | ||||||
|  |  | ||||||
| * ``commit_email`` - git commit email, string, optional, default is ``ahriman@localhost``. | * ``commit_author`` - git commit author, string, optional. In case if not set, the git will generate author for you. Note, however, that in this case it will disclosure your hostname. | ||||||
| * ``commit_user`` - git commit user, string, optional, default is ``ahriman``. |  | ||||||
| * ``push_url`` - url of the remote repository to which PKGBUILDs should be pushed after build process, string, required. | * ``push_url`` - url of the remote repository to which PKGBUILDs should be pushed after build process, string, required. | ||||||
| * ``push_branch`` - branch of the remote repository to which PKGBUILDs should be pushed after build process, string, optional, default is ``master``. | * ``push_branch`` - branch of the remote repository to which PKGBUILDs should be pushed after build process, string, optional, default is ``master``. | ||||||
|  |  | ||||||
| @ -250,17 +197,6 @@ Section name must be either ``html`` (plus optional architecture name, e.g. ``ht | |||||||
| * ``path`` - path to html report file, string, required. | * ``path`` - path to html report file, string, required. | ||||||
| * ``template_path`` - path to Jinja2 template, string, required. | * ``template_path`` - path to Jinja2 template, string, required. | ||||||
|  |  | ||||||
| ``remote-call`` type |  | ||||||
| ^^^^^^^^^^^^^^^^^^^^ |  | ||||||
|  |  | ||||||
| Section name must be either ``remote-call`` (plus optional architecture name, e.g. ``remote-call:x86_64``) or random name with ``type`` set. |  | ||||||
|  |  | ||||||
| * ``type`` - type of the report, string, optional, must be set to ``remote-call`` if exists. |  | ||||||
| * ``aur`` - check for AUR packages updates, boolean, optional, default ``no``. |  | ||||||
| * ``local`` - check for local packages updates, boolean, optional, default ``no``. |  | ||||||
| * ``manual`` - update manually built packages, boolean, optional, default ``no``. |  | ||||||
| * ``wait_timeout`` - maximum amount of time in seconds to be waited before remote process will be terminated, int, optional, default ``-1``. |  | ||||||
|  |  | ||||||
| ``telegram`` type | ``telegram`` type | ||||||
| ^^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^^ | ||||||
|  |  | ||||||
| @ -306,14 +242,6 @@ This feature requires Github key creation (see below). Section name must be eith | |||||||
| * ``timeout`` - HTTP request timeout in seconds, int, optional, default is ``30``. | * ``timeout`` - HTTP request timeout in seconds, int, optional, default is ``30``. | ||||||
| * ``username`` - Github authorization user, string, required. Basically the same as ``owner``. | * ``username`` - Github authorization user, string, required. Basically the same as ``owner``. | ||||||
|  |  | ||||||
| ``remote-service`` type |  | ||||||
| ^^^^^^^^^^^^^^^^^^^^^^^ |  | ||||||
|  |  | ||||||
| Section name must be either ``remote-service`` (plus optional architecture name, e.g. ``remote-service:x86_64``) or random name with ``type`` set. |  | ||||||
|  |  | ||||||
| * ``type`` - type of the report, string, optional, must be set to ``remote-service`` if exists. |  | ||||||
| * ``timeout`` - HTTP request timeout in seconds, int, optional, default is ``30``. |  | ||||||
|  |  | ||||||
| ``rsync`` type | ``rsync`` type | ||||||
| ^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^ | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										314
									
								
								docs/faq.rst
									
									
									
									
									
								
							
							
						
						
									
										314
									
								
								docs/faq.rst
									
									
									
									
									
								
							| @ -1,3 +1,4 @@ | |||||||
|  |  | ||||||
| FAQ | FAQ | ||||||
| === | === | ||||||
|  |  | ||||||
| @ -305,7 +306,7 @@ TL;DR | |||||||
|  |  | ||||||
|    sudo -u ahriman ahriman repo-rebuild --depends-on python |    sudo -u ahriman ahriman repo-rebuild --depends-on python | ||||||
|  |  | ||||||
| You can even rebuild the whole repository (which is particular useful in case if you would like to change packager) if you do not supply ``--depends-on`` option. This action will automatically increment ``pkgrel`` value; in case if you don't want to, the ``--no-increment`` option has to be supplied. | You can even rebuild the whole repository (which is particular useful in case if you would like to change packager) if you do not supply ``--depends-on`` option. | ||||||
|  |  | ||||||
| However, note that you do not need to rebuild repository in case if you just changed signing option, just use ``repo-sign`` command instead.  | However, note that you do not need to rebuild repository in case if you just changed signing option, just use ``repo-sign`` command instead.  | ||||||
|  |  | ||||||
| @ -391,12 +392,11 @@ The following environment variables are supported: | |||||||
| * ``AHRIMAN_FORCE_ROOT`` - force run ahriman as root instead of guessing by subcommand. | * ``AHRIMAN_FORCE_ROOT`` - force run ahriman as root instead of guessing by subcommand. | ||||||
| * ``AHRIMAN_HOST`` - host for the web interface, default is ``0.0.0.0``. | * ``AHRIMAN_HOST`` - host for the web interface, default is ``0.0.0.0``. | ||||||
| * ``AHRIMAN_MULTILIB`` - if set (default) multilib repository will be used, disabled otherwise. | * ``AHRIMAN_MULTILIB`` - if set (default) multilib repository will be used, disabled otherwise. | ||||||
| * ``AHRIMAN_OUTPUT`` - controls logging handler, e.g. ``syslog``, ``console``. The name must be found in logging configuration. Note that if ``syslog`` handler is used you will need to mount ``/dev/log`` inside container because it is not available there. | * ``AHRIMAN_OUTPUT`` - controls logging handler, e.g. ``syslog``, ``console``. The name must be found in logging configuration. Note that if ``syslog`` (the default) handler is used you will need to mount ``/dev/log`` inside container because it is not available there. | ||||||
| * ``AHRIMAN_PACKAGER`` - packager name from which packages will be built, default is ``ahriman bot <ahriman@example.com>``. | * ``AHRIMAN_PACKAGER`` - packager name from which packages will be built, default is ``ahriman bot <ahriman@example.com>``. | ||||||
| * ``AHRIMAN_PACMAN_MIRROR`` - override pacman mirror server if set. | * ``AHRIMAN_PACMAN_MIRROR`` - override pacman mirror server if set. | ||||||
| * ``AHRIMAN_PORT`` - HTTP server port if any, default is empty. | * ``AHRIMAN_PORT`` - HTTP server port if any, default is empty. | ||||||
| * ``AHRIMAN_REPOSITORY`` - repository name, default is ``aur-clone``. | * ``AHRIMAN_REPOSITORY`` - repository name, default is ``aur-clone``. | ||||||
| * ``AHRIMAN_REPOSITORY_SERVER`` - optional override for the repository url. Useful if you would like to download packages from remote instead of local filesystem. |  | ||||||
| * ``AHRIMAN_REPOSITORY_ROOT`` - repository root. Because of filesystem rights it is required to override default repository root. By default, it uses ``ahriman`` directory inside ahriman's home, which can be passed as mount volume. | * ``AHRIMAN_REPOSITORY_ROOT`` - repository root. Because of filesystem rights it is required to override default repository root. By default, it uses ``ahriman`` directory inside ahriman's home, which can be passed as mount volume. | ||||||
| * ``AHRIMAN_UNIX_SOCKET`` - full path to unix socket which is used by web server, default is empty. Note that more likely you would like to put it inside ``AHRIMAN_REPOSITORY_ROOT`` directory (e.g. ``/var/lib/ahriman/ahriman/ahriman-web.sock``) or to ``/tmp``. | * ``AHRIMAN_UNIX_SOCKET`` - full path to unix socket which is used by web server, default is empty. Note that more likely you would like to put it inside ``AHRIMAN_REPOSITORY_ROOT`` directory (e.g. ``/var/lib/ahriman/ahriman/ahriman-web.sock``) or to ``/tmp``. | ||||||
| * ``AHRIMAN_USER`` - ahriman user, usually must not be overwritten, default is ``ahriman``. | * ``AHRIMAN_USER`` - ahriman user, usually must not be overwritten, default is ``ahriman``. | ||||||
| @ -664,7 +664,7 @@ How to report by email | |||||||
|  |  | ||||||
|    .. code-block:: shell |    .. code-block:: shell | ||||||
|  |  | ||||||
|       yay -S --asdeps python-jinja |       yay -S python-jinja | ||||||
|  |  | ||||||
| #.  | #.  | ||||||
|    Configure the service: |    Configure the service: | ||||||
| @ -691,7 +691,7 @@ How to generate index page for S3 | |||||||
|  |  | ||||||
|    .. code-block:: shell |    .. code-block:: shell | ||||||
|  |  | ||||||
|       yay -S --asdeps python-jinja |       yay -S python-jinja | ||||||
|  |  | ||||||
| #.  | #.  | ||||||
|    Configure the service: |    Configure the service: | ||||||
| @ -715,7 +715,7 @@ How to post build report to telegram | |||||||
|  |  | ||||||
|    .. code-block:: shell |    .. code-block:: shell | ||||||
|  |  | ||||||
|       yay -S --asdeps python-jinja |       yay -S python-jinja | ||||||
|  |  | ||||||
| #.  | #.  | ||||||
|    Register bot in telegram. You can do it by talking with `@BotFather <https://t.me/botfather>`_. For more details please refer to `official documentation <https://core.telegram.org/bots>`_. |    Register bot in telegram. You can do it by talking with `@BotFather <https://t.me/botfather>`_. For more details please refer to `official documentation <https://core.telegram.org/bots>`_. | ||||||
| @ -723,6 +723,7 @@ How to post build report to telegram | |||||||
| #.  | #.  | ||||||
|    Optionally (if you want to post message in chat): |    Optionally (if you want to post message in chat): | ||||||
|  |  | ||||||
|  |  | ||||||
|    #. Create telegram channel.  |    #. Create telegram channel.  | ||||||
|    #. Invite your bot into the channel. |    #. Invite your bot into the channel. | ||||||
|    #. Make your channel public |    #. Make your channel public | ||||||
| @ -753,277 +754,6 @@ If you did everything fine you should receive the message with the next update. | |||||||
|  |  | ||||||
| (replace ``${CHAT_ID}`` and ``${API_KEY}`` with the values from configuration). | (replace ``${CHAT_ID}`` and ``${API_KEY}`` with the values from configuration). | ||||||
|  |  | ||||||
| Distributed builds |  | ||||||
| ------------------ |  | ||||||
|  |  | ||||||
| The service allows to run build on multiple machines and collect packages on main node. There are multiple ways to achieve it, this section describes officially supported methods. |  | ||||||
|  |  | ||||||
| Remote synchronization and remote server call |  | ||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ |  | ||||||
|  |  | ||||||
| This setup requires at least two instances of the service: |  | ||||||
|  |  | ||||||
| #. Web service (with opt-in authorization enabled), later will be referenced as ``master`` node. |  | ||||||
| #. Application instances responsible for build, later will be referenced as ``worker`` nodes. |  | ||||||
|  |  | ||||||
| In this example the following settings are assumed: |  | ||||||
|  |  | ||||||
| * Repository architecture is ``x86_64``. |  | ||||||
| * Master node address is ``master.example.com``. |  | ||||||
|  |  | ||||||
| Master node configuration |  | ||||||
| """"""""""""""""""""""""" |  | ||||||
|  |  | ||||||
| The only requirements for the master node is that API must be available for worker nodes to call (e.g. port must be exposed to internet, or local network in case of VPN, etc) and file upload must be enabled: |  | ||||||
|  |  | ||||||
| .. code-block:: ini |  | ||||||
|  |  | ||||||
|    [web] |  | ||||||
|    enable_archive_upload = yes |  | ||||||
|  |  | ||||||
| In addition, the following settings are recommended for the master node: |  | ||||||
|  |  | ||||||
| * |  | ||||||
|   As it has been mentioned above, it is recommended to enable authentication (see `How to enable basic authorization`_) and create system user which will be used later. Later this user (if any) will be referenced as ``worker-user``. |  | ||||||
|  |  | ||||||
| * |  | ||||||
|   In order to be able to spawn multiple processes at the same time, wait timeout must be configured: |  | ||||||
|  |  | ||||||
|   .. code-block:: ini |  | ||||||
|  |  | ||||||
|      [web] |  | ||||||
|      wait_timeout = 0 |  | ||||||
|  |  | ||||||
| Worker nodes configuration |  | ||||||
| """""""""""""""""""""""""" |  | ||||||
|  |  | ||||||
| #. |  | ||||||
|    First of all, in this setup you need to split your repository into chunks manually, e.g. if you have repository on master node with packages ``A``, ``B`` and ``C``, you need to split them between all available workers, as example: |  | ||||||
|  |  | ||||||
|    * Worker #1: ``A``. |  | ||||||
|    * Worker #2: ``B`` and ``C``. |  | ||||||
|  |  | ||||||
| #. |  | ||||||
|    Each worker must be configured to upload files to master node: |  | ||||||
|  |  | ||||||
|    .. code-block:: ini |  | ||||||
|  |  | ||||||
|       [upload] |  | ||||||
|       target = remote-service |  | ||||||
|  |  | ||||||
|       [remote-service] |  | ||||||
|  |  | ||||||
| #. |  | ||||||
|    Worker must be configured to access web on master node: |  | ||||||
|  |  | ||||||
|    .. code-block:: ini |  | ||||||
|  |  | ||||||
|       [web] |  | ||||||
|       address = master.example.com |  | ||||||
|       username = worker-user |  | ||||||
|       password = very-secure-password |  | ||||||
|  |  | ||||||
|    As it has been mentioned above, ``web.address`` must be available for workers. In case if unix socket is used, it can be passed as ``web.unix_socket`` variable as usual. Optional ``web.username``/``web.password`` can be supplied in case if authentication was enabled on master node. |  | ||||||
|  |  | ||||||
| #. |  | ||||||
|    Each worker must call master node on success: |  | ||||||
|  |  | ||||||
|    .. code-block:: ini |  | ||||||
|  |  | ||||||
|       [report] |  | ||||||
|       target = remote-call |  | ||||||
|  |  | ||||||
|       [remote-call] |  | ||||||
|       manual = yes |  | ||||||
|  |  | ||||||
|    After success synchronization (see above), the built packages will be put into directory, from which they will be read during manual update, thus ``remote-call.manual`` flag is required. |  | ||||||
|  |  | ||||||
| #. |  | ||||||
|    Change order of trigger runs. This step is required, because by default the report trigger is called before the upload trigger and we would like to achieve the opposite: |  | ||||||
|  |  | ||||||
|    .. code-block:: ini |  | ||||||
|  |  | ||||||
|       [build] |  | ||||||
|       triggers = ahriman.core.gitremote.RemotePullTrigger ahriman.core.upload.UploadTrigger ahriman.core.report.ReportTrigger ahriman.core.gitremote.RemotePushTrigger |  | ||||||
|  |  | ||||||
| In addition, the following settings are recommended for workers: |  | ||||||
|  |  | ||||||
| * |  | ||||||
|   You might want to wait until report trigger will be completed; in this case the following option must be set: |  | ||||||
|  |  | ||||||
|   .. code-block:: ini |  | ||||||
|  |  | ||||||
|      [remote-call] |  | ||||||
|      wait_timeout = 0 |  | ||||||
|  |  | ||||||
| Dependency management |  | ||||||
| """"""""""""""""""""" |  | ||||||
|  |  | ||||||
| By default worker nodes don't know anything about master nodes packages, thus it will try to build each dependency by its own. However, using ``AHRIMAN_REPOSITORY_SERVER`` docker variable (or ``--server`` flag for setup command), it is possible to specify address of the master node for devtools configuration. |  | ||||||
|  |  | ||||||
| Repository and packages signing |  | ||||||
| """"""""""""""""""""""""""""""" |  | ||||||
|  |  | ||||||
| You can sign packages on worker nodes and then signatures will be synced to master node. In order to do so, you need to configure worker node as following, e.g.: |  | ||||||
|  |  | ||||||
| .. code-block:: ini |  | ||||||
|  |  | ||||||
|    [sign] |  | ||||||
|    target = package |  | ||||||
|    key = 8BE91E5A773FB48AC05CC1EDBED105AED6246B39 |  | ||||||
|  |  | ||||||
| Note, however, that in this case, signatures will not be validated on master node and just will be copied to repository tree. |  | ||||||
|  |  | ||||||
| If you would like to sign only database files (aka repository sign), it has to be configured on master node only as usual, e.g.: |  | ||||||
|  |  | ||||||
| .. code-block:: ini |  | ||||||
|  |  | ||||||
|    [sign] |  | ||||||
|    target = repository |  | ||||||
|    key = 8BE91E5A773FB48AC05CC1EDBED105AED6246B39 |  | ||||||
|  |  | ||||||
| Double node minimal docker example |  | ||||||
| """""""""""""""""""""""""""""""""" |  | ||||||
|  |  | ||||||
| Master node config (``master.ini``) as: |  | ||||||
|  |  | ||||||
| .. code-block:: ini |  | ||||||
|  |  | ||||||
|    [auth] |  | ||||||
|    target = mapping |  | ||||||
|  |  | ||||||
|    [web] |  | ||||||
|    enable_archive_upload = yes |  | ||||||
|    wait_timeout = 0 |  | ||||||
|  |  | ||||||
|  |  | ||||||
| Command to run master node: |  | ||||||
|  |  | ||||||
| .. code-block:: shell |  | ||||||
|  |  | ||||||
|    docker run --privileged -p 8080:8080 -e AHRIMAN_PORT=8080 -v master.ini:/etc/ahriman.ini.d/overrides.ini arcan1s/ahriman:latest web |  | ||||||
|  |  | ||||||
| The user ``worker-user`` has been created additionally. Worker node config (``worker.ini``) as: |  | ||||||
|  |  | ||||||
| .. code-block:: ini |  | ||||||
|  |  | ||||||
|    [web] |  | ||||||
|    address = http://172.17.0.1:8080 |  | ||||||
|    username = worker-user |  | ||||||
|    password = very-secure-password |  | ||||||
|  |  | ||||||
|    [upload] |  | ||||||
|    target = remote-service |  | ||||||
|  |  | ||||||
|    [remote-service] |  | ||||||
|  |  | ||||||
|    [report] |  | ||||||
|    target = remote-call |  | ||||||
|  |  | ||||||
|    [remote-call] |  | ||||||
|    manual = yes |  | ||||||
|    wait_timeout = 0 |  | ||||||
|  |  | ||||||
|    [build] |  | ||||||
|    triggers = ahriman.core.gitremote.RemotePullTrigger ahriman.core.upload.UploadTrigger ahriman.core.report.ReportTrigger ahriman.core.gitremote.RemotePushTrigger |  | ||||||
|  |  | ||||||
| The address above (``http://172.17.0.1:8080``) is something available for worker container. |  | ||||||
|  |  | ||||||
| Command to run worker node: |  | ||||||
|  |  | ||||||
| .. code-block:: shell |  | ||||||
|  |  | ||||||
|    docker run --privileged -v worker.ini:/etc/ahriman.ini.d/overrides.ini -it arcan1s/ahriman:latest package-add arhiman --now |  | ||||||
|  |  | ||||||
| The command above will successfully build ``ahriman`` package, upload it on master node and, finally, will update master node repository. |  | ||||||
|  |  | ||||||
| Addition of new package and repository update |  | ||||||
| """"""""""""""""""""""""""""""""""""""""""""" |  | ||||||
|  |  | ||||||
| Just run on worker command as usual, the built packages will be automatically uploaded to master node. Note that automatic update process must be disabled on master node. |  | ||||||
|  |  | ||||||
| Package removal |  | ||||||
| """"""""""""""" |  | ||||||
|  |  | ||||||
| This action must be done in two steps: |  | ||||||
|  |  | ||||||
| #. Remove package on worker. |  | ||||||
| #. Remove package on master node. |  | ||||||
|  |  | ||||||
| Maintenance packages |  | ||||||
| -------------------- |  | ||||||
|  |  | ||||||
| Generate keyring package |  | ||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^ |  | ||||||
|  |  | ||||||
| The application provides special plugin which generates keyring package. This plugin heavily depends on ``sign`` group settings, however it is possible to override them. The minimal package can be generated in the following way: |  | ||||||
|  |  | ||||||
| #. |  | ||||||
|    Edit configuration: |  | ||||||
|  |  | ||||||
|    .. code-block:: ini |  | ||||||
|  |  | ||||||
|       [keyring] |  | ||||||
|       target = keyring-generator |  | ||||||
|  |  | ||||||
|    By default it will use ``sign.key`` as trusted key and all other keys as packagers ones. For all available options refer to :doc:`configuration <configuration>`. |  | ||||||
|  |  | ||||||
| #. |  | ||||||
|    Create package source files: |  | ||||||
|  |  | ||||||
|    .. code-block:: shell |  | ||||||
|  |  | ||||||
|       sudo -u ahriman ahriman repo-create-keyring |  | ||||||
|  |  | ||||||
|    This command will generate PKGBUILD, revoked and trusted listings and keyring itself and will register the package in database. |  | ||||||
|  |  | ||||||
| #. |  | ||||||
|    Build new package as usual: |  | ||||||
|  |  | ||||||
|    .. code-block:: shell |  | ||||||
|  |  | ||||||
|       sudo -u ahriman ahriman package-add aur-clone-keyring --source local --now |  | ||||||
|  |  | ||||||
|    where ``aur-clone`` is your repository name. |  | ||||||
|  |  | ||||||
| This plugin might have some issues, in case of any of them, kindly create `new issue <https://github.com/arcan1s/ahriman/issues/new/choose>`_. |  | ||||||
|  |  | ||||||
| Generate mirrorlist package |  | ||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^ |  | ||||||
|  |  | ||||||
| The application provides special plugin which generates mirrorlist package also. It is possible to distribute this package as usual later. The package can be generated in the following way: |  | ||||||
|  |  | ||||||
| #. |  | ||||||
|    Edit configuration: |  | ||||||
|  |  | ||||||
|    .. code-block:: ini |  | ||||||
|  |  | ||||||
|       [mirrorlist] |  | ||||||
|       target = mirrorlist-generator |  | ||||||
|  |  | ||||||
|       [mirrorlist-generator] |  | ||||||
|       servers = https://repo.example.com/$arch |  | ||||||
|  |  | ||||||
|    The ``mirrorlist-generator.servers`` must contain list of available mirrors, the ``$arch`` and ``$repo`` variables are supported. For more options kindly refer to :doc:`configuration <configuration>`. |  | ||||||
|  |  | ||||||
| #. |  | ||||||
|    Create package source files: |  | ||||||
|  |  | ||||||
|    .. code-block:: shell |  | ||||||
|  |  | ||||||
|       sudo -u ahriman ahriman repo-create-mirrorlist |  | ||||||
|  |  | ||||||
|    This command will generate PKGBUILD and mirrorlist file and will register the package in database. |  | ||||||
|  |  | ||||||
| #. |  | ||||||
|    Build new package as usual: |  | ||||||
|  |  | ||||||
|    .. code-block:: shell |  | ||||||
|  |  | ||||||
|       sudo -u ahriman ahriman package-add aur-clone-mirrorlist --source local --now |  | ||||||
|  |  | ||||||
|    where ``aur-clone`` is your repository name. |  | ||||||
|  |  | ||||||
| Web service | Web service | ||||||
| ----------- | ----------- | ||||||
|  |  | ||||||
| @ -1035,7 +765,7 @@ How to setup web service | |||||||
|  |  | ||||||
|    .. code-block:: shell |    .. code-block:: shell | ||||||
|  |  | ||||||
|       yay -S --asdeps python-aiohttp python-aiohttp-jinja2 python-aiohttp-apispec>=3.0.0 python-aiohttp-cors |       yay -S python-aiohttp python-aiohttp-jinja2 | ||||||
|  |  | ||||||
| #.  | #.  | ||||||
|    Configure service: |    Configure service: | ||||||
| @ -1056,18 +786,15 @@ How to enable basic authorization | |||||||
|  |  | ||||||
|    .. code-block:: shell |    .. code-block:: shell | ||||||
|  |  | ||||||
|       yay -S --asdeps python-aiohttp-security python-aiohttp-session python-cryptography |       yay -S python-aiohttp-security python-aiohttp-session python-cryptography | ||||||
|  |  | ||||||
| #.  | #.  | ||||||
|    Configure the service to enable authorization (``salt`` can be generated as any random string): |    Configure the service to enable authorization: | ||||||
|  |  | ||||||
|    .. code-block:: ini |    .. code-block:: ini | ||||||
|  |  | ||||||
|       [auth] |       [auth] | ||||||
|       target = configuration |       target = configuration | ||||||
|       salt = somerandomstring |  | ||||||
|  |  | ||||||
|    The ``salt`` parameter is optional, but recommended. |  | ||||||
|  |  | ||||||
| #. | #. | ||||||
|    In order to provide access for reporting from application instances you can (recommended way) use unix sockets by configuring the following (note, that it requires ``python-requests-unixsocket`` package to be installed): |    In order to provide access for reporting from application instances you can (recommended way) use unix sockets by configuring the following (note, that it requires ``python-requests-unixsocket`` package to be installed): | ||||||
| @ -1115,7 +842,7 @@ How to enable OAuth authorization | |||||||
|  |  | ||||||
|    .. code-block:: shell |    .. code-block:: shell | ||||||
|  |  | ||||||
|       yay -S --asdeps python-aiohttp-security python-aiohttp-session python-cryptography python-aioauth-client |       yay -S python-aiohttp-security python-aiohttp-session python-cryptography python-aioauth-client | ||||||
|  |  | ||||||
| #.  | #.  | ||||||
|    Configure the service: |    Configure the service: | ||||||
| @ -1133,7 +860,7 @@ How to enable OAuth authorization | |||||||
|    Configure ``oauth_provider`` and ``oauth_scopes`` in case if you would like to use different from Google provider. Scope must grant access to user email. ``web.address`` is required to make callback URL available from internet. |    Configure ``oauth_provider`` and ``oauth_scopes`` in case if you would like to use different from Google provider. Scope must grant access to user email. ``web.address`` is required to make callback URL available from internet. | ||||||
|  |  | ||||||
| #.  | #.  | ||||||
|    If you are not going to use unix socket, you also need to create service user (remember to set ``auth.salt`` option before if required): |    Create service user: | ||||||
|  |  | ||||||
|    .. code-block:: shell |    .. code-block:: shell | ||||||
|  |  | ||||||
| @ -1227,11 +954,6 @@ Don't know, haven't tried it. But it lacks of documentation at least. | |||||||
| * ``repoctl`` is able to store old packages. | * ``repoctl`` is able to store old packages. | ||||||
| * Ability to host repository from same command in ``repoctl`` vs external services (e.g. nginx) in ``ahriman``. | * Ability to host repository from same command in ``repoctl`` vs external services (e.g. nginx) in ``ahriman``. | ||||||
|  |  | ||||||
| `repod <https://gitlab.archlinux.org/archlinux/repod>`_ |  | ||||||
| """"""""""""""""""""""""""""""""""""""""""""""""""""""" |  | ||||||
|  |  | ||||||
| Official tool provided by distribution, has clean logic, but it is just a helper for ``repo-add``, e.g. it doesn't work with AUR and all packages builds have to be handled separately. |  | ||||||
|  |  | ||||||
| `repo-scripts <https://github.com/arcan1s/repo-scripts>`_ | `repo-scripts <https://github.com/arcan1s/repo-scripts>`_ | ||||||
| """"""""""""""""""""""""""""""""""""""""""""""""""""""""" | """"""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  |  | ||||||
| @ -1253,17 +975,13 @@ It is automation tools for ``repoctl`` mentioned above. Except for using shell i | |||||||
| How to check service logs | How to check service logs | ||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^^^^^^^^^^ | ||||||
|  |  | ||||||
| By default, the service writes logs to ``journald`` which can be accessed by using ``journalctl`` command (logs are written to the journal of the user under which command is run). In order to retrieve logs for the process you can use the following command: | By default, the service writes logs to ``/dev/log`` which can be accessed by using ``journalctl`` command (logs are written to the journal of the user under which command is run). | ||||||
|  |  | ||||||
|  | You can also edit configuration and forward logs to ``stderr``, just change ``handlers`` value, e.g.: | ||||||
|  |  | ||||||
| .. code-block:: shell | .. code-block:: shell | ||||||
|  |  | ||||||
|    sudo journalctl SYSLOG_IDENTIFIER=ahriman |    sed -i 's/handlers = syslog_handler/handlers = console_handler/g' /etc/ahriman.ini.d/logging.ini | ||||||
|  |  | ||||||
| You can also ask to forward logs to ``stderr``, just set ``--log-handler`` flag, e.g.: |  | ||||||
|  |  | ||||||
| .. code-block:: shell |  | ||||||
|  |  | ||||||
|    ahriman --log-handler console ... |  | ||||||
|  |  | ||||||
| You can even configure logging as you wish, but kindly refer to python ``logging`` module `configuration <https://docs.python.org/3/library/logging.config.html>`_. The application uses java concept to log messages, e.g. class ``Application`` imported from ``ahriman.application.application`` package will have logger called ``ahriman.application.application.Application``. In order to e.g. change logger name for whole application package it is possible to change values for ``ahriman.application`` package; thus editing ``ahriman`` logger configuration will change logging for whole application (unless there are overrides for another logger). | You can even configure logging as you wish, but kindly refer to python ``logging`` module `configuration <https://docs.python.org/3/library/logging.config.html>`_. The application uses java concept to log messages, e.g. class ``Application`` imported from ``ahriman.application.application`` package will have logger called ``ahriman.application.application.Application``. In order to e.g. change logger name for whole application package it is possible to change values for ``ahriman.application`` package; thus editing ``ahriman`` logger configuration will change logging for whole application (unless there are overrides for another logger). | ||||||
|  |  | ||||||
|  | |||||||
| @ -12,8 +12,7 @@ Features | |||||||
| * VCS packages support. | * VCS packages support. | ||||||
| * Official repository support. | * Official repository support. | ||||||
| * Ability to patch AUR packages and even create package from local PKGBUILDs. | * Ability to patch AUR packages and even create package from local PKGBUILDs. | ||||||
| * Various rebuild options with ability to automatically bump package version. | * Sign support with gpg (repository, package, per package settings). | ||||||
| * Sign support with gpg (repository, package), multiple packagers support. |  | ||||||
| * Triggers for repository updates, e.g. synchronization to remote services (rsync, s3 and github) and report generation (email, html, telegram). | * Triggers for repository updates, e.g. synchronization to remote services (rsync, s3 and github) and report generation (email, html, telegram). | ||||||
| * Repository status interface with optional authorization and control options. | * Repository status interface with optional authorization and control options. | ||||||
|  |  | ||||||
|  | |||||||
| @ -25,46 +25,31 @@ Initial setup | |||||||
|       Configure build tools (it is required for correct dependency management system): |       Configure build tools (it is required for correct dependency management system): | ||||||
|  |  | ||||||
|       #. |       #. | ||||||
|          Create build command (you can choose any name for command, basically it should be ``{name}-{arch}-build``): |          Create configuration file ``{repository}.conf``: | ||||||
|  |  | ||||||
|          .. code-block:: shell |          .. code-block:: shell | ||||||
|  |  | ||||||
|             ln -s /usr/bin/archbuild /usr/local/bin/ahriman-x86_64-build |             cp /usr/local/share/devtools-git-poc/pacman.conf.d/{extra,aur-clone}.conf | ||||||
|  |  | ||||||
|       #.  |  | ||||||
|          Create configuration file (same as previous ``{name}.conf``): |  | ||||||
|  |  | ||||||
|          .. code-block:: shell |  | ||||||
|  |  | ||||||
|             cp /usr/share/devtools/pacman.conf.d/{extra,ahriman}.conf |  | ||||||
|  |  | ||||||
|       #.  |       #.  | ||||||
|          Change configuration file, add your own repository, add multilib repository etc: |          Change configuration file, add your own repository, add multilib repository etc: | ||||||
|  |  | ||||||
|          .. code-block:: shell |          .. code-block:: shell | ||||||
|  |  | ||||||
|             echo '[multilib]' | tee -a /usr/share/devtools/pacman-ahriman.conf |             echo '[multilib]' | tee -a /usr/local/share/devtools-git-poc/pacman.conf.d/aur-clone.conf | ||||||
|             echo 'Include = /etc/pacman.d/mirrorlist' | tee -a /usr/share/devtools/pacman.conf.d/ahriman.conf |             echo 'Include = /etc/pacman.d/mirrorlist' | tee -a /usr/local/share/devtools-git-poc/pacman.conf.d/aur-clone.conf | ||||||
|  |  | ||||||
|             echo '[aur-clone]' | tee -a /usr/share/devtools/pacman-ahriman.conf |             echo '[aur-clone]' | tee -a /usr/local/share/devtools-git-poc/pacman.conf.d/aur-clone.conf | ||||||
|             echo 'SigLevel = Optional TrustAll' | tee -a /usr/share/devtools/pacman.conf.d/ahriman.conf |             echo 'SigLevel = Optional TrustAll' | tee -a /usr/local/share/devtools-git-poc/pacman.conf.d/aur-clone.conf | ||||||
|             echo 'Server = file:///var/lib/ahriman/repository/$arch' | tee -a /usr/share/devtools/pacman.conf.d/ahriman.conf |             echo 'Server = file:///var/lib/ahriman/repository/$arch' | tee -a /usr/local/share/devtools-git-poc/pacman.conf.d/aur-clone.conf | ||||||
|  |  | ||||||
|       #. |       #. | ||||||
|          Set ``build_command`` option to point to your command: |          Configure ``/etc/sudoers.d/ahriman`` to allow running devtools command without a password: | ||||||
|  |  | ||||||
|          .. code-block:: shell |          .. code-block:: shell | ||||||
|  |  | ||||||
|             echo '[build]' | tee -a /etc/ahriman.ini.d/build.ini |             echo 'Cmnd_Alias CARCHBUILD_CMD = /usr/bin/pkgctl build *' | tee -a /etc/sudoers.d/ahriman | ||||||
|             echo 'build_command = ahriman-x86_64-build' | tee -a /etc/ahriman.ini.d/build.ini |             echo 'ahriman ALL=(ALL) NOPASSWD: CARCHBUILD_CMD' | tee -a /etc/sudoers.d/ahriman | ||||||
|  |  | ||||||
|       #. |  | ||||||
|          Configure ``/etc/sudoers.d/ahriman`` to allow running command without a password: |  | ||||||
|  |  | ||||||
|          .. code-block:: shell |  | ||||||
|  |  | ||||||
|             echo 'Cmnd_Alias CARCHBUILD_CMD = /usr/local/bin/ahriman-x86_64-build *' | tee -a /etc/sudoers.d/ahriman |  | ||||||
|             echo 'ahriman ALL=(ALL) NOPASSWD:SETENV: CARCHBUILD_CMD' | tee -a /etc/sudoers.d/ahriman |  | ||||||
|             chmod 400 /etc/sudoers.d/ahriman |             chmod 400 /etc/sudoers.d/ahriman | ||||||
|  |  | ||||||
|       This command supports several arguments, kindly refer to its help message. |       This command supports several arguments, kindly refer to its help message. | ||||||
|  | |||||||
| @ -31,16 +31,6 @@ This trigger will be called right after build process (``on_result``). It will p | |||||||
|  |  | ||||||
| Trigger which can be used for reporting. It implements ``on_result`` method and thus being called on each build update and generates report (e.g. html, telegram etc) according to the current settings. | Trigger which can be used for reporting. It implements ``on_result`` method and thus being called on each build update and generates report (e.g. html, telegram etc) according to the current settings. | ||||||
|  |  | ||||||
| ``ahriman.core.support.KeyringTrigger`` |  | ||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ |  | ||||||
|  |  | ||||||
| Generator for keyring package. This trigger will extract keys from local keychain and pack them into keyring specific format. This trigger will generate sources including PKGBUILD, which can be used later for package building. |  | ||||||
|  |  | ||||||
| ``ahriman.core.support.MirrorlistTrigger`` |  | ||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ |  | ||||||
|  |  | ||||||
| Simple generator for mirrorlist package, based on the urls which were set by configuration. This trigger will generate sources including PKGBUILD, which can be used later for package building. |  | ||||||
|  |  | ||||||
| ``ahriman.core.upload.UploadTrigger`` | ``ahriman.core.upload.UploadTrigger`` | ||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | ||||||
|  |  | ||||||
| @ -83,7 +73,7 @@ The ``context`` also implements collection methods such as ``__iter__`` and ``__ | |||||||
| Trigger example | Trigger example | ||||||
| --------------- | --------------- | ||||||
|  |  | ||||||
| Lets consider example of reporting trigger (e.g. `slack <https://slack.com/>`_, which provides easy HTTP API for integration triggers). | Lets consider example of reporting trigger (e.g. `slack <https://slack.com/>`_, which provides easy HTTP API for integration triggers).gre | ||||||
|  |  | ||||||
| In order to post message to slack we will need a specific trigger url (something like ``https://hooks.slack.com/services/company_id/trigger_id``), channel (e.g. ``#archrepo``) and username (``repo-bot``). | In order to post message to slack we will need a specific trigger url (something like ``https://hooks.slack.com/services/company_id/trigger_id``), channel (e.g. ``#archrepo``) and username (``repo-bot``). | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										
											BIN
										
									
								
								github-logo.png
									
									
									
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								github-logo.png
									
									
									
									
									
								
							
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 62 KiB | 
| @ -1,14 +1,14 @@ | |||||||
| # Maintainer: Evgeniy Alekseev | # Maintainer: Evgeniy Alekseev | ||||||
|  |  | ||||||
| pkgname='ahriman' | pkgname='ahriman' | ||||||
| pkgver=2.11.0 | pkgver=2.8.0 | ||||||
| pkgrel=1 | pkgrel=1 | ||||||
| pkgdesc="ArcH linux ReposItory MANager" | pkgdesc="ArcH linux ReposItory MANager" | ||||||
| arch=('any') | arch=('any') | ||||||
| url="https://github.com/arcan1s/ahriman" | url="https://github.com/arcan1s/ahriman" | ||||||
| license=('GPL3') | license=('GPL3') | ||||||
| depends=('devtools>=1:1.0.0' 'git' 'pyalpm' 'python-cerberus' 'python-inflection' 'python-passlib' 'python-requests' 'python-srcinfo') | depends=('devtools-git-poc' 'git' 'pyalpm' 'python-cerberus' 'python-inflection' 'python-passlib' 'python-requests' 'python-srcinfo') | ||||||
| makedepends=('python-build' 'python-flit' 'python-installer' 'python-wheel') | makedepends=('python-build' 'python-installer' 'python-wheel') | ||||||
| optdepends=('breezy: -bzr packages support' | optdepends=('breezy: -bzr packages support' | ||||||
|             'darcs: -darcs packages support' |             'darcs: -darcs packages support' | ||||||
|             'mercurial: -hg packages support' |             'mercurial: -hg packages support' | ||||||
| @ -24,13 +24,11 @@ optdepends=('breezy: -bzr packages support' | |||||||
|             'python-cryptography: web server with authorization' |             'python-cryptography: web server with authorization' | ||||||
|             'python-requests-unixsocket: client report to web server by unix socket' |             'python-requests-unixsocket: client report to web server by unix socket' | ||||||
|             'python-jinja: html report generation' |             'python-jinja: html report generation' | ||||||
|             'python-systemd: journal support' |  | ||||||
|             'rsync: sync by using rsync' |             'rsync: sync by using rsync' | ||||||
|             'subversion: -svn packages support') |             'subversion: -svn packages support') | ||||||
| source=("https://github.com/arcan1s/ahriman/releases/download/$pkgver/$pkgname-$pkgver-src.tar.xz" | source=("https://github.com/arcan1s/ahriman/releases/download/$pkgver/$pkgname-$pkgver-src.tar.xz" | ||||||
|         'ahriman.sysusers' |         'ahriman.sysusers' | ||||||
|         'ahriman.tmpfiles') |         'ahriman.tmpfiles') | ||||||
| install="$pkgname.install" |  | ||||||
| backup=('etc/ahriman.ini' | backup=('etc/ahriman.ini' | ||||||
|         'etc/ahriman.ini.d/logging.ini') |         'etc/ahriman.ini.d/logging.ini') | ||||||
|  |  | ||||||
| @ -45,10 +43,8 @@ package() { | |||||||
|  |  | ||||||
|   python -m installer --destdir="$pkgdir" "dist/$pkgname-$pkgver-py3-none-any.whl" |   python -m installer --destdir="$pkgdir" "dist/$pkgname-$pkgver-py3-none-any.whl" | ||||||
|  |  | ||||||
|   # thanks too PEP517, which we all wanted, you need to install data files manually nowadays |   # python-installer actually thinks that you cannot just copy files to root | ||||||
|   pushd package && find . -type f -exec install -Dm644 "{}" "$pkgdir/usr/{}" \; && popd |   # thus we need to copy them manually | ||||||
|  |  | ||||||
|   # keep usr/share configs as reference and copy them to /etc |  | ||||||
|   install -Dm644 "$pkgdir/usr/share/$pkgname/settings/ahriman.ini" "$pkgdir/etc/ahriman.ini" |   install -Dm644 "$pkgdir/usr/share/$pkgname/settings/ahriman.ini" "$pkgdir/etc/ahriman.ini" | ||||||
|   install -Dm644 "$pkgdir/usr/share/$pkgname/settings/ahriman.ini.d/logging.ini" "$pkgdir/etc/ahriman.ini.d/logging.ini" |   install -Dm644 "$pkgdir/usr/share/$pkgname/settings/ahriman.ini.d/logging.ini" "$pkgdir/etc/ahriman.ini.d/logging.ini" | ||||||
|  |  | ||||||
| @ -56,6 +52,6 @@ package() { | |||||||
|   install -Dm644 "$srcdir/$pkgname.tmpfiles" "$pkgdir/usr/lib/tmpfiles.d/$pkgname.conf" |   install -Dm644 "$srcdir/$pkgname.tmpfiles" "$pkgdir/usr/lib/tmpfiles.d/$pkgname.conf" | ||||||
| } | } | ||||||
|  |  | ||||||
| sha512sums=('19841842641520b573cdde6cb80a7cfcd69756d323fdfeebc2eee2d264a1325ead4ab2f8383bb369f7896bfc1de59d7358f133f4afeb90a9b9f0695f482a58d0' | sha512sums=('112b0d8aac68e5330bbdd2b86a59c8a9af8ab7a7c636489623c8460bb90f1318585851edd2a97a8ce20e2d2ad93b847b522685df707c190aa39d23ab908fa8ef' | ||||||
|             '53d37efec812afebf86281716259f9ea78a307b83897166c72777251c3eebcb587ecee375d907514781fb2a5c808cbb24ef9f3f244f12740155d0603bf213131' |             '53d37efec812afebf86281716259f9ea78a307b83897166c72777251c3eebcb587ecee375d907514781fb2a5c808cbb24ef9f3f244f12740155d0603bf213131' | ||||||
|             '62b2eccc352d33853ef243c9cddd63663014aa97b87242f1b5bc5099a7dbd69ff3821f24ffc58e1b7f2387bd4e9e9712cc4c67f661b1724ad99cdf09b3717794') |             '62b2eccc352d33853ef243c9cddd63663014aa97b87242f1b5bc5099a7dbd69ff3821f24ffc58e1b7f2387bd4e9e9712cc4c67f661b1724ad99cdf09b3717794') | ||||||
|  | |||||||
| @ -1,25 +0,0 @@ | |||||||
| post_upgrade() { |  | ||||||
|     local breakpoints=( |  | ||||||
|         2.9.0-1 |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     for v in "${breakpoints[@]}"; do |  | ||||||
|         if [[ $(vercmp "$v" "$2") -eq 1 ]]; then |  | ||||||
|             "_${v//[.-]/_}_changes" |  | ||||||
|         fi |  | ||||||
|     done |  | ||||||
| } |  | ||||||
|  |  | ||||||
| _2_9_0_1_changes() { |  | ||||||
|     cat << EOF |  | ||||||
| It was found that you were upgrading from old-devtools package to the new one, which requires manual intervention: |  | ||||||
|  |  | ||||||
| * make sure that devtools are upgraded to the latest release; |  | ||||||
| * merge service configuration if required; |  | ||||||
| * run setup command (i.e. sudo ahriman service-setup) with the same arguments as you did before; |  | ||||||
| * remove build chroot: sudo rm -r /var/lib/ahriman/chroot/ahriman-x86_64/; |  | ||||||
| * update local databases: sudo -u ahriman ahriman update --no-aur --no-local --no-manual -yy; |  | ||||||
|  |  | ||||||
| For more information kindly refer to changelog https://github.com/arcan1s/ahriman/releases/tag/2.9.0 |  | ||||||
| EOF |  | ||||||
| } |  | ||||||
| @ -1 +1,2 @@ | |||||||
| d /var/lib/ahriman 0755 ahriman ahriman | d /var/lib/ahriman 0755 ahriman ahriman | ||||||
|  | d /var/log/ahriman 0755 ahriman ahriman | ||||||
							
								
								
									
										3
									
								
								package/bin/ahriman
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										3
									
								
								package/bin/ahriman
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,3 @@ | |||||||
|  | #!/bin/sh | ||||||
|  |  | ||||||
|  | exec python -B -m ahriman.application.ahriman "$@" | ||||||
| @ -7,7 +7,7 @@ suppress_http_log_errors = yes | |||||||
| [alpm] | [alpm] | ||||||
| database = /var/lib/pacman | database = /var/lib/pacman | ||||||
| mirror = https://geo.mirror.pkgbuild.com/$repo/os/$arch | mirror = https://geo.mirror.pkgbuild.com/$repo/os/$arch | ||||||
| repositories = core extra multilib | repositories = core extra community multilib | ||||||
| root = / | root = / | ||||||
| use_ahriman_cache = yes | use_ahriman_cache = yes | ||||||
|  |  | ||||||
| @ -19,13 +19,10 @@ oauth_scopes = https://www.googleapis.com/auth/userinfo.email | |||||||
| allow_read_only = yes | allow_read_only = yes | ||||||
|  |  | ||||||
| [build] | [build] | ||||||
| archbuild_flags = | build_command = pkgctl | ||||||
| build_command = extra-x86_64-build |  | ||||||
| ignore_packages = | ignore_packages = | ||||||
| makechrootpkg_flags = | pkgctl_flags = | ||||||
| makepkg_flags = --nocolor --ignorearch |  | ||||||
| triggers = ahriman.core.gitremote.RemotePullTrigger ahriman.core.report.ReportTrigger ahriman.core.upload.UploadTrigger ahriman.core.gitremote.RemotePushTrigger | triggers = ahriman.core.gitremote.RemotePullTrigger ahriman.core.report.ReportTrigger ahriman.core.upload.UploadTrigger ahriman.core.gitremote.RemotePushTrigger | ||||||
| triggers_known = ahriman.core.gitremote.RemotePullTrigger ahriman.core.gitremote.RemotePushTrigger ahriman.core.report.ReportTrigger ahriman.core.upload.UploadTrigger ahriman.core.support.KeyringTrigger ahriman.core.support.MirrorlistTrigger |  | ||||||
| vcs_allowed_age = 604800 | vcs_allowed_age = 604800 | ||||||
|  |  | ||||||
| [repository] | [repository] | ||||||
| @ -35,12 +32,6 @@ root = /var/lib/ahriman | |||||||
| [sign] | [sign] | ||||||
| target = | target = | ||||||
|  |  | ||||||
| [keyring] |  | ||||||
| target = |  | ||||||
|  |  | ||||||
| [mirrorlist] |  | ||||||
| target = |  | ||||||
|  |  | ||||||
| [remote-pull] | [remote-pull] | ||||||
| target = | target = | ||||||
|  |  | ||||||
|  | |||||||
| @ -2,23 +2,17 @@ | |||||||
| keys = root,http,stderr,boto3,botocore,nose,s3transfer | keys = root,http,stderr,boto3,botocore,nose,s3transfer | ||||||
|  |  | ||||||
| [handlers] | [handlers] | ||||||
| keys = console_handler,journald_handler,syslog_handler | keys = console_handler,syslog_handler | ||||||
|  |  | ||||||
| [formatters] | [formatters] | ||||||
| keys = generic_format,syslog_format | keys = generic_format,syslog_format | ||||||
|  |  | ||||||
| [handler_console_handler] | [handler_console_handler] | ||||||
| class = logging.StreamHandler | class = StreamHandler | ||||||
| level = DEBUG | level = DEBUG | ||||||
| formatter = generic_format | formatter = generic_format | ||||||
| args = (sys.stderr,) | args = (sys.stderr,) | ||||||
|  |  | ||||||
| [handler_journald_handler] |  | ||||||
| class = ahriman.core.log.journal_handler.JournalHandler |  | ||||||
| level = DEBUG |  | ||||||
| formatter = syslog_format |  | ||||||
| kwargs = {"SYSLOG_IDENTIFIER": "ahriman"} |  | ||||||
|  |  | ||||||
| [handler_syslog_handler] | [handler_syslog_handler] | ||||||
| class = logging.handlers.SysLogHandler | class = logging.handlers.SysLogHandler | ||||||
| level = DEBUG | level = DEBUG | ||||||
| @ -27,16 +21,20 @@ args = ("/dev/log",) | |||||||
|  |  | ||||||
| [formatter_generic_format] | [formatter_generic_format] | ||||||
| format = [%(levelname)s %(asctime)s] [%(name)s]: %(message)s | format = [%(levelname)s %(asctime)s] [%(name)s]: %(message)s | ||||||
|  | datefmt = | ||||||
|  |  | ||||||
| [formatter_syslog_format] | [formatter_syslog_format] | ||||||
| format = [%(levelname)s] [%(name)s]: %(message)s | format = [%(levelname)s] [%(name)s]: %(message)s | ||||||
|  | datefmt = | ||||||
|  |  | ||||||
| [logger_root] | [logger_root] | ||||||
| level = DEBUG | level = DEBUG | ||||||
|  | handlers = syslog_handler | ||||||
| qualname = root | qualname = root | ||||||
|  |  | ||||||
| [logger_http] | [logger_http] | ||||||
| level = DEBUG | level = DEBUG | ||||||
|  | handlers = syslog_handler | ||||||
| qualname = http | qualname = http | ||||||
| propagate = 0 | propagate = 0 | ||||||
|  |  | ||||||
| @ -47,20 +45,24 @@ qualname = stderr | |||||||
|  |  | ||||||
| [logger_boto3] | [logger_boto3] | ||||||
| level = INFO | level = INFO | ||||||
|  | handlers = syslog_handler | ||||||
| qualname = boto3 | qualname = boto3 | ||||||
| propagate = 0 | propagate = 0 | ||||||
|  |  | ||||||
| [logger_botocore] | [logger_botocore] | ||||||
| level = INFO | level = INFO | ||||||
|  | handlers = syslog_handler | ||||||
| qualname = botocore | qualname = botocore | ||||||
| propagate = 0 | propagate = 0 | ||||||
|  |  | ||||||
| [logger_nose] | [logger_nose] | ||||||
| level = INFO | level = INFO | ||||||
|  | handlers = syslog_handler | ||||||
| qualname = nose | qualname = nose | ||||||
| propagate = 0 | propagate = 0 | ||||||
|  |  | ||||||
| [logger_s3transfer] | [logger_s3transfer] | ||||||
| level = INFO | level = INFO | ||||||
|  | handlers = syslog_handler | ||||||
| qualname = s3transfer | qualname = s3transfer | ||||||
| propagate = 0 | propagate = 0 | ||||||
|  | |||||||
| @ -63,8 +63,6 @@ | |||||||
|  |  | ||||||
|             <table id="packages" class="table table-striped table-hover" |             <table id="packages" class="table table-striped table-hover" | ||||||
|                    data-export-options='{"fileName": "packages"}' |                    data-export-options='{"fileName": "packages"}' | ||||||
|                    data-filter-control="true" |  | ||||||
|                    data-filter-control-visible="false" |  | ||||||
|                    data-page-list="[10, 25, 50, 100, all]" |                    data-page-list="[10, 25, 50, 100, all]" | ||||||
|                    data-page-size="10" |                    data-page-size="10" | ||||||
|                    data-pagination="true" |                    data-pagination="true" | ||||||
| @ -74,7 +72,6 @@ | |||||||
|                    data-show-columns-search="true" |                    data-show-columns-search="true" | ||||||
|                    data-show-columns-toggle-all="true" |                    data-show-columns-toggle-all="true" | ||||||
|                    data-show-export="true" |                    data-show-export="true" | ||||||
|                    data-show-filter-control-switch="true" |  | ||||||
|                    data-show-fullscreen="true" |                    data-show-fullscreen="true" | ||||||
|                    data-show-search-clear-button="true" |                    data-show-search-clear-button="true" | ||||||
|                    data-sortable="true" |                    data-sortable="true" | ||||||
| @ -85,14 +82,13 @@ | |||||||
|                 <thead class="table-primary"> |                 <thead class="table-primary"> | ||||||
|                     <tr> |                     <tr> | ||||||
|                         <th data-checkbox="true"></th> |                         <th data-checkbox="true"></th> | ||||||
|                         <th data-sortable="true" data-switchable="false" data-field="base" data-filter-control="input" data-filter-control-placeholder="(any base)">package base</th> |                         <th data-sortable="true" data-switchable="false" data-field="base">package base</th> | ||||||
|                         <th data-sortable="true" data-field="version" data-filter-control="input" data-filter-control-placeholder="(any version)">version</th> |                         <th data-sortable="true" data-field="version">version</th> | ||||||
|                         <th data-sortable="true" data-field="packages" data-filter-control="input" data-filter-control-placeholder="(any package)">packages</th> |                         <th data-sortable="true" data-field="packages">packages</th> | ||||||
|                         <th data-sortable="true" data-visible="false" data-field="groups" data-filter-control="select" data-filter-data="func:filterListGroups" data-filter-custom-search="filterList" data-filter-control-placeholder="(any group)">groups</th> |                         <th data-sortable="true" data-visible="false" data-field="groups">groups</th> | ||||||
|                         <th data-sortable="true" data-visible="false" data-field="licenses" data-filter-control="select" data-filter-data="func:filterListLicenses" data-filter-custom-search="filterList" data-filter-control-placeholder="(any license)">licenses</th> |                         <th data-sortable="true" data-visible="false" data-field="licenses">licenses</th> | ||||||
|                         <th data-sortable="true" data-visible="false" data-field="packager" data-filter-control="select" data-filter-custom-search="filterContains" data-filter-control-placeholder="(any packager)">packager</th> |                         <th data-sortable="true" data-field="timestamp">last update</th> | ||||||
|                         <th data-sortable="true" data-field="timestamp" data-filter-control="input" data-filter-custom-search="filterDateRange" data-filter-control-placeholder="(any date)">last update</th> |                         <th data-sortable="true" data-cell-style="statusFormat" data-field="status">status</th> | ||||||
|                         <th data-sortable="true" data-cell-style="statusFormat" data-field="status" data-filter-control="select" data-filter-control-placeholder="(any status)">status</th> |  | ||||||
|                     </tr> |                     </tr> | ||||||
|                 </thead> |                 </thead> | ||||||
|             </table> |             </table> | ||||||
|  | |||||||
| @ -84,7 +84,7 @@ | |||||||
|                     showSuccess("Success", `Key ${key} has been imported`); |                     showSuccess("Success", `Key ${key} has been imported`); | ||||||
|                 }, |                 }, | ||||||
|                 error: (jqXHR, _, errorThrown) => { |                 error: (jqXHR, _, errorThrown) => { | ||||||
|                     const message = _ => `Could not import key ${key} from ${server}`; |                     const message = _ => { return `Could not import key ${key} from ${server}`; }; | ||||||
|                     showFailure("Action failed", message, jqXHR, errorThrown); |                     showFailure("Action failed", message, jqXHR, errorThrown); | ||||||
|                 }, |                 }, | ||||||
|             }); |             }); | ||||||
|  | |||||||
| @ -60,8 +60,8 @@ | |||||||
|         const packages = packageInput.val(); |         const packages = packageInput.val(); | ||||||
|         if (packages) { |         if (packages) { | ||||||
|             packageAddModal.modal("hide"); |             packageAddModal.modal("hide"); | ||||||
|             const onSuccess = update => `Packages ${update} have been added`; |             const onSuccess = update => { return `Packages ${update} have been added`; }; | ||||||
|             const onFailure = error => `Package addition failed: ${error}`; |             const onFailure = error => { return `Package addition failed: ${error}`; }; | ||||||
|             doPackageAction("/api/v1/service/add", [packages], onSuccess, onFailure); |             doPackageAction("/api/v1/service/add", [packages], onSuccess, onFailure); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @ -70,8 +70,8 @@ | |||||||
|         const packages = packageInput.val(); |         const packages = packageInput.val(); | ||||||
|         if (packages) { |         if (packages) { | ||||||
|             packageAddModal.modal("hide"); |             packageAddModal.modal("hide"); | ||||||
|             const onSuccess = update => `Packages ${update} have been requested`; |             const onSuccess = update => { return `Packages ${update} have been requested`; }; | ||||||
|             const onFailure = error => `Package request failed: ${error}`; |             const onFailure = error => { return `Package request failed: ${error}`; }; | ||||||
|             doPackageAction("/api/v1/service/request", [packages], onSuccess, onFailure); |             doPackageAction("/api/v1/service/request", [packages], onSuccess, onFailure); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -61,7 +61,7 @@ | |||||||
|             error: (jqXHR, _, errorThrown) => { |             error: (jqXHR, _, errorThrown) => { | ||||||
|                 // show failed modal in case if first time loading |                 // show failed modal in case if first time loading | ||||||
|                 if (isPackageBaseSet) { |                 if (isPackageBaseSet) { | ||||||
|                     const message = error => `Could not load package ${packageBase} logs: ${error}`; |                     const message = error => { return `Could not load package ${packageBase} logs: ${error}`; }; | ||||||
|                     showFailure("Load failure", message, jqXHR, errorThrown); |                     showFailure("Load failure", message, jqXHR, errorThrown); | ||||||
|                 } |                 } | ||||||
|             }, |             }, | ||||||
|  | |||||||
| @ -33,8 +33,8 @@ | |||||||
|         const packages = dependencyInput.val(); |         const packages = dependencyInput.val(); | ||||||
|         if (packages) { |         if (packages) { | ||||||
|             packageRebuildModal.modal("hide"); |             packageRebuildModal.modal("hide"); | ||||||
|             const onSuccess = update => `Repository rebuild has been run for packages which depend on ${update}`; |             const onSuccess = update => { return `Repository rebuild has been run for packages which depend on ${update}`; }; | ||||||
|             const onFailure = error => `Repository rebuild failed: ${error}`; |             const onFailure = error => { return `Repository rebuild failed: ${error}`; }; | ||||||
|             doPackageAction("/api/v1/service/rebuild", [packages], onSuccess, onFailure); |             doPackageAction("/api/v1/service/rebuild", [packages], onSuccess, onFailure); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -15,25 +15,6 @@ | |||||||
|             table.bootstrapTable(method, {field: "id", values: [data.id]}); |             table.bootstrapTable(method, {field: "id", values: [data.id]}); | ||||||
|         } else showLogs(data.id); |         } else showLogs(data.id); | ||||||
|     }); |     }); | ||||||
|     table.on("created-controls.bs.table", () => { |  | ||||||
|         const pickerInput = $(".bootstrap-table-filter-control-timestamp"); |  | ||||||
|         pickerInput.daterangepicker({ |  | ||||||
|             autoUpdateInput: false, |  | ||||||
|             locale: { |  | ||||||
|                 cancelLabel: "Clear", |  | ||||||
|             }, |  | ||||||
|         }); |  | ||||||
|  |  | ||||||
|         pickerInput.on("apply.daterangepicker", (event, picker) => { |  | ||||||
|             pickerInput.val(`${picker.startDate.format("YYYY-MM-DD")} - ${picker.endDate.format("YYYY-MM-DD")}`); |  | ||||||
|             table.bootstrapTable("triggerSearch"); |  | ||||||
|         }); |  | ||||||
|  |  | ||||||
|         pickerInput.on("cancel.daterangepicker", () => { |  | ||||||
|             pickerInput.val(""); |  | ||||||
|             table.bootstrapTable("triggerSearch"); |  | ||||||
|         }); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const repositoryBadge = $("#badge-repository"); |     const repositoryBadge = $("#badge-repository"); | ||||||
|     const statusBadge = $("#badge-status"); |     const statusBadge = $("#badge-status"); | ||||||
| @ -56,21 +37,21 @@ | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     function getSelection() { |     function getSelection() { | ||||||
|         return table.bootstrapTable("getSelections").map(row => row.id); |         return table.bootstrapTable("getSelections").map(row => { return row.id; }); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     function removePackages() { |     function removePackages() { | ||||||
|         const onSuccess = update => `Packages ${update} have been removed`; |         const onSuccess = update => { return `Packages ${update} have been removed`; }; | ||||||
|         const onFailure = error => `Could not remove packages: ${error}`; |         const onFailure = error => { return `Could not remove packages: ${error}`; }; | ||||||
|         doPackageAction("/api/v1/service/remove", getSelection(), onSuccess, onFailure); |         doPackageAction("/api/v1/service/remove", getSelection(), onSuccess, onFailure); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     function updatePackages() { |     function updatePackages() { | ||||||
|         const currentSelection = getSelection(); |         const currentSelection = getSelection(); | ||||||
|         const [url, onSuccess] = currentSelection.length === 0 |         const [url, onSuccess] = currentSelection.length === 0 | ||||||
|             ? ["/api/v1/service/update", _ => "Repository update has been run"] |             ? ["/api/v1/service/update", _ => { return "Repository update has been run"; }] | ||||||
|             : ["/api/v1/service/add", update => `Run update for packages ${update}`]; |             : ["/api/v1/service/add", update => { return `Run update for packages ${update}`; }]; | ||||||
|         const onFailure = error => `Packages update failed: ${error}`; |         const onFailure = error => { return `Packages update failed: ${error}`; }; | ||||||
|         doPackageAction(url, currentSelection, onSuccess, onFailure); |         doPackageAction(url, currentSelection, onSuccess, onFailure); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @ -100,24 +81,23 @@ | |||||||
|             success: response => { |             success: response => { | ||||||
|                 const extractListProperties = (description, property) => { |                 const extractListProperties = (description, property) => { | ||||||
|                     return Object.values(description.packages) |                     return Object.values(description.packages) | ||||||
|                         .map(pkg => pkg[property]) |                         .map(pkg => { return pkg[property]; }) | ||||||
|                         .reduce((left, right) => left.concat(right), []); |                         .reduce((left, right) => { return left.concat(right); }, []); | ||||||
|                 }; |                 }; | ||||||
|                 const listToTable = data => { |                 const listToTable = data => { | ||||||
|                     return Array.from(new Set(data)) |                     return Array.from(new Set(data)) | ||||||
|                         .sort() |                         .sort() | ||||||
|                         .map(entry => safe(entry)) |                         .map(entry => { return safe(entry); }) | ||||||
|                         .join("<br>"); |                         .join("<br>"); | ||||||
|                 }; |                 }; | ||||||
|  |  | ||||||
|                 const payload = response.map(description => { |                 const payload = response.map(description => { | ||||||
|                     const package_base = description.package.base; |                     const package_base = description.package.base; | ||||||
|                     const web_url = description.package.remote.web_url; |                     const web_url = description.package.remote?.web_url; | ||||||
|                     return { |                     return { | ||||||
|                         id: package_base, |                         id: package_base, | ||||||
|                         base: web_url ? `<a href="${safe(web_url)}" title="${safe(package_base)}">${safe(package_base)}</a>` : safe(package_base), |                         base: web_url ? `<a href="${safe(web_url)}" title="${safe(package_base)}">${safe(package_base)}</a>` : safe(package_base), | ||||||
|                         version: safe(description.package.version), |                         version: safe(description.package.version), | ||||||
|                         packager: description.package.packager ? safe(description.package.packager) : "", |  | ||||||
|                         packages: listToTable(Object.keys(description.package.packages)), |                         packages: listToTable(Object.keys(description.package.packages)), | ||||||
|                         groups: listToTable(extractListProperties(description.package, "groups")), |                         groups: listToTable(extractListProperties(description.package, "groups")), | ||||||
|                         licenses: listToTable(extractListProperties(description.package, "licenses")), |                         licenses: listToTable(extractListProperties(description.package, "licenses")), | ||||||
| @ -140,8 +120,8 @@ | |||||||
|                     table.bootstrapTable("hideLoading"); |                     table.bootstrapTable("hideLoading"); | ||||||
|                 } else { |                 } else { | ||||||
|                     // other errors |                     // other errors | ||||||
|                     const message = error => `Could not load list of packages: ${error}`; |                     const messaga = error => { return `Could not load list of packages: ${error}`; }; | ||||||
|                     showFailure("Load failure", message, jqXHR, errorThrown); |                     showFailure("Load failure", messaga, jqXHR, errorThrown); | ||||||
|                 } |                 } | ||||||
|                 hideControls(true); |                 hideControls(true); | ||||||
|             }, |             }, | ||||||
| @ -177,18 +157,6 @@ | |||||||
|         return {classes: cellClass(value)}; |         return {classes: cellClass(value)}; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     function filterListGroups() { |  | ||||||
|         return extractDataList(table.bootstrapTable("getData"), "groups"); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     function filterListLicenses() { |  | ||||||
|         return extractDataList(table.bootstrapTable("getData"), "licenses"); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     function filterListPackagers() { |  | ||||||
|         return extractDataList(table.bootstrapTable("getData"), "packager"); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     $(() => { |     $(() => { | ||||||
|         table.bootstrapTable({}); |         table.bootstrapTable({}); | ||||||
|         statusBadge.popover(); |         statusBadge.popover(); | ||||||
|  | |||||||
| @ -31,8 +31,6 @@ SigLevel = Database{% if has_repo_signed %}Required{% else %}Never{% endif %} Pa | |||||||
|         <div class="container"> |         <div class="container"> | ||||||
|             <table id="packages" class="table table-striped table-hover" |             <table id="packages" class="table table-striped table-hover" | ||||||
|                    data-export-options='{"fileName": "packages"}' |                    data-export-options='{"fileName": "packages"}' | ||||||
|                    data-filter-control="true" |  | ||||||
|                    data-filter-control-visible="false" |  | ||||||
|                    data-page-list="[10, 25, 50, 100, all]" |                    data-page-list="[10, 25, 50, 100, all]" | ||||||
|                    data-page-size="10" |                    data-page-size="10" | ||||||
|                    data-pagination="true" |                    data-pagination="true" | ||||||
| @ -42,7 +40,6 @@ SigLevel = Database{% if has_repo_signed %}Required{% else %}Never{% endif %} Pa | |||||||
|                    data-show-columns-search="true" |                    data-show-columns-search="true" | ||||||
|                    data-show-columns-toggle-all="true" |                    data-show-columns-toggle-all="true" | ||||||
|                    data-show-export="true" |                    data-show-export="true" | ||||||
|                    data-show-filter-control-switch="true" |  | ||||||
|                    data-show-fullscreen="true" |                    data-show-fullscreen="true" | ||||||
|                    data-show-search-clear-button="true" |                    data-show-search-clear-button="true" | ||||||
|                    data-sortable="true" |                    data-sortable="true" | ||||||
| @ -51,17 +48,17 @@ SigLevel = Database{% if has_repo_signed %}Required{% else %}Never{% endif %} Pa | |||||||
|                    data-toggle="table"> |                    data-toggle="table"> | ||||||
|                 <thead class="table-primary"> |                 <thead class="table-primary"> | ||||||
|                     <tr> |                     <tr> | ||||||
|                         <th data-sortable="true" data-switchable="false" data-field="name" data-filter-control="input" data-filter-control-placeholder="(any package)">package</th> |                         <th data-sortable="true" data-switchable="false">package</th> | ||||||
|                         <th data-sortable="true" data-field="version" data-filter-control="input" data-filter-control-placeholder="(any version)">version</th> |                         <th data-sortable="true">version</th> | ||||||
|                         <th data-sortable="true" data-visible="false" data-field="architecture" data-filter-control="select" data-filter-control-placeholder="(any arch)">architecture</th> |                         <th data-sortable="true" data-visible="false">architecture</th> | ||||||
|                         <th data-sortable="true" data-visible="false" data-field="description" data-filter-control="input" data-filter-control-placeholder="(any description)">description</th> |                         <th data-sortable="true" data-visible="false">description</th> | ||||||
|                         <th data-sortable="true" data-visible="false" data-field="url">upstream url</th> |                         <th data-sortable="true" data-visible="false">upstream url</th> | ||||||
|                         <th data-sortable="true" data-visible="false" data-field="licenses" data-filter-control="select" data-filter-data="func:filterListLicenses" data-filter-custom-search="filterList" data-filter-control-placeholder="(any license)">licenses</th> |                         <th data-sortable="true" data-visible="false">licenses</th> | ||||||
|                         <th data-sortable="true" data-visible="false" data-field="groups" data-filter-control="select" data-filter-data="func:filterListGroups" data-filter-custom-search="filterList" data-filter-control-placeholder="(any group)">groups</th> |                         <th data-sortable="true" data-visible="false">groups</th> | ||||||
|                         <th data-sortable="true" data-visible="false" data-field="depends" data-filter-control="select" data-filter-data="func:filterListDepends" data-filter-custom-search="filterList" data-filter-control-placeholder="(any depends)">depends</th> |                         <th data-sortable="true" data-visible="false">depends</th> | ||||||
|                         <th data-sortable="true" data-field="archive_size">archive size</th> |                         <th data-sortable="true">archive size</th> | ||||||
|                         <th data-sortable="true" data-field="installed_size">installed size</th> |                         <th data-sortable="true">installed size</th> | ||||||
|                         <th data-sortable="true" data-field="timestamp" data-filter-control="input" data-filter-custom-search="filterDateRange" data-filter-control-placeholder="(any date)">build date</th> |                         <th data-sortable="true">build date</th> | ||||||
|                     </tr> |                     </tr> | ||||||
|                 </thead> |                 </thead> | ||||||
|  |  | ||||||
| @ -99,27 +96,6 @@ SigLevel = Database{% if has_repo_signed %}Required{% else %}Never{% endif %} Pa | |||||||
|         </div> |         </div> | ||||||
|  |  | ||||||
|         <script> |         <script> | ||||||
|             const table = $("#packages"); |  | ||||||
|             table.on("created-controls.bs.table", () => { |  | ||||||
|                 const pickerInput = $(".bootstrap-table-filter-control-timestamp"); |  | ||||||
|                 pickerInput.daterangepicker({ |  | ||||||
|                     autoUpdateInput: false, |  | ||||||
|                     locale: { |  | ||||||
|                         cancelLabel: "Clear", |  | ||||||
|                     }, |  | ||||||
|                 }); |  | ||||||
|  |  | ||||||
|                 pickerInput.on("apply.daterangepicker", (event, picker) => { |  | ||||||
|                     pickerInput.val(`${picker.startDate.format("YYYY-MM-DD")} - ${picker.endDate.format("YYYY-MM-DD")}`); |  | ||||||
|                     table.bootstrapTable("triggerSearch"); |  | ||||||
|                 }); |  | ||||||
|  |  | ||||||
|                 pickerInput.on("cancel.daterangepicker", () => { |  | ||||||
|                     pickerInput.val(""); |  | ||||||
|                     table.bootstrapTable("triggerSearch"); |  | ||||||
|                 }); |  | ||||||
|             }); |  | ||||||
|  |  | ||||||
|             const pacmanConf = $("#pacman-conf"); |             const pacmanConf = $("#pacman-conf"); | ||||||
|             const pacmanConfCopyButton = $("#copy-btn"); |             const pacmanConfCopyButton = $("#copy-btn"); | ||||||
|  |  | ||||||
| @ -127,18 +103,6 @@ SigLevel = Database{% if has_repo_signed %}Required{% else %}Never{% endif %} Pa | |||||||
|                 const conf = pacmanConf.text(); |                 const conf = pacmanConf.text(); | ||||||
|                 await copyToClipboard(conf, pacmanConfCopyButton); |                 await copyToClipboard(conf, pacmanConfCopyButton); | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             function filterListDepends() { |  | ||||||
|                 return extractDataList(table.bootstrapTable("getData"), "depends"); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             function filterListGroups() { |  | ||||||
|                 return extractDataList(table.bootstrapTable("getData"), "groups"); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             function filterListLicenses() { |  | ||||||
|                 return extractDataList(table.bootstrapTable("getData"), "licenses"); |  | ||||||
|             } |  | ||||||
|         </script> |         </script> | ||||||
|  |  | ||||||
|     </body> |     </body> | ||||||
|  | |||||||
| @ -1,21 +1,16 @@ | |||||||
| <script src="https://cdn.jsdelivr.net/npm/jquery@3.7.0/dist/jquery.min.js" integrity="sha384-NXgwF8Kv9SSAr+jemKKcbvQsz+teULH/a5UNJvZc6kP47hZgl62M1vGnw6gHQhb1" crossorigin="anonymous" type="application/javascript"></script> | <script src="https://cdn.jsdelivr.net/npm/jquery/dist/jquery.min.js"></script> | ||||||
|  |  | ||||||
| <script src="https://cdn.jsdelivr.net/npm/moment@2.29.4/moment.min.js" integrity="sha384-8hHkOkbWN1TLWwet/jpbJ0zbx3FJDeYJgQ8dX1mRrv/vfCfHCqFSFZYCgaMML3z9" crossorigin="anonymous" type="application/javascript"></script> | <script src="https://unpkg.com/tableexport.jquery.plugin/tableExport.min.js"></script> | ||||||
| <script src="https://cdn.jsdelivr.net/npm/daterangepicker@3.1.0/daterangepicker.min.js" integrity="sha384-u4eJN1VWrTf/FnYYQJo2kqJyVxEQf5UmWY4iUcNAoLenOEtEuCkfwc5bKvZOWBi5" crossorigin="anonymous" type="application/javascript"></script> |  | ||||||
|  |  | ||||||
| <script src="https://cdn.jsdelivr.net/npm/tableexport.jquery.plugin@1.28.0/tableExport.min.js" integrity="sha384-1Rz4Kz/y1rSWw+ZsjTcxB684XgofbO8iizY+UFIzCwFeQ+QUyhBNWBMh/STOyomI" crossorigin="anonymous" type="application/javascript"></script> | <script src="https://unpkg.com/jquery-resizable-columns@0.2.3/dist/jquery.resizableColumns.min.js"></script> | ||||||
|  |  | ||||||
| <script src="https://cdn.jsdelivr.net/npm/jquery-resizable-columns@0.2.3/dist/jquery.resizableColumns.min.js" integrity="sha384-IazMVNyYoUNx6357fWJoqtHYUWWCNHIXxFVtbpVgvImQNWuRP2WbHPaIb3QF8j97" crossorigin="anonymous" type="application/javascript"></script> | <script src="https://cdn.jsdelivr.net/npm/@popperjs/core@2.11.6/dist/umd/popper.min.js" integrity="sha384-oBqDVmMz9ATKxIep9tiCxS/Z9fNfEXiDAYTujMAeBAsjFuCZSmKbSSUnQlmh/jp3" crossorigin="anonymous"></script> | ||||||
|  | <script src="https://cdn.jsdelivr.net/npm/bootstrap@5.2.2/dist/js/bootstrap.min.js" integrity="sha384-IDwe1+LCz02ROU9k972gdyvl+AESN10+x7tBKgc9I5HFtuNz0wWnPclzo6p9vxnk" crossorigin="anonymous"></script> | ||||||
|  | <script src="https://unpkg.com/bootstrap-table@1.21.1/dist/bootstrap-table.min.js"></script> | ||||||
|  |  | ||||||
| <script src="https://cdn.jsdelivr.net/npm/@popperjs/core@2.11.8/dist/umd/popper.min.js" integrity="sha384-I7E8VVD/ismYTF4hNIPjVp/Zjvgyol6VFvRkX/vR+Vc4jQkC+hVqc2pM8ODewa9r" crossorigin="anonymous" type="application/javascript"></script> | <script src="https://unpkg.com/bootstrap-table@1.21.1/dist/extensions/export/bootstrap-table-export.min.js"></script> | ||||||
| <script src="https://cdn.jsdelivr.net/npm/bootstrap@5.2.3/dist/js/bootstrap.min.js" integrity="sha384-cuYeSxntonz0PPNlHhBs68uyIAVpIIOZZ5JqeqvYYIcEL727kskC66kF92t6Xl2V" crossorigin="anonymous" type="application/javascript"></script> |  | ||||||
| <script src="https://cdn.jsdelivr.net/npm/bootstrap-table@1.22.1/dist/bootstrap-table.min.js" integrity="sha384-GVLHfbEvuGA/RFiQ3MK2ClEJkWYJXABg55t9LpoDPZFGIsSq8xhFlQydm5poV2jW" crossorigin="anonymous" type="application/javascript"></script> |  | ||||||
|  |  | ||||||
| <script src="https://cdn.jsdelivr.net/npm/bootstrap-table@1.21.4/dist/extensions/export/bootstrap-table-export.min.js" integrity="sha384-jeldDadm+qM2RwGER3qVqxFgWVpAEJ7Jie+0rlYj8ni3KkQA654T8TSXDtol022X" crossorigin="anonymous" type="application/javascript"></script> | <script src="https://unpkg.com/bootstrap-table@1.21.1/dist/extensions/resizable/bootstrap-table-resizable.js"></script> | ||||||
|  |  | ||||||
| <script src="https://cdn.jsdelivr.net/npm/bootstrap-table@1.21.4/dist/extensions/resizable/bootstrap-table-resizable.js" integrity="sha384-wd8Vc6Febikdnsnk9vthRWRvMwffw246vhqiqNO3aSNe1maTEA07Vh3zAQiSyDji" crossorigin="anonymous" type="application/javascript"></script> |  | ||||||
|  |  | ||||||
| <script src="https://cdn.jsdelivr.net/npm/bootstrap-table@1.21.4/dist/extensions/filter-control/bootstrap-table-filter-control.js" integrity="sha384-B6xNXlSOaOFxjlKo9OW3htbox+9/DcaEcjPPEi1+pTMwH5Tzc/s2wNTYriHz7Tb8" crossorigin="anonymous" type="application/javascript"></script> |  | ||||||
|  |  | ||||||
| <script> | <script> | ||||||
|     async function copyToClipboard(text, button) { |     async function copyToClipboard(text, button) { | ||||||
| @ -43,28 +38,4 @@ | |||||||
|             .replace(/>/g, ">") |             .replace(/>/g, ">") | ||||||
|             .replace(/"/g, """); |             .replace(/"/g, """); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     function extractDataList(data, column) { |  | ||||||
|         const elements = data.flatMap(row => row[column].split("<br>")).filter(v => v); // remove empty elements from array |  | ||||||
|         return Array.from(new Set(elements)).sort(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     function filterContains(text, value) { |  | ||||||
|         return value.includes(text.toLowerCase().trim()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     function filterDateRange(text, value) { |  | ||||||
|         const asOfStartOfDay = date => date.setUTCHours(0, 0, 0, 0); |  | ||||||
|  |  | ||||||
|         const [minDate, maxDate] = text.split(" - "); |  | ||||||
|         const buildDate = asOfStartOfDay(new Date(value)); |  | ||||||
|  |  | ||||||
|         return (buildDate >= asOfStartOfDay(new Date(minDate))) && (buildDate <= asOfStartOfDay(new Date(maxDate))); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     function filterList(index, value, field, data) { |  | ||||||
|         const dataList = extractDataList(data, field); |  | ||||||
|         // the library removes all symbols from string, so it is just string |  | ||||||
|         return value.includes(dataList[index].toLowerCase()); |  | ||||||
|     } |  | ||||||
| </script> | </script> | ||||||
|  | |||||||
| @ -1,15 +1,11 @@ | |||||||
| <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@5.2.3/dist/css/bootstrap.min.css" integrity="sha384-rbsA2VBKQhggwzxH7pPCaAqO46MgnOM80zW1RWuH61DGLwZJEdK2Kadq2F9CUG65" crossorigin="anonymous" type="text/css"> | <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@5.2.2/dist/css/bootstrap.min.css" integrity="sha384-Zenh87qX5JnK2Jl0vWa8Ck2rdkQ2Bzep5IDxbcnCeuOxjzrPF/et3URy9Bv1WTRi" crossorigin="anonymous" type="text/css"> | ||||||
| <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap-icons@1.10.4/font/bootstrap-icons.css" integrity="sha384-LrVLJJYk9OiJmjNDakUBU7kS9qCT8wk1j2OU7ncpsfB3QS37UPdkCuq3ZD1MugNY" crossorigin="anonymous" type="text/css"> | <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap-icons@1.10.2/font/bootstrap-icons.css" type="text/css"> | ||||||
|  |  | ||||||
| <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap-table@1.21.4/dist/bootstrap-table.min.css" integrity="sha384-pTEAhytv7JmEG2D7qiW5gY0lI5EKZ9n3CNmj6Qp+U3qhnmH2qnnN9KJbVwbtMHN0" crossorigin="anonymous" type="text/css"> | <link rel="stylesheet" href="https://unpkg.com/bootstrap-table@1.21.1/dist/bootstrap-table.min.css" type="text/css"> | ||||||
|  |  | ||||||
| <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/jquery-resizable-columns@0.2.3/dist/jquery.resizableColumns.css" integrity="sha384-1sLxvR8mXzjhvFY9f8mzXl97DNLepeZ0PnRiMMdm/rQsKjsrPZPJxYle2wwT2PMg" crossorigin="anonymous" type="text/css"> | <link rel="stylesheet" href="https://unpkg.com/jquery-resizable-columns@0.2.3/dist/jquery.resizableColumns.css" type="text/css"> | ||||||
|  |  | ||||||
| <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap-table@1.21.4/dist/extensions/filter-control/bootstrap-table-filter-control.css" integrity="sha384-4Glx18jZ0Un+yDG6KUpYJ/af8hkssJ02jRASuFv23gfCl0mTXaVMPI9cB4cn3GvE" crossorigin="anonymous" type="text/css"> | <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootswatch@5.2.2/dist/cosmo/bootstrap.min.css" integrity="sha256-5t++JZpgVLzo9vF7snO5Qw0y3fA5/NkoJENWB7kpg0E=" crossorigin="anonymous" type="text/css"> | ||||||
|  |  | ||||||
| <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootswatch@5.2.3/dist/cosmo/bootstrap.min.css" integrity="sha384-P1PBFVifKf1Ww0gS5B8A0siIeDpcFd4uU7S68LA1XMdE0R+y1WN3DR4HcLc9csRC" crossorigin="anonymous" type="text/css"> |  | ||||||
|  |  | ||||||
| <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/daterangepicker@3.1.0/daterangepicker.css" integrity="sha384-zLkQsiLfAQqGeIJeKLC+rcCR1YoYaQFLCL7cLDUoKE1ajKJzySpjzWGfYS2vjSG+" crossorigin="anonymous" type="text/css"> |  | ||||||
|  |  | ||||||
| <style> | <style> | ||||||
|     .pre-scrollable { |     .pre-scrollable { | ||||||
|  | |||||||
| @ -1,84 +0,0 @@ | |||||||
| [build-system] |  | ||||||
| requires = ["flit_core"] |  | ||||||
| build-backend = "flit_core.buildapi" |  | ||||||
|  |  | ||||||
| [project] |  | ||||||
| name = "ahriman" |  | ||||||
|  |  | ||||||
| description = "ArcH linux ReposItory MANager" |  | ||||||
| readme = "README.md" |  | ||||||
|  |  | ||||||
| requires-python = ">=3.11" |  | ||||||
|  |  | ||||||
| license = {file = "COPYING"} |  | ||||||
| authors = [ |  | ||||||
|     {name = "ahriman team"}, |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| dependencies = [ |  | ||||||
|     "cerberus", |  | ||||||
|     "inflection", |  | ||||||
|     "passlib", |  | ||||||
|     "requests", |  | ||||||
|     "srcinfo", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| dynamic = ["version"] |  | ||||||
|  |  | ||||||
| [project.urls] |  | ||||||
| Documentation = "https://ahriman.readthedocs.io/" |  | ||||||
| Repository = "https://github.com/arcan1s/ahriman" |  | ||||||
| Changelog = "https://github.com/arcan1s/ahriman/releases" |  | ||||||
|  |  | ||||||
| [project.scripts] |  | ||||||
| ahriman = "ahriman.application.ahriman:run" |  | ||||||
|  |  | ||||||
| [project.optional-dependencies] |  | ||||||
| check = [ |  | ||||||
|     "autopep8", |  | ||||||
|     "bandit", |  | ||||||
|     "mypy", |  | ||||||
|     "pylint", |  | ||||||
| ] |  | ||||||
|  docs = [ |  | ||||||
|      "Sphinx", |  | ||||||
|      "argparse-manpage", |  | ||||||
|      "pydeps", |  | ||||||
|      "shtab", |  | ||||||
|      "sphinx-argparse", |  | ||||||
|      "sphinx-rtd-theme>=1.1.1",  # https://stackoverflow.com/a/74355734 |  | ||||||
|  ] |  | ||||||
| journald = [ |  | ||||||
|     "systemd-python", |  | ||||||
| ] |  | ||||||
| # FIXME technically this dependency is required, but in some cases we do not have access to |  | ||||||
| # the libalpm which is required in order to install the package. Thus in case if we do not |  | ||||||
| # really need to run the application we can move it to "optional" dependencies |  | ||||||
| pacman = [ |  | ||||||
|     "pyalpm", |  | ||||||
| ] |  | ||||||
| s3 = [ |  | ||||||
|     "boto3", |  | ||||||
| ] |  | ||||||
| tests = [ |  | ||||||
|     "pytest", |  | ||||||
|     "pytest-aiohttp", |  | ||||||
|     "pytest-cov", |  | ||||||
|     "pytest-helpers-namespace", |  | ||||||
|     "pytest-mock", |  | ||||||
|     "pytest-resource-path", |  | ||||||
|     "pytest-spec", |  | ||||||
| ] |  | ||||||
| web = [ |  | ||||||
|     "Jinja2", |  | ||||||
|     "aioauth-client", |  | ||||||
|     "aiohttp", |  | ||||||
|     "aiohttp-apispec", |  | ||||||
|     "aiohttp_cors", |  | ||||||
|     "aiohttp_jinja2", |  | ||||||
|     "aiohttp_debugtoolbar", |  | ||||||
|     "aiohttp_session", |  | ||||||
|     "aiohttp_security", |  | ||||||
|     "cryptography", |  | ||||||
|     "requests-unixsocket",  # required by unix socket support |  | ||||||
| ] |  | ||||||
							
								
								
									
										156
									
								
								setup.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										156
									
								
								setup.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,156 @@ | |||||||
|  | from pathlib import Path | ||||||
|  | from setuptools import find_packages, setup | ||||||
|  | from typing import Any | ||||||
|  |  | ||||||
|  |  | ||||||
|  | metadata_path = Path(__file__).resolve().parent / "src/ahriman/version.py" | ||||||
|  | metadata: dict[str, Any] = {} | ||||||
|  | with metadata_path.open() as metadata_file: | ||||||
|  |     exec(metadata_file.read(), metadata)  # pylint: disable=exec-used | ||||||
|  |  | ||||||
|  |  | ||||||
|  | setup( | ||||||
|  |     name="ahriman", | ||||||
|  |  | ||||||
|  |     version=metadata["__version__"], | ||||||
|  |     zip_safe=False, | ||||||
|  |  | ||||||
|  |     description="ArcH linux ReposItory MANager", | ||||||
|  |  | ||||||
|  |     author="ahriman team", | ||||||
|  |     author_email="", | ||||||
|  |     url="https://github.com/arcan1s/ahriman", | ||||||
|  |  | ||||||
|  |     license="GPL3", | ||||||
|  |  | ||||||
|  |     packages=find_packages("src"), | ||||||
|  |     package_dir={"": "src"}, | ||||||
|  |     package_data={"": ["py.typed"]}, | ||||||
|  |  | ||||||
|  |     dependency_links=[ | ||||||
|  |     ], | ||||||
|  |     install_requires=[ | ||||||
|  |         "cerberus", | ||||||
|  |         "inflection", | ||||||
|  |         "passlib", | ||||||
|  |         "requests", | ||||||
|  |         "srcinfo", | ||||||
|  |     ], | ||||||
|  |     setup_requires=[ | ||||||
|  |     ], | ||||||
|  |     tests_require=[ | ||||||
|  |         "pytest", | ||||||
|  |         "pytest-aiohttp", | ||||||
|  |         "pytest-cov", | ||||||
|  |         "pytest-helpers-namespace", | ||||||
|  |         "pytest-mock", | ||||||
|  |         "pytest-spec", | ||||||
|  |         "pytest-resource-path", | ||||||
|  |     ], | ||||||
|  |  | ||||||
|  |     include_package_data=True, | ||||||
|  |     scripts=[ | ||||||
|  |         "package/bin/ahriman", | ||||||
|  |     ], | ||||||
|  |     data_files=[ | ||||||
|  |         # configuration | ||||||
|  |         ("share/ahriman/settings", [ | ||||||
|  |             "package/share/ahriman/settings/ahriman.ini", | ||||||
|  |         ]), | ||||||
|  |         ("share/ahriman/settings/ahriman.ini.d", [ | ||||||
|  |             "package/share/ahriman/settings/ahriman.ini.d/logging.ini", | ||||||
|  |         ]), | ||||||
|  |         # systemd files | ||||||
|  |         ("lib/systemd/system", [ | ||||||
|  |             "package/lib/systemd/system/ahriman@.service", | ||||||
|  |             "package/lib/systemd/system/ahriman@.timer", | ||||||
|  |             "package/lib/systemd/system/ahriman-web@.service", | ||||||
|  |         ]), | ||||||
|  |         # templates | ||||||
|  |         ("share/ahriman/templates", [ | ||||||
|  |             "package/share/ahriman/templates/api.jinja2", | ||||||
|  |             "package/share/ahriman/templates/build-status.jinja2", | ||||||
|  |             "package/share/ahriman/templates/email-index.jinja2", | ||||||
|  |             "package/share/ahriman/templates/error.jinja2", | ||||||
|  |             "package/share/ahriman/templates/repo-index.jinja2", | ||||||
|  |             "package/share/ahriman/templates/shell", | ||||||
|  |             "package/share/ahriman/templates/telegram-index.jinja2", | ||||||
|  |         ]), | ||||||
|  |         ("share/ahriman/templates/build-status", [ | ||||||
|  |             "package/share/ahriman/templates/build-status/alerts.jinja2", | ||||||
|  |             "package/share/ahriman/templates/build-status/key-import-modal.jinja2", | ||||||
|  |             "package/share/ahriman/templates/build-status/login-modal.jinja2", | ||||||
|  |             "package/share/ahriman/templates/build-status/package-add-modal.jinja2", | ||||||
|  |             "package/share/ahriman/templates/build-status/package-info-modal.jinja2", | ||||||
|  |             "package/share/ahriman/templates/build-status/package-rebuild-modal.jinja2", | ||||||
|  |             "package/share/ahriman/templates/build-status/table.jinja2", | ||||||
|  |         ]), | ||||||
|  |         ("share/ahriman/templates/static", [ | ||||||
|  |             "package/share/ahriman/templates/static/favicon.ico", | ||||||
|  |         ]), | ||||||
|  |         ("share/ahriman/templates/utils", [ | ||||||
|  |             "package/share/ahriman/templates/utils/bootstrap-scripts.jinja2", | ||||||
|  |             "package/share/ahriman/templates/utils/style.jinja2", | ||||||
|  |         ]), | ||||||
|  |         # man pages | ||||||
|  |         ("share/man/man1", [ | ||||||
|  |             "docs/ahriman.1", | ||||||
|  |         ]), | ||||||
|  |         # shell completions | ||||||
|  |         ("share/bash-completion/completions", [ | ||||||
|  |             "docs/completions/bash/_ahriman", | ||||||
|  |         ]), | ||||||
|  |         ("share/zsh/site-functions", [ | ||||||
|  |             "docs/completions/zsh/_ahriman", | ||||||
|  |         ]), | ||||||
|  |     ], | ||||||
|  |  | ||||||
|  |     extras_require={ | ||||||
|  |         "check": [ | ||||||
|  |             "autopep8", | ||||||
|  |             "bandit", | ||||||
|  |             "mypy", | ||||||
|  |             "pylint", | ||||||
|  |         ], | ||||||
|  |         "docs": [ | ||||||
|  |             "Sphinx", | ||||||
|  |             "argparse-manpage", | ||||||
|  |             "pydeps", | ||||||
|  |             "shtab", | ||||||
|  |             "sphinx-argparse", | ||||||
|  |             "sphinx-rtd-theme>=1.1.1",  # https://stackoverflow.com/a/74355734 | ||||||
|  |             "sphinxcontrib-napoleon", | ||||||
|  |         ], | ||||||
|  |         # FIXME technically this dependency is required, but in some cases we do not have access to | ||||||
|  |         # the libalpm which is required in order to install the package. Thus in case if we do not | ||||||
|  |         # really need to run the application we can move it to "optional" dependencies | ||||||
|  |         "pacman": [ | ||||||
|  |             "pyalpm", | ||||||
|  |         ], | ||||||
|  |         "s3": [ | ||||||
|  |             "boto3", | ||||||
|  |         ], | ||||||
|  |         "tests": [ | ||||||
|  |             "pytest", | ||||||
|  |             "pytest-aiohttp", | ||||||
|  |             "pytest-cov", | ||||||
|  |             "pytest-helpers-namespace", | ||||||
|  |             "pytest-mock", | ||||||
|  |             "pytest-resource-path", | ||||||
|  |             "pytest-spec", | ||||||
|  |         ], | ||||||
|  |         "web": [ | ||||||
|  |             "Jinja2", | ||||||
|  |             "aioauth-client", | ||||||
|  |             "aiohttp", | ||||||
|  |             "aiohttp-apispec", | ||||||
|  |             "aiohttp_cors", | ||||||
|  |             "aiohttp_jinja2", | ||||||
|  |             "aiohttp_debugtoolbar", | ||||||
|  |             "aiohttp_session", | ||||||
|  |             "aiohttp_security", | ||||||
|  |             "cryptography", | ||||||
|  |             "requests-unixsocket",  # required by unix socket support | ||||||
|  |         ], | ||||||
|  |     }, | ||||||
|  | ) | ||||||
| @ -17,4 +17,3 @@ | |||||||
| # You should have received a copy of the GNU General Public License | # You should have received a copy of the GNU General Public License | ||||||
| # along with this program. If not, see <http://www.gnu.org/licenses/>. | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
| # | # | ||||||
| __version__ = "2.11.0" |  | ||||||
|  | |||||||
| @ -19,17 +19,17 @@ | |||||||
| # | # | ||||||
| # pylint: disable=too-many-lines | # pylint: disable=too-many-lines | ||||||
| import argparse | import argparse | ||||||
|  | import sys | ||||||
| import tempfile | import tempfile | ||||||
|  |  | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
| from typing import TypeVar | from typing import TypeVar | ||||||
|  |  | ||||||
| from ahriman import __version__ | from ahriman import version | ||||||
| from ahriman.application import handlers | from ahriman.application import handlers | ||||||
| from ahriman.core.util import enum_values, extract_user | from ahriman.core.util import enum_values | ||||||
| from ahriman.models.action import Action | from ahriman.models.action import Action | ||||||
| from ahriman.models.build_status import BuildStatusEnum | from ahriman.models.build_status import BuildStatusEnum | ||||||
| from ahriman.models.log_handler import LogHandler |  | ||||||
| from ahriman.models.package_source import PackageSource | from ahriman.models.package_source import PackageSource | ||||||
| from ahriman.models.sign_settings import SignSettings | from ahriman.models.sign_settings import SignSettings | ||||||
| from ahriman.models.user_access import UserAccess | from ahriman.models.user_access import UserAccess | ||||||
| @ -58,7 +58,6 @@ def _formatter(prog: str) -> argparse.HelpFormatter: | |||||||
|     return argparse.ArgumentDefaultsHelpFormatter(prog, width=120) |     return argparse.ArgumentDefaultsHelpFormatter(prog, width=120) | ||||||
|  |  | ||||||
|  |  | ||||||
| # pylint: disable=too-many-statements |  | ||||||
| def _parser() -> argparse.ArgumentParser: | def _parser() -> argparse.ArgumentParser: | ||||||
|     """ |     """ | ||||||
|     command line parser generator |     command line parser generator | ||||||
| @ -76,21 +75,12 @@ def _parser() -> argparse.ArgumentParser: | |||||||
|     parser.add_argument("--force", help="force run, remove file lock", action="store_true") |     parser.add_argument("--force", help="force run, remove file lock", action="store_true") | ||||||
|     parser.add_argument("-l", "--lock", help="lock file", type=Path, |     parser.add_argument("-l", "--lock", help="lock file", type=Path, | ||||||
|                         default=Path(tempfile.gettempdir()) / "ahriman.lock") |                         default=Path(tempfile.gettempdir()) / "ahriman.lock") | ||||||
|     parser.add_argument("--log-handler", help="explicit log handler specification. If none set, the handler will be " |  | ||||||
|                                               "guessed from environment", |  | ||||||
|                         type=LogHandler, choices=enum_values(LogHandler)) |  | ||||||
|     parser.add_argument("--report", help="force enable or disable reporting to web service", |     parser.add_argument("--report", help="force enable or disable reporting to web service", | ||||||
|                         action=argparse.BooleanOptionalAction, default=True) |                         action=argparse.BooleanOptionalAction, default=True) | ||||||
|     parser.add_argument("-R", "--repository", help="target repository. For several subcommands it can be used " |  | ||||||
|                                                    "multiple times", action="append") |  | ||||||
|     parser.add_argument("-q", "--quiet", help="force disable any logging", action="store_true") |     parser.add_argument("-q", "--quiet", help="force disable any logging", action="store_true") | ||||||
|     parser.add_argument("--unsafe", help="allow to run ahriman as non-ahriman user. Some actions might be unavailable", |     parser.add_argument("--unsafe", help="allow to run ahriman as non-ahriman user. Some actions might be unavailable", | ||||||
|                         action="store_true") |                         action="store_true") | ||||||
|     parser.add_argument("--wait-timeout", help="wait for lock to be free. Negative value will lead to " |     parser.add_argument("-V", "--version", action="version", version=version.__version__) | ||||||
|                                                "immediate application run even if there is lock file. " |  | ||||||
|                                                "In case of zero value, the application will wait infinitely", |  | ||||||
|                         type=int, default=-1) |  | ||||||
|     parser.add_argument("-V", "--version", action="version", version=__version__) |  | ||||||
|  |  | ||||||
|     subparsers = parser.add_subparsers(title="command", help="command to run", dest="command", required=True) |     subparsers = parser.add_subparsers(title="command", help="command to run", dest="command", required=True) | ||||||
|  |  | ||||||
| @ -110,8 +100,6 @@ def _parser() -> argparse.ArgumentParser: | |||||||
|     _set_patch_set_add_parser(subparsers) |     _set_patch_set_add_parser(subparsers) | ||||||
|     _set_repo_backup_parser(subparsers) |     _set_repo_backup_parser(subparsers) | ||||||
|     _set_repo_check_parser(subparsers) |     _set_repo_check_parser(subparsers) | ||||||
|     _set_repo_create_keyring_parser(subparsers) |  | ||||||
|     _set_repo_create_mirrorlist_parser(subparsers) |  | ||||||
|     _set_repo_daemon_parser(subparsers) |     _set_repo_daemon_parser(subparsers) | ||||||
|     _set_repo_rebuild_parser(subparsers) |     _set_repo_rebuild_parser(subparsers) | ||||||
|     _set_repo_remove_unknown_parser(subparsers) |     _set_repo_remove_unknown_parser(subparsers) | ||||||
| @ -192,8 +180,8 @@ def _set_help_commands_unsafe_parser(root: SubParserAction) -> argparse.Argument | |||||||
|     """ |     """ | ||||||
|     parser = root.add_parser("help-commands-unsafe", help="list unsafe commands", |     parser = root.add_parser("help-commands-unsafe", help="list unsafe commands", | ||||||
|                              description="list unsafe commands as defined in default args", formatter_class=_formatter) |                              description="list unsafe commands as defined in default args", formatter_class=_formatter) | ||||||
|     parser.add_argument("command", help="instead of showing commands, just test command line for unsafe subcommand " |     parser.add_argument("--command", help="instead of showing commands, just test command line for unsafe subcommand " | ||||||
|                                         "and return 0 in case if command is safe and 1 otherwise", nargs="*") |                                           "and return 0 in case if command is safe and 1 otherwise") | ||||||
|     parser.set_defaults(handler=handlers.UnsafeCommands, architecture=[""], lock=None, report=False, quiet=True, |     parser.set_defaults(handler=handlers.UnsafeCommands, architecture=[""], lock=None, report=False, quiet=True, | ||||||
|                         unsafe=True, parser=_parser) |                         unsafe=True, parser=_parser) | ||||||
|     return parser |     return parser | ||||||
| @ -261,15 +249,12 @@ def _set_package_add_parser(root: SubParserAction) -> argparse.ArgumentParser: | |||||||
|     parser.add_argument("--dependencies", help="process missing package dependencies", |     parser.add_argument("--dependencies", help="process missing package dependencies", | ||||||
|                         action=argparse.BooleanOptionalAction, default=True) |                         action=argparse.BooleanOptionalAction, default=True) | ||||||
|     parser.add_argument("-e", "--exit-code", help="return non-zero exit status if result is empty", action="store_true") |     parser.add_argument("-e", "--exit-code", help="return non-zero exit status if result is empty", action="store_true") | ||||||
|     parser.add_argument("--increment", help="increment package release (pkgrel) version on duplicate", |  | ||||||
|                         action=argparse.BooleanOptionalAction, default=True) |  | ||||||
|     parser.add_argument("-n", "--now", help="run update function after", action="store_true") |     parser.add_argument("-n", "--now", help="run update function after", action="store_true") | ||||||
|     parser.add_argument("-y", "--refresh", help="download fresh package databases from the mirror before actions, " |     parser.add_argument("-y", "--refresh", help="download fresh package databases from the mirror before actions, " | ||||||
|                                                 "-yy to force refresh even if up to date", |                                                 "-yy to force refresh even if up to date", | ||||||
|                         action="count", default=False) |                         action="count", default=False) | ||||||
|     parser.add_argument("-s", "--source", help="explicitly specify the package source for this command", |     parser.add_argument("-s", "--source", help="explicitly specify the package source for this command", | ||||||
|                         type=PackageSource, choices=enum_values(PackageSource), default=PackageSource.Auto) |                         type=PackageSource, choices=enum_values(PackageSource), default=PackageSource.Auto) | ||||||
|     parser.add_argument("-u", "--username", help="build as user", default=extract_user()) |  | ||||||
|     parser.set_defaults(handler=handlers.Add) |     parser.set_defaults(handler=handlers.Add) | ||||||
|     return parser |     return parser | ||||||
|  |  | ||||||
| @ -489,46 +474,7 @@ def _set_repo_check_parser(root: SubParserAction) -> argparse.ArgumentParser: | |||||||
|     parser.add_argument("-y", "--refresh", help="download fresh package databases from the mirror before actions, " |     parser.add_argument("-y", "--refresh", help="download fresh package databases from the mirror before actions, " | ||||||
|                                                 "-yy to force refresh even if up to date", |                                                 "-yy to force refresh even if up to date", | ||||||
|                         action="count", default=False) |                         action="count", default=False) | ||||||
|     parser.set_defaults(handler=handlers.Update, dependencies=False, dry_run=True, aur=True, local=True, manual=False, |     parser.set_defaults(handler=handlers.Update, dependencies=False, dry_run=True, aur=True, local=True, manual=False) | ||||||
|                         username=None) |  | ||||||
|     return parser |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def _set_repo_create_keyring_parser(root: SubParserAction) -> argparse.ArgumentParser: |  | ||||||
|     """ |  | ||||||
|     add parser for create-keyring subcommand |  | ||||||
|  |  | ||||||
|     Args: |  | ||||||
|         root(SubParserAction): subparsers for the commands |  | ||||||
|  |  | ||||||
|     Returns: |  | ||||||
|         argparse.ArgumentParser: created argument parser |  | ||||||
|     """ |  | ||||||
|     parser = root.add_parser("repo-create-keyring", help="create keyring package", |  | ||||||
|                              description="create package which contains list of trusted keys as set by " |  | ||||||
|                                          "configuration. Note, that this action will only create package, the package " |  | ||||||
|                                          "itself has to be built manually", |  | ||||||
|                              formatter_class=_formatter) |  | ||||||
|     parser.set_defaults(handler=handlers.Triggers, trigger=["ahriman.core.support.KeyringTrigger"]) |  | ||||||
|     return parser |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def _set_repo_create_mirrorlist_parser(root: SubParserAction) -> argparse.ArgumentParser: |  | ||||||
|     """ |  | ||||||
|     add parser for create-mirrorlist subcommand |  | ||||||
|  |  | ||||||
|     Args: |  | ||||||
|         root(SubParserAction): subparsers for the commands |  | ||||||
|  |  | ||||||
|     Returns: |  | ||||||
|         argparse.ArgumentParser: created argument parser |  | ||||||
|     """ |  | ||||||
|     parser = root.add_parser("repo-create-mirrorlist", help="create mirrorlist package", |  | ||||||
|                              description="create package which contains list of available mirrors as set by " |  | ||||||
|                                          "configuration. Note, that this action will only create package, the package " |  | ||||||
|                                          "itself has to be built manually", |  | ||||||
|                              formatter_class=_formatter) |  | ||||||
|     parser.set_defaults(handler=handlers.Triggers, trigger=["ahriman.core.support.MirrorlistTrigger"]) |  | ||||||
|     return parser |     return parser | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -584,12 +530,7 @@ def _set_repo_rebuild_parser(root: SubParserAction) -> argparse.ArgumentParser: | |||||||
|                              "instance. Note, however, that in order to restore packages you need to have original " |                              "instance. Note, however, that in order to restore packages you need to have original " | ||||||
|                              "ahriman instance run with web service and have run repo-update at least once.", |                              "ahriman instance run with web service and have run repo-update at least once.", | ||||||
|                         action="store_true") |                         action="store_true") | ||||||
|     parser.add_argument("--increment", help="increment package release (pkgrel) on duplicate", |  | ||||||
|                         action=argparse.BooleanOptionalAction, default=True) |  | ||||||
|     parser.add_argument("-e", "--exit-code", help="return non-zero exit status if result is empty", action="store_true") |     parser.add_argument("-e", "--exit-code", help="return non-zero exit status if result is empty", action="store_true") | ||||||
|     parser.add_argument("-s", "--status", help="filter packages by status. Requires --from-database to be set", |  | ||||||
|                         type=BuildStatusEnum, choices=enum_values(BuildStatusEnum)) |  | ||||||
|     parser.add_argument("-u", "--username", help="build as user", default=extract_user()) |  | ||||||
|     parser.set_defaults(handler=handlers.Rebuild) |     parser.set_defaults(handler=handlers.Rebuild) | ||||||
|     return parser |     return parser | ||||||
|  |  | ||||||
| @ -717,8 +658,6 @@ def _set_repo_tree_parser(root: SubParserAction) -> argparse.ArgumentParser: | |||||||
|     parser = root.add_parser("repo-tree", help="dump repository tree", |     parser = root.add_parser("repo-tree", help="dump repository tree", | ||||||
|                              description="dump repository tree based on packages dependencies", |                              description="dump repository tree based on packages dependencies", | ||||||
|                              formatter_class=_formatter) |                              formatter_class=_formatter) | ||||||
|     parser.add_argument("-p", "--partitions", help="also divide packages by independent partitions", |  | ||||||
|                         type=int, default=1) |  | ||||||
|     parser.set_defaults(handler=handlers.Structure, lock=None, report=False, quiet=True, unsafe=True) |     parser.set_defaults(handler=handlers.Structure, lock=None, report=False, quiet=True, unsafe=True) | ||||||
|     return parser |     return parser | ||||||
|  |  | ||||||
| @ -762,13 +701,10 @@ def _set_repo_update_parser(root: SubParserAction) -> argparse.ArgumentParser: | |||||||
|                         action=argparse.BooleanOptionalAction, default=True) |                         action=argparse.BooleanOptionalAction, default=True) | ||||||
|     parser.add_argument("--dry-run", help="just perform check for updates, same as check command", action="store_true") |     parser.add_argument("--dry-run", help="just perform check for updates, same as check command", action="store_true") | ||||||
|     parser.add_argument("-e", "--exit-code", help="return non-zero exit status if result is empty", action="store_true") |     parser.add_argument("-e", "--exit-code", help="return non-zero exit status if result is empty", action="store_true") | ||||||
|     parser.add_argument("--increment", help="increment package release (pkgrel) on duplicate", |  | ||||||
|                         action=argparse.BooleanOptionalAction, default=True) |  | ||||||
|     parser.add_argument("--local", help="enable or disable checking of local packages for updates", |     parser.add_argument("--local", help="enable or disable checking of local packages for updates", | ||||||
|                         action=argparse.BooleanOptionalAction, default=True) |                         action=argparse.BooleanOptionalAction, default=True) | ||||||
|     parser.add_argument("--manual", help="include or exclude manual updates", |     parser.add_argument("--manual", help="include or exclude manual updates", | ||||||
|                         action=argparse.BooleanOptionalAction, default=True) |                         action=argparse.BooleanOptionalAction, default=True) | ||||||
|     parser.add_argument("-u", "--username", help="build as user", default=extract_user()) |  | ||||||
|     parser.add_argument("--vcs", help="fetch actual version of VCS packages", |     parser.add_argument("--vcs", help="fetch actual version of VCS packages", | ||||||
|                         action=argparse.BooleanOptionalAction, default=True) |                         action=argparse.BooleanOptionalAction, default=True) | ||||||
|     parser.add_argument("-y", "--refresh", help="download fresh package databases from the mirror before actions, " |     parser.add_argument("-y", "--refresh", help="download fresh package databases from the mirror before actions, " | ||||||
| @ -884,18 +820,16 @@ def _set_service_setup_parser(root: SubParserAction) -> argparse.ArgumentParser: | |||||||
|                              description="create initial service configuration, requires root", |                              description="create initial service configuration, requires root", | ||||||
|                              epilog="Create _minimal_ configuration for the service according to provided options.", |                              epilog="Create _minimal_ configuration for the service according to provided options.", | ||||||
|                              formatter_class=_formatter) |                              formatter_class=_formatter) | ||||||
|     parser.add_argument("--build-as-user", help="force makepkg user to the specific one") |     parser.add_argument("--build-command", help="path to build command", default=Path("/usr") / "bin" / "pkgctl") | ||||||
|     parser.add_argument("--from-configuration", help="path to default devtools pacman configuration", |     parser.add_argument("--from-configuration", help="path to default devtools pacman configuration", type=Path, | ||||||
|                         type=Path, default=Path("/usr") / "share" / "devtools" / "pacman.conf.d" / "extra.conf") |                         default=Path("/usr") / "local" / "share" / "devtools-git-poc" / "pacman.conf.d" / "extra.conf") | ||||||
|     parser.add_argument("--generate-salt", help="generate salt for user passwords", |  | ||||||
|                         action=argparse.BooleanOptionalAction, default=False) |  | ||||||
|     parser.add_argument("--makeflags-jobs", help="append MAKEFLAGS variable with parallelism set to number of cores", |     parser.add_argument("--makeflags-jobs", help="append MAKEFLAGS variable with parallelism set to number of cores", | ||||||
|                         action=argparse.BooleanOptionalAction, default=True) |                         action=argparse.BooleanOptionalAction, default=True) | ||||||
|     parser.add_argument("--mirror", help="use the specified explicitly mirror instead of including mirrorlist") |     parser.add_argument("--mirror", help="use the specified explicitly mirror instead of including mirrorlist") | ||||||
|     parser.add_argument("--multilib", help="add or do not multilib repository", |     parser.add_argument("--multilib", help="add or do not multilib repository", | ||||||
|                         action=argparse.BooleanOptionalAction, default=True) |                         action=argparse.BooleanOptionalAction, default=True) | ||||||
|     parser.add_argument("--packager", help="packager name and email", required=True) |     parser.add_argument("--packager", help="packager name and email", required=True) | ||||||
|     parser.add_argument("--server", help="server to be used for devtools. If none set, local files will be used") |     parser.add_argument("--repository", help="repository name", required=True) | ||||||
|     parser.add_argument("--sign-key", help="sign key id") |     parser.add_argument("--sign-key", help="sign key id") | ||||||
|     parser.add_argument("--sign-target", help="sign options", action="append", |     parser.add_argument("--sign-target", help="sign options", action="append", | ||||||
|                         type=SignSettings.from_option, choices=enum_values(SignSettings)) |                         type=SignSettings.from_option, choices=enum_values(SignSettings)) | ||||||
| @ -916,7 +850,8 @@ def _set_service_shell_parser(root: SubParserAction) -> argparse.ArgumentParser: | |||||||
|         argparse.ArgumentParser: created argument parser |         argparse.ArgumentParser: created argument parser | ||||||
|     """ |     """ | ||||||
|     parser = root.add_parser("service-shell", aliases=["shell"], help="invoke python shell", |     parser = root.add_parser("service-shell", aliases=["shell"], help="invoke python shell", | ||||||
|                              description="drop into python shell", formatter_class=_formatter) |                              description="drop into python shell while having created application", | ||||||
|  |                              formatter_class=_formatter) | ||||||
|     parser.add_argument("code", help="instead of dropping into shell, just execute the specified code", nargs="?") |     parser.add_argument("code", help="instead of dropping into shell, just execute the specified code", nargs="?") | ||||||
|     parser.add_argument("-v", "--verbose", help=argparse.SUPPRESS, action="store_true") |     parser.add_argument("-v", "--verbose", help=argparse.SUPPRESS, action="store_true") | ||||||
|     parser.set_defaults(handler=handlers.Shell, lock=None, report=False) |     parser.set_defaults(handler=handlers.Shell, lock=None, report=False) | ||||||
| @ -936,15 +871,15 @@ def _set_user_add_parser(root: SubParserAction) -> argparse.ArgumentParser: | |||||||
|     parser = root.add_parser("user-add", help="create or update user", |     parser = root.add_parser("user-add", help="create or update user", | ||||||
|                              description="update user for web services with the given password and role. " |                              description="update user for web services with the given password and role. " | ||||||
|                                          "In case if password was not entered it will be asked interactively", |                                          "In case if password was not entered it will be asked interactively", | ||||||
|  |                              epilog="In case of first run (i.e. if password salt is not set yet) this action requires " | ||||||
|  |                                     "root privileges because it performs write to filesystem configuration.", | ||||||
|                              formatter_class=_formatter) |                              formatter_class=_formatter) | ||||||
|     parser.add_argument("username", help="username for web service") |     parser.add_argument("username", help="username for web service") | ||||||
|     parser.add_argument("--key", help="optional PGP key used by this user. The private key must be imported") |  | ||||||
|     parser.add_argument("--packager", help="optional packager id used for build process in form of " |  | ||||||
|                                            "`Name Surname <mail@example.com>`") |  | ||||||
|     parser.add_argument("-p", "--password", help="user password. Blank password will be treated as empty password, " |     parser.add_argument("-p", "--password", help="user password. Blank password will be treated as empty password, " | ||||||
|                                                  "which is in particular must be used for OAuth2 authorization type.") |                                                  "which is in particular must be used for OAuth2 authorization type.") | ||||||
|     parser.add_argument("-r", "--role", help="user access level", |     parser.add_argument("-r", "--role", help="user access level", | ||||||
|                         type=UserAccess, choices=enum_values(UserAccess), default=UserAccess.Read) |                         type=UserAccess, choices=enum_values(UserAccess), default=UserAccess.Read) | ||||||
|  |     parser.add_argument("-s", "--secure", help="set file permissions to user-only", action="store_true") | ||||||
|     parser.set_defaults(handler=handlers.Users, action=Action.Update, architecture=[""], lock=None, report=False, |     parser.set_defaults(handler=handlers.Users, action=Action.Update, architecture=[""], lock=None, report=False, | ||||||
|                         quiet=True) |                         quiet=True) | ||||||
|     return parser |     return parser | ||||||
| @ -966,8 +901,8 @@ def _set_user_list_parser(root: SubParserAction) -> argparse.ArgumentParser: | |||||||
|     parser.add_argument("username", help="filter users by username", nargs="?") |     parser.add_argument("username", help="filter users by username", nargs="?") | ||||||
|     parser.add_argument("-e", "--exit-code", help="return non-zero exit status if result is empty", action="store_true") |     parser.add_argument("-e", "--exit-code", help="return non-zero exit status if result is empty", action="store_true") | ||||||
|     parser.add_argument("-r", "--role", help="filter users by role", type=UserAccess, choices=enum_values(UserAccess)) |     parser.add_argument("-r", "--role", help="filter users by role", type=UserAccess, choices=enum_values(UserAccess)) | ||||||
|     parser.set_defaults(handler=handlers.Users, action=Action.List, architecture=[""], lock=None, report=False, |     parser.set_defaults(handler=handlers.Users, action=Action.List, architecture=[""], lock=None, report=False,  # nosec | ||||||
|                         quiet=True, unsafe=True) |                         password="", quiet=True, unsafe=True) | ||||||
|     return parser |     return parser | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -985,8 +920,8 @@ def _set_user_remove_parser(root: SubParserAction) -> argparse.ArgumentParser: | |||||||
|                              description="remove user from the user mapping and update the configuration", |                              description="remove user from the user mapping and update the configuration", | ||||||
|                              formatter_class=_formatter) |                              formatter_class=_formatter) | ||||||
|     parser.add_argument("username", help="username for web service") |     parser.add_argument("username", help="username for web service") | ||||||
|     parser.set_defaults(handler=handlers.Users, action=Action.Remove, architecture=[""], lock=None, report=False, |     parser.set_defaults(handler=handlers.Users, action=Action.Remove, architecture=[""], lock=None, report=False,  # nosec | ||||||
|                         quiet=True) |                         password="", quiet=True) | ||||||
|     return parser |     return parser | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -1006,15 +941,18 @@ def _set_web_parser(root: SubParserAction) -> argparse.ArgumentParser: | |||||||
|     return parser |     return parser | ||||||
|  |  | ||||||
|  |  | ||||||
| def run() -> int: | def run() -> None: | ||||||
|     """ |     """ | ||||||
|     run application instance |     run application instance | ||||||
|  |  | ||||||
|     Returns: |  | ||||||
|         int: application status code |  | ||||||
|     """ |     """ | ||||||
|  |     if __name__ == "__main__": | ||||||
|         args_parser = _parser() |         args_parser = _parser() | ||||||
|         args = args_parser.parse_args() |         args = args_parser.parse_args() | ||||||
|  |  | ||||||
|         handler: handlers.Handler = args.handler |         handler: handlers.Handler = args.handler | ||||||
|     return handler.execute(args) |         status = handler.execute(args) | ||||||
|  |  | ||||||
|  |         sys.exit(status) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | run() | ||||||
|  | |||||||
| @ -17,12 +17,10 @@ | |||||||
| # You should have received a copy of the GNU General Public License | # You should have received a copy of the GNU General Public License | ||||||
| # along with this program. If not, see <http://www.gnu.org/licenses/>. | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
| # | # | ||||||
| from collections.abc import Callable, Iterable | from collections.abc import Iterable | ||||||
|  |  | ||||||
| from ahriman.application.application.application_packages import ApplicationPackages | from ahriman.application.application.application_packages import ApplicationPackages | ||||||
| from ahriman.application.application.application_repository import ApplicationRepository | from ahriman.application.application.application_repository import ApplicationRepository | ||||||
| from ahriman.core.formatters import UpdatePrinter |  | ||||||
| from ahriman.core.tree import Tree |  | ||||||
| from ahriman.models.package import Package | from ahriman.models.package import Package | ||||||
| from ahriman.models.result import Result | from ahriman.models.result import Result | ||||||
|  |  | ||||||
| @ -37,15 +35,14 @@ class Application(ApplicationPackages, ApplicationRepository): | |||||||
|  |  | ||||||
|             >>> from ahriman.core.configuration import Configuration |             >>> from ahriman.core.configuration import Configuration | ||||||
|             >>> from ahriman.models.package_source import PackageSource |             >>> from ahriman.models.package_source import PackageSource | ||||||
|             >>> from ahriman.models.repository_id import RepositoryId |  | ||||||
|             >>> |             >>> | ||||||
|             >>> configuration = Configuration() |             >>> configuration = Configuration() | ||||||
|             >>> application = Application(RepositoryId("x86_64", None), configuration, report=True) |             >>> application = Application("x86_64", configuration, report=True, unsafe=False) | ||||||
|             >>> # add packages to build queue |             >>> # add packages to build queue | ||||||
|             >>> application.add(["ahriman"], PackageSource.AUR) |             >>> application.add(["ahriman"], PackageSource.AUR, without_dependencies=False) | ||||||
|             >>> |             >>> | ||||||
|             >>> # check for updates |             >>> # check for updates | ||||||
|             >>> updates = application.updates([], aur=True, local=True, manual=True, vcs=True) |             >>> updates = application.updates([], aur=True, local=True, manual=True, vcs=True, log_fn=print) | ||||||
|             >>> # updates for specified packages |             >>> # updates for specified packages | ||||||
|             >>> application.update(updates) |             >>> application.update(updates) | ||||||
|  |  | ||||||
| @ -92,22 +89,6 @@ class Application(ApplicationPackages, ApplicationRepository): | |||||||
|         """ |         """ | ||||||
|         self.repository.triggers.on_stop() |         self.repository.triggers.on_stop() | ||||||
|  |  | ||||||
|     def print_updates(self, packages: list[Package], *, log_fn: Callable[[str], None]) -> None: |  | ||||||
|         """ |  | ||||||
|         print list of packages to be built. This method will build dependency tree and print updates accordingly |  | ||||||
|  |  | ||||||
|         Args: |  | ||||||
|             packages(list[Package]): package list to be printed |  | ||||||
|             log_fn(Callable[[str], None]): logger function to log updates |  | ||||||
|         """ |  | ||||||
|         local_versions = {package.base: package.version for package in self.repository.packages()} |  | ||||||
|  |  | ||||||
|         tree = Tree.resolve(packages) |  | ||||||
|         for level in tree: |  | ||||||
|             for package in level: |  | ||||||
|                 UpdatePrinter(package, local_versions.get(package.base)).print( |  | ||||||
|                     verbose=True, log_fn=log_fn, separator=" -> ") |  | ||||||
|  |  | ||||||
|     def with_dependencies(self, packages: list[Package], *, process_dependencies: bool) -> list[Package]: |     def with_dependencies(self, packages: list[Package], *, process_dependencies: bool) -> list[Package]: | ||||||
|         """ |         """ | ||||||
|         add missing dependencies to list of packages |         add missing dependencies to list of packages | ||||||
| @ -115,25 +96,21 @@ class Application(ApplicationPackages, ApplicationRepository): | |||||||
|         Args: |         Args: | ||||||
|             packages(list[Package]): list of source packages of which dependencies have to be processed |             packages(list[Package]): list of source packages of which dependencies have to be processed | ||||||
|             process_dependencies(bool): if no set, dependencies will not be processed |             process_dependencies(bool): if no set, dependencies will not be processed | ||||||
|  |  | ||||||
|         Returns: |  | ||||||
|             list[Package]: updated packages list. Packager for dependencies will be copied from |  | ||||||
|         original package |  | ||||||
|         """ |         """ | ||||||
|         def missing_dependencies(source: Iterable[Package]) -> dict[str, str | None]: |         def missing_dependencies(source: Iterable[Package]) -> set[str]: | ||||||
|             # append list of known packages with packages which are in current sources |             # build initial list of dependencies | ||||||
|             satisfied_packages = known_packages | { |             result = set() | ||||||
|                 single |             for package in source: | ||||||
|                 for package in source |                 result.update(package.depends_build) | ||||||
|                 for single in package.packages_full |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             return { |             # remove ones which are already well-known | ||||||
|                 dependency: package.packager |             result = result.difference(known_packages) | ||||||
|                 for package in source |  | ||||||
|                 for dependency in package.depends_build |             # remove ones which are in this list already | ||||||
|                 if dependency not in satisfied_packages |             for package in source: | ||||||
|             } |                 result = result.difference(package.packages_full) | ||||||
|  |  | ||||||
|  |             return result | ||||||
|  |  | ||||||
|         if not process_dependencies or not packages: |         if not process_dependencies or not packages: | ||||||
|             return packages |             return packages | ||||||
| @ -142,16 +119,8 @@ class Application(ApplicationPackages, ApplicationRepository): | |||||||
|         with_dependencies = {package.base: package for package in packages} |         with_dependencies = {package.base: package for package in packages} | ||||||
|  |  | ||||||
|         while missing := missing_dependencies(with_dependencies.values()): |         while missing := missing_dependencies(with_dependencies.values()): | ||||||
|             for package_name, username in missing.items(): |             for package_name in missing: | ||||||
|                 if (source_dir := self.repository.paths.cache_for(package_name)).is_dir(): |                 package = Package.from_aur(package_name, self.repository.pacman) | ||||||
|                     # there is local cache, load package from it |  | ||||||
|                     package = Package.from_build(source_dir, self.repository.architecture, username) |  | ||||||
|                 else: |  | ||||||
|                     package = Package.from_aur(package_name, self.repository.pacman, username) |  | ||||||
|                 with_dependencies[package.base] = package |                 with_dependencies[package.base] = package | ||||||
|  |  | ||||||
|                 # register package in local database |  | ||||||
|                 self.database.remote_update(package) |  | ||||||
|                 self.repository.reporter.set_unknown(package) |  | ||||||
|  |  | ||||||
|         return list(with_dependencies.values()) |         return list(with_dependencies.values()) | ||||||
|  | |||||||
| @ -26,7 +26,6 @@ from typing import Any | |||||||
|  |  | ||||||
| from ahriman.application.application.application_properties import ApplicationProperties | from ahriman.application.application.application_properties import ApplicationProperties | ||||||
| from ahriman.core.build_tools.sources import Sources | from ahriman.core.build_tools.sources import Sources | ||||||
| from ahriman.core.exceptions import UnknownPackageError |  | ||||||
| from ahriman.core.util import package_like | from ahriman.core.util import package_like | ||||||
| from ahriman.models.package import Package | from ahriman.models.package import Package | ||||||
| from ahriman.models.package_source import PackageSource | from ahriman.models.package_source import PackageSource | ||||||
| @ -44,26 +43,20 @@ class ApplicationPackages(ApplicationProperties): | |||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             source(str): path to package archive |             source(str): path to package archive | ||||||
|  |  | ||||||
|         Raises: |  | ||||||
|             UnknownPackageError: if specified path doesn't exist |  | ||||||
|         """ |         """ | ||||||
|         local_path = Path(source) |         local_path = Path(source) | ||||||
|         if not local_path.is_file(): |  | ||||||
|             raise UnknownPackageError(source) |  | ||||||
|  |  | ||||||
|         dst = self.repository.paths.packages / local_path.name |         dst = self.repository.paths.packages / local_path.name | ||||||
|         shutil.copy(local_path, dst) |         shutil.copy(local_path, dst) | ||||||
|  |  | ||||||
|     def _add_aur(self, source: str, username: str | None) -> None: |     def _add_aur(self, source: str) -> None: | ||||||
|         """ |         """ | ||||||
|         add package from AUR |         add package from AUR | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             source(str): package base name |             source(str): package base name | ||||||
|             username(str | None): optional override of username for build process |  | ||||||
|         """ |         """ | ||||||
|         package = Package.from_aur(source, self.repository.pacman, username) |         package = Package.from_aur(source, self.repository.pacman) | ||||||
|  |  | ||||||
|         self.database.build_queue_insert(package) |         self.database.build_queue_insert(package) | ||||||
|         self.database.remote_update(package) |         self.database.remote_update(package) | ||||||
|  |  | ||||||
| @ -73,37 +66,23 @@ class ApplicationPackages(ApplicationProperties): | |||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             source(str): path to local directory |             source(str): path to local directory | ||||||
|  |  | ||||||
|         Raises: |  | ||||||
|             UnknownPackageError: if specified package is unknown or doesn't exist |  | ||||||
|         """ |         """ | ||||||
|         local_dir = Path(source) |         local_dir = Path(source) | ||||||
|         if not local_dir.is_dir(): |  | ||||||
|             raise UnknownPackageError(source) |  | ||||||
|  |  | ||||||
|         for full_path in filter(package_like, local_dir.iterdir()): |         for full_path in filter(package_like, local_dir.iterdir()): | ||||||
|             self._add_archive(str(full_path)) |             self._add_archive(str(full_path)) | ||||||
|  |  | ||||||
|     def _add_local(self, source: str, username: str | None) -> None: |     def _add_local(self, source: str) -> None: | ||||||
|         """ |         """ | ||||||
|         add package from local PKGBUILDs |         add package from local PKGBUILDs | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             source(str): path to directory with local source files |             source(str): path to directory with local source files | ||||||
|             username(str | None): optional override of username for build process |  | ||||||
|  |  | ||||||
|         Raises: |  | ||||||
|             UnknownPackageError: if specified package is unknown or doesn't exist |  | ||||||
|         """ |         """ | ||||||
|         if (source_dir := Path(source)).is_dir(): |         source_dir = Path(source) | ||||||
|             package = Package.from_build(source_dir, self.architecture, username) |         package = Package.from_build(source_dir, self.architecture) | ||||||
|         cache_dir = self.repository.paths.cache_for(package.base) |         cache_dir = self.repository.paths.cache_for(package.base) | ||||||
|             shutil.copytree(source_dir, cache_dir, dirs_exist_ok=True)  # copy package to store in caches |         shutil.copytree(source_dir, cache_dir)  # copy package to store in caches | ||||||
|         Sources.init(cache_dir)  # we need to run init command in directory where we do have permissions |         Sources.init(cache_dir)  # we need to run init command in directory where we do have permissions | ||||||
|         elif (source_dir := self.repository.paths.cache_for(source)).is_dir(): |  | ||||||
|             package = Package.from_build(source_dir, self.architecture, username) |  | ||||||
|         else: |  | ||||||
|             raise UnknownPackageError(source) |  | ||||||
|  |  | ||||||
|         self.database.build_queue_insert(package) |         self.database.build_queue_insert(package) | ||||||
|  |  | ||||||
| @ -113,47 +92,39 @@ class ApplicationPackages(ApplicationProperties): | |||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             source(str): remote URL of the package archive |             source(str): remote URL of the package archive | ||||||
|  |  | ||||||
|         Raises: |  | ||||||
|             UnknownPackageError: if specified package is unknown or doesn't exist |  | ||||||
|         """ |         """ | ||||||
|  |         dst = self.repository.paths.packages / Path(source).name  # URL is path, is not it? | ||||||
|         # timeout=None to suppress pylint warns. Also suppress bandit warnings |         # timeout=None to suppress pylint warns. Also suppress bandit warnings | ||||||
|         try: |  | ||||||
|         response = requests.get(source, stream=True, timeout=None)  # nosec |         response = requests.get(source, stream=True, timeout=None)  # nosec | ||||||
|         response.raise_for_status() |         response.raise_for_status() | ||||||
|         except Exception: |  | ||||||
|             raise UnknownPackageError(source) |  | ||||||
|  |  | ||||||
|         dst = self.repository.paths.packages / Path(source).name  # URL is path, is not it? |  | ||||||
|         with dst.open("wb") as local_file: |         with dst.open("wb") as local_file: | ||||||
|             for chunk in response.iter_content(chunk_size=1024): |             for chunk in response.iter_content(chunk_size=1024): | ||||||
|                 local_file.write(chunk) |                 local_file.write(chunk) | ||||||
|  |  | ||||||
|     def _add_repository(self, source: str, username: str | None) -> None: |     def _add_repository(self, source: str, *_: Any) -> None: | ||||||
|         """ |         """ | ||||||
|         add package from official repository |         add package from official repository | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             source(str): package base name |             source(str): package base name | ||||||
|             username(str | None): optional override of username for build process |  | ||||||
|         """ |         """ | ||||||
|         package = Package.from_official(source, self.repository.pacman, username) |         package = Package.from_official(source, self.repository.pacman) | ||||||
|         self.database.build_queue_insert(package) |         self.database.build_queue_insert(package) | ||||||
|         self.database.remote_update(package) |         self.database.remote_update(package) | ||||||
|  |  | ||||||
|     def add(self, names: Iterable[str], source: PackageSource, username: str | None = None) -> None: |     def add(self, names: Iterable[str], source: PackageSource) -> None: | ||||||
|         """ |         """ | ||||||
|         add packages for the next build |         add packages for the next build | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             names(Iterable[str]): list of package bases to add |             names(Iterable[str]): list of package bases to add | ||||||
|             source(PackageSource): package source to add |             source(PackageSource): package source to add | ||||||
|             username(str | None, optional): optional override of username for build process (Default value = None) |  | ||||||
|         """ |         """ | ||||||
|         for name in names: |         for name in names: | ||||||
|             resolved_source = source.resolve(name, self.repository.paths) |             resolved_source = source.resolve(name) | ||||||
|             fn = getattr(self, f"_add_{resolved_source.value}") |             fn = getattr(self, f"_add_{resolved_source.value}") | ||||||
|             fn(name, username) |             fn(name) | ||||||
|  |  | ||||||
|     def on_result(self, result: Result) -> None: |     def on_result(self, result: Result) -> None: | ||||||
|         """ |         """ | ||||||
|  | |||||||
| @ -22,7 +22,6 @@ from ahriman.core.database import SQLite | |||||||
| from ahriman.core.log import LazyLogging | from ahriman.core.log import LazyLogging | ||||||
| from ahriman.core.repository import Repository | from ahriman.core.repository import Repository | ||||||
| from ahriman.models.pacman_synchronization import PacmanSynchronization | from ahriman.models.pacman_synchronization import PacmanSynchronization | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class ApplicationProperties(LazyLogging): | class ApplicationProperties(LazyLogging): | ||||||
| @ -30,36 +29,27 @@ class ApplicationProperties(LazyLogging): | |||||||
|     application base properties class |     application base properties class | ||||||
|  |  | ||||||
|     Attributes: |     Attributes: | ||||||
|  |         architecture(str): repository architecture | ||||||
|         configuration(Configuration): configuration instance |         configuration(Configuration): configuration instance | ||||||
|         database(SQLite): database instance |         database(SQLite): database instance | ||||||
|         repository(Repository): repository instance |         repository(Repository): repository instance | ||||||
|         repository_id(RepositoryId): repository unique identifier |  | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     def __init__(self, repository_id: RepositoryId, configuration: Configuration, *, report: bool, |     def __init__(self, architecture: str, configuration: Configuration, *, report: bool, unsafe: bool, | ||||||
|                  refresh_pacman_database: PacmanSynchronization = PacmanSynchronization.Disabled) -> None: |                  refresh_pacman_database: PacmanSynchronization = PacmanSynchronization.Disabled) -> None: | ||||||
|         """ |         """ | ||||||
|         default constructor |         default constructor | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|             refresh_pacman_database(PacmanSynchronization, optional): pacman database synchronization level |             refresh_pacman_database(PacmanSynchronization, optional): pacman database synchronization level | ||||||
|                 (Default value = PacmanSynchronization.Disabled) |                 (Default value = PacmanSynchronization.Disabled) | ||||||
|         """ |         """ | ||||||
|         self.configuration = configuration |         self.configuration = configuration | ||||||
|         self.repository_id = repository_id |         self.architecture = architecture | ||||||
|         self.database = SQLite.load(configuration) |         self.database = SQLite.load(configuration) | ||||||
|         self.repository = Repository.load(repository_id, configuration, self.database, report=report, |         self.repository = Repository.load(architecture, configuration, self.database, report=report, unsafe=unsafe, | ||||||
|                                           refresh_pacman_database=refresh_pacman_database) |                                           refresh_pacman_database=refresh_pacman_database) | ||||||
|  |  | ||||||
|     @property |  | ||||||
|     def architecture(self) -> str: |  | ||||||
|         """ |  | ||||||
|         repository architecture for backward compatibility |  | ||||||
|  |  | ||||||
|         Returns: |  | ||||||
|             str: repository architecture |  | ||||||
|         """ |  | ||||||
|         return self.repository_id.architecture |  | ||||||
|  | |||||||
| @ -17,14 +17,14 @@ | |||||||
| # You should have received a copy of the GNU General Public License | # You should have received a copy of the GNU General Public License | ||||||
| # along with this program. If not, see <http://www.gnu.org/licenses/>. | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
| # | # | ||||||
| from collections.abc import Iterable | from collections.abc import Callable, Iterable | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
|  |  | ||||||
| from ahriman.application.application.application_properties import ApplicationProperties | from ahriman.application.application.application_properties import ApplicationProperties | ||||||
| from ahriman.core.build_tools.sources import Sources | from ahriman.core.build_tools.sources import Sources | ||||||
|  | from ahriman.core.formatters import UpdatePrinter | ||||||
| from ahriman.core.tree import Tree | from ahriman.core.tree import Tree | ||||||
| from ahriman.models.package import Package | from ahriman.models.package import Package | ||||||
| from ahriman.models.packagers import Packagers |  | ||||||
| from ahriman.models.result import Result | from ahriman.models.result import Result | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -83,7 +83,7 @@ class ApplicationRepository(ApplicationProperties): | |||||||
|                 if archive.filepath is None: |                 if archive.filepath is None: | ||||||
|                     self.logger.warning("filepath is empty for %s", package.base) |                     self.logger.warning("filepath is empty for %s", package.base) | ||||||
|                     continue  # avoid mypy warning |                     continue  # avoid mypy warning | ||||||
|                 self.repository.sign.process_sign_package(archive.filepath, None) |                 self.repository.sign.process_sign_package(archive.filepath, package.base) | ||||||
|         # sign repository database if set |         # sign repository database if set | ||||||
|         self.repository.sign.process_sign_repository(self.repository.repo.repo_path) |         self.repository.sign.process_sign_repository(self.repository.repo.repo_path) | ||||||
|         # process triggers |         # process triggers | ||||||
| @ -104,14 +104,14 @@ class ApplicationRepository(ApplicationProperties): | |||||||
|             packages: list[str] = [] |             packages: list[str] = [] | ||||||
|             for single in probe.packages: |             for single in probe.packages: | ||||||
|                 try: |                 try: | ||||||
|                     _ = Package.from_aur(single, self.repository.pacman, None) |                     _ = Package.from_aur(single, self.repository.pacman) | ||||||
|                 except Exception: |                 except Exception: | ||||||
|                     packages.append(single) |                     packages.append(single) | ||||||
|             return packages |             return packages | ||||||
|  |  | ||||||
|         def unknown_local(probe: Package) -> list[str]: |         def unknown_local(probe: Package) -> list[str]: | ||||||
|             cache_dir = self.repository.paths.cache_for(probe.base) |             cache_dir = self.repository.paths.cache_for(probe.base) | ||||||
|             local = Package.from_build(cache_dir, self.architecture, None) |             local = Package.from_build(cache_dir, self.architecture) | ||||||
|             packages = set(probe.packages.keys()).difference(local.packages.keys()) |             packages = set(probe.packages.keys()).difference(local.packages.keys()) | ||||||
|             return list(packages) |             return list(packages) | ||||||
|  |  | ||||||
| @ -123,16 +123,12 @@ class ApplicationRepository(ApplicationProperties): | |||||||
|                 result.extend(unknown_aur(package))  # local package not found |                 result.extend(unknown_aur(package))  # local package not found | ||||||
|         return result |         return result | ||||||
|  |  | ||||||
|     def update(self, updates: Iterable[Package], packagers: Packagers | None = None, *, |     def update(self, updates: Iterable[Package]) -> Result: | ||||||
|                bump_pkgrel: bool = False) -> Result: |  | ||||||
|         """ |         """ | ||||||
|         run package updates |         run package updates | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             updates(Iterable[Package]): list of packages to update |             updates(Iterable[Package]): list of packages to update | ||||||
|             packagers(Packagers | None, optional): optional override of username for build process |  | ||||||
|                 (Default value = None) |  | ||||||
|             bump_pkgrel(bool, optional): bump pkgrel in case of local version conflict (Default value = False) |  | ||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             Result: update result |             Result: update result | ||||||
| @ -140,7 +136,7 @@ class ApplicationRepository(ApplicationProperties): | |||||||
|         def process_update(paths: Iterable[Path], result: Result) -> None: |         def process_update(paths: Iterable[Path], result: Result) -> None: | ||||||
|             if not paths: |             if not paths: | ||||||
|                 return  # don't need to process if no update supplied |                 return  # don't need to process if no update supplied | ||||||
|             update_result = self.repository.process_update(paths, packagers) |             update_result = self.repository.process_update(paths) | ||||||
|             self.on_result(result.merge(update_result)) |             self.on_result(result.merge(update_result)) | ||||||
|  |  | ||||||
|         # process built packages |         # process built packages | ||||||
| @ -152,14 +148,14 @@ class ApplicationRepository(ApplicationProperties): | |||||||
|         tree = Tree.resolve(updates) |         tree = Tree.resolve(updates) | ||||||
|         for num, level in enumerate(tree): |         for num, level in enumerate(tree): | ||||||
|             self.logger.info("processing level #%i %s", num, [package.base for package in level]) |             self.logger.info("processing level #%i %s", num, [package.base for package in level]) | ||||||
|             build_result = self.repository.process_build(level, packagers, bump_pkgrel=bump_pkgrel) |             build_result = self.repository.process_build(level) | ||||||
|             packages = self.repository.packages_built() |             packages = self.repository.packages_built() | ||||||
|             process_update(packages, build_result) |             process_update(packages, build_result) | ||||||
|  |  | ||||||
|         return build_result |         return build_result | ||||||
|  |  | ||||||
|     def updates(self, filter_packages: Iterable[str], *, |     def updates(self, filter_packages: Iterable[str], *, | ||||||
|                 aur: bool, local: bool, manual: bool, vcs: bool) -> list[Package]: |                 aur: bool, local: bool, manual: bool, vcs: bool, log_fn: Callable[[str], None]) -> list[Package]: | ||||||
|         """ |         """ | ||||||
|         get list of packages to run update process |         get list of packages to run update process | ||||||
|  |  | ||||||
| @ -169,6 +165,7 @@ class ApplicationRepository(ApplicationProperties): | |||||||
|             local(bool): enable or disable checking of local packages for updates |             local(bool): enable or disable checking of local packages for updates | ||||||
|             manual(bool): include or exclude manual updates |             manual(bool): include or exclude manual updates | ||||||
|             vcs(bool): enable or disable checking of VCS packages |             vcs(bool): enable or disable checking of VCS packages | ||||||
|  |             log_fn(Callable[[str], None]): logger function to log updates | ||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             list[Package]: list of out-of-dated packages |             list[Package]: list of out-of-dated packages | ||||||
| @ -182,4 +179,14 @@ class ApplicationRepository(ApplicationProperties): | |||||||
|         if manual: |         if manual: | ||||||
|             updates.update({package.base: package for package in self.repository.updates_manual()}) |             updates.update({package.base: package for package in self.repository.updates_manual()}) | ||||||
|  |  | ||||||
|         return [package for _, package in sorted(updates.items())] |         local_versions = {package.base: package.version for package in self.repository.packages()} | ||||||
|  |         updated_packages = [package for _, package in sorted(updates.items())] | ||||||
|  |  | ||||||
|  |         # reorder updates according to the dependency tree | ||||||
|  |         tree = Tree.resolve(updated_packages) | ||||||
|  |         for level in tree: | ||||||
|  |             for package in level: | ||||||
|  |                 UpdatePrinter(package, local_versions.get(package.base)).print( | ||||||
|  |                     verbose=True, log_fn=log_fn, separator=" -> ") | ||||||
|  |  | ||||||
|  |         return updated_packages | ||||||
|  | |||||||
| @ -22,8 +22,6 @@ import argparse | |||||||
| from ahriman.application.application import Application | from ahriman.application.application import Application | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.models.packagers import Packagers |  | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Add(Handler): | class Add(Handler): | ||||||
| @ -32,27 +30,27 @@ class Add(Handler): | |||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         application = Application(repository_id, configuration, report=report, refresh_pacman_database=args.refresh) |         application = Application(architecture, configuration, | ||||||
|  |                                   report=report, unsafe=unsafe, refresh_pacman_database=args.refresh) | ||||||
|         application.on_start() |         application.on_start() | ||||||
|         application.add(args.package, args.source, args.username) |         application.add(args.package, args.source) | ||||||
|         if not args.now: |         if not args.now: | ||||||
|             return |             return | ||||||
|  |  | ||||||
|         packages = application.updates(args.package, aur=False, local=False, manual=True, vcs=False) |         packages = application.updates(args.package, aur=False, local=False, manual=True, vcs=False, | ||||||
|  |                                        log_fn=application.logger.info) | ||||||
|         packages = application.with_dependencies(packages, process_dependencies=args.dependencies) |         packages = application.with_dependencies(packages, process_dependencies=args.dependencies) | ||||||
|         packagers = Packagers(args.username, {package.base: package.packager for package in packages}) |         result = application.update(packages) | ||||||
|  |  | ||||||
|         application.print_updates(packages, log_fn=application.logger.info) |  | ||||||
|         result = application.update(packages, packagers, bump_pkgrel=args.increment) |  | ||||||
|         Add.check_if_empty(args.exit_code, result.is_empty) |         Add.check_if_empty(args.exit_code, result.is_empty) | ||||||
|  | |||||||
| @ -26,7 +26,6 @@ from tarfile import TarFile | |||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.core.database import SQLite | from ahriman.core.database import SQLite | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Backup(Handler): | class Backup(Handler): | ||||||
| @ -37,16 +36,17 @@ class Backup(Handler): | |||||||
|     ALLOW_AUTO_ARCHITECTURE_RUN = False  # it should be called only as "no-architecture" |     ALLOW_AUTO_ARCHITECTURE_RUN = False  # it should be called only as "no-architecture" | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         backup_paths = Backup.get_paths(configuration) |         backup_paths = Backup.get_paths(configuration) | ||||||
|         with TarFile(args.path, mode="w") as archive:  # well we don't actually use compression |         with TarFile(args.path, mode="w") as archive:  # well we don't actually use compression | ||||||
|  | |||||||
| @ -22,7 +22,6 @@ import argparse | |||||||
| from ahriman.application.application import Application | from ahriman.application.application import Application | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Clean(Handler): | class Clean(Handler): | ||||||
| @ -31,18 +30,19 @@ class Clean(Handler): | |||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         application = Application(repository_id, configuration, report=report) |         application = Application(architecture, configuration, report=report, unsafe=unsafe) | ||||||
|         application.on_start() |         application.on_start() | ||||||
|         application.clean(cache=args.cache, chroot=args.chroot, manual=args.manual, packages=args.packages, |         application.clean(cache=args.cache, chroot=args.chroot, manual=args.manual, packages=args.packages, | ||||||
|                           pacman=args.pacman) |                           pacman=args.pacman) | ||||||
|  | |||||||
| @ -22,7 +22,6 @@ import threading | |||||||
|  |  | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Daemon(Handler): | class Daemon(Handler): | ||||||
| @ -31,20 +30,21 @@ class Daemon(Handler): | |||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         from ahriman.application.handlers import Update |         from ahriman.application.handlers import Update | ||||||
|         Update.run(args, repository_id, configuration, report=report) |         Update.run(args, architecture, configuration, report=report, unsafe=unsafe) | ||||||
|         timer = threading.Timer(args.interval, Daemon.run, args=[args, repository_id, configuration], |         timer = threading.Timer(args.interval, Daemon.run, args=[args, architecture, configuration], | ||||||
|                                 kwargs={"report": report}) |                                 kwargs={"report": report, "unsafe": unsafe}) | ||||||
|         timer.start() |         timer.start() | ||||||
|         timer.join() |         timer.join() | ||||||
|  | |||||||
| @ -21,8 +21,7 @@ import argparse | |||||||
|  |  | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.core.formatters import ConfigurationPathsPrinter, ConfigurationPrinter, StringPrinter | from ahriman.core.formatters import ConfigurationPrinter | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Dump(Handler): | class Dump(Handler): | ||||||
| @ -33,23 +32,18 @@ class Dump(Handler): | |||||||
|     ALLOW_AUTO_ARCHITECTURE_RUN = False |     ALLOW_AUTO_ARCHITECTURE_RUN = False | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         root, _ = configuration.check_loaded() |  | ||||||
|         ConfigurationPathsPrinter(root, configuration.includes).print(verbose=True, separator=" = ") |  | ||||||
|  |  | ||||||
|         # empty line |  | ||||||
|         StringPrinter("").print(verbose=False) |  | ||||||
|  |  | ||||||
|         dump = configuration.dump() |         dump = configuration.dump() | ||||||
|         for section, values in sorted(dump.items()): |         for section, values in sorted(dump.items()): | ||||||
|             ConfigurationPrinter(section, values).print(verbose=not args.secure, separator=" = ") |             ConfigurationPrinter(section, values).print(verbose=not args.secure, separator=" = ") | ||||||
|  | |||||||
| @ -26,7 +26,6 @@ from ahriman.application.lock import Lock | |||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.core.exceptions import ExitCode, MissingArchitectureError, MultipleArchitecturesError | from ahriman.core.exceptions import ExitCode, MissingArchitectureError, MultipleArchitecturesError | ||||||
| from ahriman.core.log import Log | from ahriman.core.log import Log | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
| from ahriman.models.repository_paths import RepositoryPaths | from ahriman.models.repository_paths import RepositoryPaths | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -51,26 +50,53 @@ class Handler: | |||||||
|     ALLOW_MULTI_ARCHITECTURE_RUN = True |     ALLOW_MULTI_ARCHITECTURE_RUN = True | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def call(cls, args: argparse.Namespace, repository_id: RepositoryId) -> bool: |     def architectures_extract(cls, args: argparse.Namespace) -> list[str]: | ||||||
|  |         """ | ||||||
|  |         get known architectures | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             args(argparse.Namespace): command line args | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             list[str]: list of architectures for which tree is created | ||||||
|  |  | ||||||
|  |         Raises: | ||||||
|  |             MissingArchitecture: if no architecture set and automatic detection is not allowed or failed | ||||||
|  |         """ | ||||||
|  |         if not cls.ALLOW_AUTO_ARCHITECTURE_RUN and args.architecture is None: | ||||||
|  |             # for some parsers (e.g. config) we need to run with specific architecture | ||||||
|  |             # for those cases architecture must be set explicitly | ||||||
|  |             raise MissingArchitectureError(args.command) | ||||||
|  |         if args.architecture:  # architecture is specified explicitly | ||||||
|  |             return sorted(set(args.architecture)) | ||||||
|  |  | ||||||
|  |         configuration = Configuration() | ||||||
|  |         configuration.load(args.configuration) | ||||||
|  |         # wtf??? | ||||||
|  |         root = configuration.getpath("repository", "root")  # pylint: disable=assignment-from-no-return | ||||||
|  |         architectures = RepositoryPaths.known_architectures(root) | ||||||
|  |  | ||||||
|  |         if not architectures:  # well we did not find anything | ||||||
|  |             raise MissingArchitectureError(args.command) | ||||||
|  |         return sorted(architectures) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def call(cls, args: argparse.Namespace, architecture: str) -> bool: | ||||||
|         """ |         """ | ||||||
|         additional function to wrap all calls for multiprocessing library |         additional function to wrap all calls for multiprocessing library | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             bool: True on success, False otherwise |             bool: True on success, False otherwise | ||||||
|         """ |         """ | ||||||
|         try: |         try: | ||||||
|             configuration = Configuration.from_path(args.configuration, repository_id) |             configuration = Configuration.from_path(args.configuration, architecture) | ||||||
|  |             Log.load(configuration, quiet=args.quiet, report=args.report) | ||||||
|             log_handler = Log.handler(args.log_handler) |             with Lock(args, architecture, configuration): | ||||||
|             Log.load(configuration, log_handler, quiet=args.quiet, report=args.report) |                 cls.run(args, architecture, configuration, report=args.report, unsafe=args.unsafe) | ||||||
|  |  | ||||||
|             with Lock(args, repository_id, configuration): |  | ||||||
|                 cls.run(args, repository_id, configuration, report=args.report) |  | ||||||
|  |  | ||||||
|             return True |             return True | ||||||
|         except ExitCode: |         except ExitCode: | ||||||
|             return False |             return False | ||||||
| @ -91,72 +117,35 @@ class Handler: | |||||||
|             int: 0 on success, 1 otherwise |             int: 0 on success, 1 otherwise | ||||||
|  |  | ||||||
|         Raises: |         Raises: | ||||||
|             MultipleArchitecturesError: if more than one architecture supplied and no multi architecture supported |             MultipleArchitectures: if more than one architecture supplied and no multi architecture supported | ||||||
|         """ |         """ | ||||||
|         repositories = cls.repositories_extract(args) |         architectures = cls.architectures_extract(args) | ||||||
|  |  | ||||||
|         # actually we do not have to spawn another process if it is single-process application, do we? |         # actually we do not have to spawn another process if it is single-process application, do we? | ||||||
|         if len(repositories) > 1: |         if len(architectures) > 1: | ||||||
|             if not cls.ALLOW_MULTI_ARCHITECTURE_RUN: |             if not cls.ALLOW_MULTI_ARCHITECTURE_RUN: | ||||||
|                 raise MultipleArchitecturesError(args.command) |                 raise MultipleArchitecturesError(args.command) | ||||||
|  |  | ||||||
|             with Pool(len(repositories)) as pool: |             with Pool(len(architectures)) as pool: | ||||||
|                 result = pool.starmap(cls.call, [(args, repository_id) for repository_id in repositories]) |                 result = pool.starmap( | ||||||
|  |                     cls.call, [(args, architecture) for architecture in architectures]) | ||||||
|         else: |         else: | ||||||
|             result = [cls.call(args, repositories.pop())] |             result = [cls.call(args, architectures.pop())] | ||||||
|  |  | ||||||
|         return 0 if all(result) else 1 |         return 0 if all(result) else 1 | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def repositories_extract(cls, args: argparse.Namespace) -> list[RepositoryId]: |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|         """ |             report: bool, unsafe: bool) -> None: | ||||||
|         get known architectures |  | ||||||
|  |  | ||||||
|         Args: |  | ||||||
|             args(argparse.Namespace): command line args |  | ||||||
|  |  | ||||||
|         Returns: |  | ||||||
|             tuple[str | None, str]: list of repository names and architectures for which tree is created |  | ||||||
|  |  | ||||||
|         Raises: |  | ||||||
|             MissingArchitectureError: if no architecture set and automatic detection is not allowed or failed |  | ||||||
|         """ |  | ||||||
|         if not cls.ALLOW_AUTO_ARCHITECTURE_RUN and args.architecture is None: |  | ||||||
|             # for some parsers (e.g. config) we need to run with specific architecture |  | ||||||
|             # for those cases architecture must be set explicitly |  | ||||||
|             raise MissingArchitectureError(args.command) |  | ||||||
|         if args.architecture:  # architecture is specified explicitly |  | ||||||
|             repositories = args.repository or [None]  # fallback for legacy mode |  | ||||||
|             return sorted( |  | ||||||
|                 set( |  | ||||||
|                     RepositoryId(architecture, repository) |  | ||||||
|                     for architecture in args.architecture |  | ||||||
|                     for repository in repositories |  | ||||||
|                 ) |  | ||||||
|             ) |  | ||||||
|  |  | ||||||
|         configuration = Configuration() |  | ||||||
|         configuration.load(args.configuration) |  | ||||||
|         # wtf??? |  | ||||||
|         root = configuration.getpath("repository", "root")  # pylint: disable=assignment-from-no-return |  | ||||||
|         name = configuration.get("repository", "name", fallback="")  # will only be used for legacy mode |  | ||||||
|         architectures = RepositoryPaths.known_architectures(root, name) |  | ||||||
|  |  | ||||||
|         if not architectures:  # well we did not find anything |  | ||||||
|             raise MissingArchitectureError(args.command) |  | ||||||
|         return sorted(architectures) |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |  | ||||||
|             report: bool) -> None: |  | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|  |  | ||||||
|         Raises: |         Raises: | ||||||
|             NotImplementedError: not implemented method |             NotImplementedError: not implemented method | ||||||
| @ -176,4 +165,4 @@ class Handler: | |||||||
|             ExitCode: if result is empty and check is enabled |             ExitCode: if result is empty and check is enabled | ||||||
|         """ |         """ | ||||||
|         if enabled and predicate: |         if enabled and predicate: | ||||||
|             raise ExitCode |             raise ExitCode() | ||||||
|  | |||||||
| @ -21,7 +21,6 @@ import argparse | |||||||
|  |  | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Help(Handler): | class Help(Handler): | ||||||
| @ -32,16 +31,17 @@ class Help(Handler): | |||||||
|     ALLOW_AUTO_ARCHITECTURE_RUN = False  # it should be called only as "no-architecture" |     ALLOW_AUTO_ARCHITECTURE_RUN = False  # it should be called only as "no-architecture" | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         parser: argparse.ArgumentParser = args.parser() |         parser: argparse.ArgumentParser = args.parser() | ||||||
|         if args.command is None: |         if args.command is None: | ||||||
|  | |||||||
| @ -22,7 +22,6 @@ import argparse | |||||||
| from ahriman.application.application import Application | from ahriman.application.application import Application | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class KeyImport(Handler): | class KeyImport(Handler): | ||||||
| @ -33,16 +32,17 @@ class KeyImport(Handler): | |||||||
|     ALLOW_AUTO_ARCHITECTURE_RUN = False  # it should be called only as "no-architecture" |     ALLOW_AUTO_ARCHITECTURE_RUN = False  # it should be called only as "no-architecture" | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         application = Application(repository_id, configuration, report=report) |         application = Application(architecture, configuration, report=report, unsafe=unsafe) | ||||||
|         application.repository.sign.key_import(args.key_server, args.key) |         application.repository.sign.key_import(args.key_server, args.key) | ||||||
|  | |||||||
| @ -30,7 +30,6 @@ from ahriman.core.formatters import PatchPrinter | |||||||
| from ahriman.models.action import Action | from ahriman.models.action import Action | ||||||
| from ahriman.models.package import Package | from ahriman.models.package import Package | ||||||
| from ahriman.models.pkgbuild_patch import PkgbuildPatch | from ahriman.models.pkgbuild_patch import PkgbuildPatch | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Patch(Handler): | class Patch(Handler): | ||||||
| @ -39,30 +38,30 @@ class Patch(Handler): | |||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         application = Application(repository_id, configuration, report=report) |         application = Application(architecture, configuration, report=report, unsafe=unsafe) | ||||||
|         application.on_start() |         application.on_start() | ||||||
|  |  | ||||||
|         match args.action: |         if args.action == Action.Update and args.variable is not None: | ||||||
|             case Action.Update if args.variable is not None: |  | ||||||
|             patch = Patch.patch_create_from_function(args.variable, args.patch) |             patch = Patch.patch_create_from_function(args.variable, args.patch) | ||||||
|             Patch.patch_set_create(application, args.package, patch) |             Patch.patch_set_create(application, args.package, patch) | ||||||
|             case Action.Update: |         elif args.action == Action.Update and args.variable is None: | ||||||
|                 package_base, patch = Patch.patch_create_from_diff(args.package, repository_id.architecture, args.track) |             package_base, patch = Patch.patch_create_from_diff(args.package, architecture, args.track) | ||||||
|             Patch.patch_set_create(application, package_base, patch) |             Patch.patch_set_create(application, package_base, patch) | ||||||
|             case Action.List: |         elif args.action == Action.List: | ||||||
|             Patch.patch_set_list(application, args.package, args.variable, args.exit_code) |             Patch.patch_set_list(application, args.package, args.variable, args.exit_code) | ||||||
|             case Action.Remove: |         elif args.action == Action.Remove: | ||||||
|             Patch.patch_set_remove(application, args.package, args.variable) |             Patch.patch_set_remove(application, args.package, args.variable) | ||||||
|  |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
| @ -79,7 +78,7 @@ class Patch(Handler): | |||||||
|             tuple[str, PkgbuildPatch]: package base and created PKGBUILD patch based on the diff from master HEAD |             tuple[str, PkgbuildPatch]: package base and created PKGBUILD patch based on the diff from master HEAD | ||||||
|                 to current files |                 to current files | ||||||
|         """ |         """ | ||||||
|         package = Package.from_build(sources_dir, architecture, None) |         package = Package.from_build(sources_dir, architecture) | ||||||
|         patch = Sources.patch_create(sources_dir, *track) |         patch = Sources.patch_create(sources_dir, *track) | ||||||
|         return package.base, PkgbuildPatch(None, patch) |         return package.base, PkgbuildPatch(None, patch) | ||||||
|  |  | ||||||
|  | |||||||
| @ -22,9 +22,8 @@ import argparse | |||||||
| from ahriman.application.application import Application | from ahriman.application.application import Application | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.models.build_status import BuildStatusEnum | from ahriman.core.formatters import UpdatePrinter | ||||||
| from ahriman.models.package import Package | from ahriman.models.package import Package | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Rebuild(Handler): | class Rebuild(Handler): | ||||||
| @ -33,50 +32,45 @@ class Rebuild(Handler): | |||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         application = Application(repository_id, configuration, report=report) |         application = Application(architecture, configuration, report=report, unsafe=unsafe) | ||||||
|         application.on_start() |         application.on_start() | ||||||
|  |  | ||||||
|         packages = Rebuild.extract_packages(application, args.status, from_database=args.from_database) |         packages = Rebuild.extract_packages(application, from_database=args.from_database) | ||||||
|         updates = application.repository.packages_depend_on(packages, args.depends_on or None) |         updates = application.repository.packages_depend_on(packages, args.depends_on or None) | ||||||
|  |  | ||||||
|         Rebuild.check_if_empty(args.exit_code, not updates) |         Rebuild.check_if_empty(args.exit_code, not updates) | ||||||
|         if args.dry_run: |         if args.dry_run: | ||||||
|             application.print_updates(updates, log_fn=print) |             for package in updates: | ||||||
|  |                 UpdatePrinter(package, package.version).print(verbose=True) | ||||||
|             return |             return | ||||||
|  |  | ||||||
|         result = application.update(updates, args.username, bump_pkgrel=args.increment) |         result = application.update(updates) | ||||||
|         Rebuild.check_if_empty(args.exit_code, result.is_empty) |         Rebuild.check_if_empty(args.exit_code, result.is_empty) | ||||||
|  |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def extract_packages(application: Application, status: BuildStatusEnum | None, *, |     def extract_packages(application: Application, *, from_database: bool) -> list[Package]: | ||||||
|                          from_database: bool) -> list[Package]: |  | ||||||
|         """ |         """ | ||||||
|         extract packages from database file |         extract packages from database file | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             application(Application): application instance |             application(Application): application instance | ||||||
|             status(BuildStatusEnum | None): optional filter by package status |  | ||||||
|             from_database(bool): extract packages from database instead of repository filesystem |             from_database(bool): extract packages from database instead of repository filesystem | ||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             list[Package]: list of packages which were stored in database |             list[Package]: list of packages which were stored in database | ||||||
|         """ |         """ | ||||||
|         if from_database: |         if from_database: | ||||||
|             return [ |             return [package for (package, _) in application.database.packages_get()] | ||||||
|                 package |  | ||||||
|                 for (package, last_status) in application.database.packages_get() |  | ||||||
|                 if status is None or last_status.status == status |  | ||||||
|             ] |  | ||||||
|  |  | ||||||
|         return application.repository.packages() |         return application.repository.packages() | ||||||
|  | |||||||
| @ -22,7 +22,6 @@ import argparse | |||||||
| from ahriman.application.application import Application | from ahriman.application.application import Application | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Remove(Handler): | class Remove(Handler): | ||||||
| @ -31,17 +30,18 @@ class Remove(Handler): | |||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         application = Application(repository_id, configuration, report=report) |         application = Application(architecture, configuration, report=report, unsafe=unsafe) | ||||||
|         application.on_start() |         application.on_start() | ||||||
|         application.remove(args.package) |         application.remove(args.package) | ||||||
|  | |||||||
| @ -23,7 +23,6 @@ from ahriman.application.application import Application | |||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.core.formatters import StringPrinter | from ahriman.core.formatters import StringPrinter | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class RemoveUnknown(Handler): | class RemoveUnknown(Handler): | ||||||
| @ -32,18 +31,19 @@ class RemoveUnknown(Handler): | |||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         application = Application(repository_id, configuration, report=report) |         application = Application(architecture, configuration, report=report, unsafe=unsafe) | ||||||
|         application.on_start() |         application.on_start() | ||||||
|         unknown_packages = application.unknown() |         unknown_packages = application.unknown() | ||||||
|  |  | ||||||
|  | |||||||
| @ -23,7 +23,6 @@ from tarfile import TarFile | |||||||
|  |  | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Restore(Handler): | class Restore(Handler): | ||||||
| @ -34,16 +33,17 @@ class Restore(Handler): | |||||||
|     ALLOW_AUTO_ARCHITECTURE_RUN = False  # it should be called only as "no-architecture" |     ALLOW_AUTO_ARCHITECTURE_RUN = False  # it should be called only as "no-architecture" | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         with TarFile(args.path) as archive: |         with TarFile(args.path) as archive: | ||||||
|             archive.extractall(path=args.output) |             archive.extractall(path=args.output) | ||||||
|  | |||||||
| @ -29,7 +29,6 @@ from ahriman.core.configuration import Configuration | |||||||
| from ahriman.core.exceptions import OptionError | from ahriman.core.exceptions import OptionError | ||||||
| from ahriman.core.formatters import AurPrinter | from ahriman.core.formatters import AurPrinter | ||||||
| from ahriman.models.aur_package import AURPackage | from ahriman.models.aur_package import AURPackage | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Search(Handler): | class Search(Handler): | ||||||
| @ -48,18 +47,19 @@ class Search(Handler): | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         application = Application(repository_id, configuration, report=report) |         application = Application(architecture, configuration, report=report, unsafe=unsafe) | ||||||
|  |  | ||||||
|         official_packages_list = Official.multisearch(*args.search, pacman=application.repository.pacman) |         official_packages_list = Official.multisearch(*args.search, pacman=application.repository.pacman) | ||||||
|         aur_packages_list = AUR.multisearch(*args.search, pacman=application.repository.pacman) |         aur_packages_list = AUR.multisearch(*args.search, pacman=application.repository.pacman) | ||||||
| @ -83,7 +83,7 @@ class Search(Handler): | |||||||
|             list[AURPackage]: sorted list for packages |             list[AURPackage]: sorted list for packages | ||||||
|  |  | ||||||
|         Raises: |         Raises: | ||||||
|             OptionError: if search fields is not in list of allowed ones |             InvalidOption: if search fields is not in list of allowed ones | ||||||
|         """ |         """ | ||||||
|         if sort_by not in Search.SORT_FIELDS: |         if sort_by not in Search.SORT_FIELDS: | ||||||
|             raise OptionError(sort_by) |             raise OptionError(sort_by) | ||||||
|  | |||||||
| @ -19,13 +19,12 @@ | |||||||
| # | # | ||||||
| import argparse | import argparse | ||||||
|  |  | ||||||
| from ahriman import __version__ | from ahriman import version | ||||||
| from ahriman.application.application import Application | from ahriman.application.application import Application | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.core.formatters import UpdatePrinter | from ahriman.core.formatters import UpdatePrinter | ||||||
| from ahriman.models.package import Package | from ahriman.models.package import Package | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class ServiceUpdates(Handler): | class ServiceUpdates(Handler): | ||||||
| @ -36,22 +35,23 @@ class ServiceUpdates(Handler): | |||||||
|     ALLOW_AUTO_ARCHITECTURE_RUN = False  # it should be called only as "no-architecture" |     ALLOW_AUTO_ARCHITECTURE_RUN = False  # it should be called only as "no-architecture" | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         application = Application(repository_id, configuration, report=report) |         application = Application(architecture, configuration, report=report, unsafe=unsafe) | ||||||
|  |  | ||||||
|         remote = Package.from_aur("ahriman", application.repository.pacman, None) |         remote = Package.from_aur("ahriman", application.repository.pacman) | ||||||
|         _, release = remote.version.rsplit("-", 1)  # we don't store pkgrel locally, so we just append it |         release = remote.version.rsplit("-", 1)[-1]  # we don't store pkgrel locally, so we just append it | ||||||
|         local_version = f"{__version__}-{release}" |         local_version = f"{version.__version__}-{release}" | ||||||
|  |  | ||||||
|         # technically we would like to compare versions, but it is fine to raise an exception in case if locally |         # technically we would like to compare versions, but it is fine to raise an exception in case if locally | ||||||
|         # installed package is newer than in AUR |         # installed package is newer than in AUR | ||||||
|  | |||||||
| @ -25,9 +25,7 @@ from pwd import getpwuid | |||||||
| from ahriman.application.application import Application | from ahriman.application.application import Application | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
| from ahriman.models.repository_paths import RepositoryPaths | from ahriman.models.repository_paths import RepositoryPaths | ||||||
| from ahriman.models.user import User |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Setup(Handler): | class Setup(Handler): | ||||||
| @ -35,121 +33,99 @@ class Setup(Handler): | |||||||
|     setup handler |     setup handler | ||||||
|  |  | ||||||
|     Attributes: |     Attributes: | ||||||
|         ARCHBUILD_COMMAND_PATH(Path): (class attribute) default devtools command |  | ||||||
|         MIRRORLIST_PATH(Path): (class attribute) path to pacman default mirrorlist (used by multilib repository) |         MIRRORLIST_PATH(Path): (class attribute) path to pacman default mirrorlist (used by multilib repository) | ||||||
|         SUDOERS_DIR_PATH(Path): (class attribute) path to sudoers.d includes directory |         SUDOERS_DIR_PATH(Path): (class attribute) path to sudoers.d includes directory | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     ALLOW_AUTO_ARCHITECTURE_RUN = False |     ALLOW_AUTO_ARCHITECTURE_RUN = False | ||||||
|  |  | ||||||
|     ARCHBUILD_COMMAND_PATH = Path("/usr") / "bin" / "archbuild" |     MIRRORLIST_PATH = Path("/etc/pacman.d/mirrorlist") | ||||||
|     MIRRORLIST_PATH = Path("/etc") / "pacman.d" / "mirrorlist" |     SUDOERS_DIR_PATH = Path("/etc/sudoers.d") | ||||||
|     SUDOERS_DIR_PATH = Path("/etc") / "sudoers.d" |  | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         Setup.configuration_create_ahriman(args, repository_id, configuration) |         Setup.configuration_create_ahriman(args, architecture, args.repository, configuration) | ||||||
|         configuration.reload() |         configuration.reload() | ||||||
|  |  | ||||||
|         application = Application(repository_id, configuration, report=report) |         application = Application(architecture, configuration, report=report, unsafe=unsafe) | ||||||
|  |  | ||||||
|         Setup.configuration_create_makepkg(args.packager, args.makeflags_jobs, application.repository.paths) |         Setup.configuration_create_makepkg(args.packager, args.makeflags_jobs, application.repository.paths) | ||||||
|         Setup.executable_create(application.repository.paths, repository_id) |         Setup.configuration_create_devtools(architecture, args.from_configuration, args.mirror, | ||||||
|         repository_server = f"file://{application.repository.paths.repository}" if args.server is None else args.server |                                             args.multilib, args.repository, application.repository.paths) | ||||||
|         Setup.configuration_create_devtools(repository_id, args.from_configuration, args.mirror, args.multilib, |         Setup.configuration_create_sudo(args.build_command) | ||||||
|                                             repository_server) |  | ||||||
|         Setup.configuration_create_sudo(application.repository.paths, repository_id) |  | ||||||
|  |  | ||||||
|         application.repository.repo.init() |         application.repository.repo.init() | ||||||
|         # lazy database sync |         # lazy database sync | ||||||
|         application.repository.pacman.handle  # pylint: disable=pointless-statement |         application.repository.pacman.handle  # pylint: disable=pointless-statement | ||||||
|  |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def build_command(root: Path, repository_id: RepositoryId) -> Path: |     def configuration_create_ahriman(args: argparse.Namespace, architecture: str, repository: str, | ||||||
|         """ |  | ||||||
|         generate build command name |  | ||||||
|  |  | ||||||
|         Args: |  | ||||||
|             root(Path): root directory for the build command (must be root of the repository) |  | ||||||
|             repository_id(RepositoryId): repository unique identifier |  | ||||||
|  |  | ||||||
|         Returns: |  | ||||||
|             Path: valid devtools command name |  | ||||||
|         """ |  | ||||||
|         return root / f"{repository_id.name}-{repository_id.architecture}-build" |  | ||||||
|  |  | ||||||
|     @staticmethod |  | ||||||
|     def configuration_create_ahriman( |  | ||||||
|             args: argparse.Namespace, |  | ||||||
|             repository_id: RepositoryId, |  | ||||||
|                                      root: Configuration) -> None: |                                      root: Configuration) -> None: | ||||||
|         """ |         """ | ||||||
|         create service specific configuration |         create service specific configuration | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|  |             repository(str): repository name | ||||||
|             root(Configuration): root configuration instance |             root(Configuration): root configuration instance | ||||||
|         """ |         """ | ||||||
|         configuration = Configuration() |         configuration = Configuration() | ||||||
|  |  | ||||||
|         section = Configuration.section_name("build", repository_id.name, repository_id.architecture) |         configuration.set_option("repository", "name", repository) | ||||||
|         build_command = Setup.build_command(root.repository_paths.root, repository_id) |  | ||||||
|         configuration.set_option(section, "build_command", str(build_command)) |  | ||||||
|         configuration.set_option("repository", "name", repository_id.name) |  | ||||||
|         if args.build_as_user is not None: |  | ||||||
|             configuration.set_option(section, "makechrootpkg_flags", f"-U {args.build_as_user}") |  | ||||||
|  |  | ||||||
|         section = Configuration.section_name("alpm", repository_id.name, repository_id.architecture) |         section = Configuration.section_name("build", architecture) | ||||||
|  |         configuration.set_option(section, "build_command", str(args.build_command)) | ||||||
|  |  | ||||||
|  |         section = Configuration.section_name("alpm", architecture) | ||||||
|         if args.mirror is not None: |         if args.mirror is not None: | ||||||
|             configuration.set_option(section, "mirror", args.mirror) |             configuration.set_option(section, "mirror", args.mirror) | ||||||
|         if not args.multilib: |         if not args.multilib: | ||||||
|             repositories = filter(lambda r: r != "multilib", root.getlist("alpm", "repositories")) |             repositories = filter(lambda r: r != "multilib", root.getlist("alpm", "repositories")) | ||||||
|             configuration.set_option(section, "repositories", " ".join(repositories)) |             configuration.set_option(section, "repositories", " ".join(repositories)) | ||||||
|  |  | ||||||
|         section = Configuration.section_name("sign", repository_id.name, repository_id.architecture) |         section = Configuration.section_name("sign", architecture) | ||||||
|         if args.sign_key is not None: |         if args.sign_key is not None: | ||||||
|             configuration.set_option(section, "target", " ".join([target.name.lower() for target in args.sign_target])) |             configuration.set_option(section, "target", " ".join([target.name.lower() for target in args.sign_target])) | ||||||
|             configuration.set_option(section, "key", args.sign_key) |             configuration.set_option(section, "key", args.sign_key) | ||||||
|  |  | ||||||
|         section = Configuration.section_name("web", repository_id.name, repository_id.architecture) |         section = Configuration.section_name("web", architecture) | ||||||
|         if args.web_port is not None: |         if args.web_port is not None: | ||||||
|             configuration.set_option(section, "port", str(args.web_port)) |             configuration.set_option(section, "port", str(args.web_port)) | ||||||
|         if args.web_unix_socket is not None: |         if args.web_unix_socket is not None: | ||||||
|             configuration.set_option(section, "unix_socket", str(args.web_unix_socket)) |             configuration.set_option(section, "unix_socket", str(args.web_unix_socket)) | ||||||
|  |  | ||||||
|         if args.generate_salt: |  | ||||||
|             configuration.set_option("auth", "salt", User.generate_password(20)) |  | ||||||
|  |  | ||||||
|         target = root.include / "00-setup-overrides.ini" |         target = root.include / "00-setup-overrides.ini" | ||||||
|         with target.open("w") as ahriman_configuration: |         with target.open("w") as ahriman_configuration: | ||||||
|             configuration.write(ahriman_configuration) |             configuration.write(ahriman_configuration) | ||||||
|  |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def configuration_create_devtools(repository_id: RepositoryId, source: Path, mirror: str | None, |     def configuration_create_devtools(architecture: str, source: Path, mirror: str | None, | ||||||
|                                       multilib: bool, repository_server: str) -> None: |                                       multilib: bool, repository: str, paths: RepositoryPaths) -> None: | ||||||
|         """ |         """ | ||||||
|         create configuration for devtools based on ``source`` configuration |         create configuration for devtools based on ``source`` configuration | ||||||
|  |  | ||||||
|         Notes: |         Note: | ||||||
|             devtools does not allow to specify the pacman configuration, thus we still have to use configuration in /usr |             devtools does not allow to specify the pacman configuration, thus we still have to use configuration in /usr | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             source(Path): path to source configuration file |             source(Path): path to source configuration file | ||||||
|             mirror(str | None): link to package server mirror |             mirror(str | None): link to package server mirror | ||||||
|             multilib(bool): add or do not multilib repository to the configuration |             multilib(bool): add or do not multilib repository to the configuration | ||||||
|             repository_server(str): url of the repository |             repository(str): repository name | ||||||
|  |             paths(RepositoryPaths): repository paths instance | ||||||
|         """ |         """ | ||||||
|         # allow_no_value=True is required because pacman uses boolean configuration in which just keys present |         # allow_no_value=True is required because pacman uses boolean configuration in which just keys present | ||||||
|         # (e.g. NoProgressBar) which will lead to exception |         # (e.g. NoProgressBar) which will lead to exception | ||||||
| @ -163,7 +139,7 @@ class Setup(Handler): | |||||||
|         configuration.read(source) |         configuration.read(source) | ||||||
|  |  | ||||||
|         # set our architecture now |         # set our architecture now | ||||||
|         configuration.set_option("options", "Architecture", repository_id.architecture) |         configuration.set_option("options", "Architecture", architecture) | ||||||
|  |  | ||||||
|         # add multilib |         # add multilib | ||||||
|         if multilib: |         if multilib: | ||||||
| @ -178,10 +154,10 @@ class Setup(Handler): | |||||||
|                 configuration.set_option(section, "Server", mirror) |                 configuration.set_option(section, "Server", mirror) | ||||||
|  |  | ||||||
|         # add repository itself |         # add repository itself | ||||||
|         configuration.set_option(repository_id.name, "SigLevel", "Never")  # we don't care |         configuration.set_option(repository, "SigLevel", "Optional TrustAll")  # we don't care | ||||||
|         configuration.set_option(repository_id.name, "Server", repository_server) |         configuration.set_option(repository, "Server", f"file://{paths.repository}") | ||||||
|  |  | ||||||
|         target = source.parent / f"{repository_id.name}-{repository_id.architecture}.conf" |         target = source.parent / f"{repository}.conf" | ||||||
|         with target.open("w") as devtools_configuration: |         with target.open("w") as devtools_configuration: | ||||||
|             configuration.write(devtools_configuration) |             configuration.write(devtools_configuration) | ||||||
|  |  | ||||||
| @ -205,29 +181,13 @@ class Setup(Handler): | |||||||
|         (home_dir / ".makepkg.conf").write_text(content, encoding="utf8") |         (home_dir / ".makepkg.conf").write_text(content, encoding="utf8") | ||||||
|  |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def configuration_create_sudo(paths: RepositoryPaths, repository_id: RepositoryId) -> None: |     def configuration_create_sudo(build_command: Path) -> None: | ||||||
|         """ |         """ | ||||||
|         create configuration to run build command with sudo without password |         create configuration to run build command with sudo without password | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             paths(RepositoryPaths): repository paths instance |             build_command(Path): path to build command | ||||||
|             repository_id(RepositoryId): repository unique identifier |  | ||||||
|         """ |         """ | ||||||
|         command = Setup.build_command(paths.root, repository_id) |         sudoers_file = Setup.SUDOERS_DIR_PATH / f"ahriman-{build_command.name}" | ||||||
|         sudoers_file = Setup.build_command(Setup.SUDOERS_DIR_PATH, repository_id) |         sudoers_file.write_text(f"ahriman ALL=(ALL) NOPASSWD: {build_command} build *\n", encoding="utf8") | ||||||
|         sudoers_file.write_text(f"ahriman ALL=(ALL) NOPASSWD:SETENV: {command} *\n", encoding="utf8") |  | ||||||
|         sudoers_file.chmod(0o400)  # security! |         sudoers_file.chmod(0o400)  # security! | ||||||
|  |  | ||||||
|     @staticmethod |  | ||||||
|     def executable_create(paths: RepositoryPaths, repository_id: RepositoryId) -> None: |  | ||||||
|         """ |  | ||||||
|         create executable for the service |  | ||||||
|  |  | ||||||
|         Args: |  | ||||||
|             paths(RepositoryPaths): repository paths instance |  | ||||||
|             repository_id(RepositoryId): repository unique identifier |  | ||||||
|         """ |  | ||||||
|         command = Setup.build_command(paths.root, repository_id) |  | ||||||
|         command.unlink(missing_ok=True) |  | ||||||
|         command.symlink_to(Setup.ARCHBUILD_COMMAND_PATH) |  | ||||||
|         paths.chown(command)  # we would like to keep owner inside ahriman's home |  | ||||||
|  | |||||||
| @ -23,10 +23,10 @@ import sys | |||||||
|  |  | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
|  |  | ||||||
|  | from ahriman.application.application import Application | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.core.formatters import StringPrinter | from ahriman.core.formatters import StringPrinter | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Shell(Handler): | class Shell(Handler): | ||||||
| @ -37,29 +37,26 @@ class Shell(Handler): | |||||||
|     ALLOW_MULTI_ARCHITECTURE_RUN = False |     ALLOW_MULTI_ARCHITECTURE_RUN = False | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|  |         # pylint: disable=possibly-unused-variable | ||||||
|  |         application = Application(architecture, configuration, report=report, unsafe=unsafe) | ||||||
|         if args.verbose: |         if args.verbose: | ||||||
|             # licensed by https://creativecommons.org/licenses/by-sa/3.0 |             # licensed by https://creativecommons.org/licenses/by-sa/3.0 | ||||||
|             path = Path(sys.prefix) / "share" / "ahriman" / "templates" / "shell" |             path = Path(sys.prefix) / "share" / "ahriman" / "templates" / "shell" | ||||||
|             StringPrinter(path.read_text(encoding="utf8")).print(verbose=False) |             StringPrinter(path.read_text(encoding="utf8")).print(verbose=False) | ||||||
|  |         # we only want to pass application instance inside | ||||||
|         local_variables = { |  | ||||||
|             "architecture": repository_id.architecture, |  | ||||||
|             "configuration": configuration, |  | ||||||
|             "repository_id": repository_id, |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         if args.code is None: |         if args.code is None: | ||||||
|             code.interact(local=local_variables) |             code.interact(local={"application": application}) | ||||||
|         else: |         else: | ||||||
|             code.InteractiveConsole(locals=local_variables).runcode(args.code) |             code.InteractiveConsole(locals={"application": application}).runcode(args.code) | ||||||
|  | |||||||
| @ -22,7 +22,6 @@ import argparse | |||||||
| from ahriman.application.application import Application | from ahriman.application.application import Application | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Sign(Handler): | class Sign(Handler): | ||||||
| @ -31,15 +30,16 @@ class Sign(Handler): | |||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         Application(repository_id, configuration, report=report).sign(args.package) |         Application(architecture, configuration, report=report, unsafe=unsafe).sign(args.package) | ||||||
|  | |||||||
| @ -27,7 +27,6 @@ from ahriman.core.configuration import Configuration | |||||||
| from ahriman.core.formatters import PackagePrinter, StatusPrinter | from ahriman.core.formatters import PackagePrinter, StatusPrinter | ||||||
| from ahriman.models.build_status import BuildStatus | from ahriman.models.build_status import BuildStatus | ||||||
| from ahriman.models.package import Package | from ahriman.models.package import Package | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Status(Handler): | class Status(Handler): | ||||||
| @ -38,28 +37,29 @@ class Status(Handler): | |||||||
|     ALLOW_AUTO_ARCHITECTURE_RUN = False |     ALLOW_AUTO_ARCHITECTURE_RUN = False | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         # we are using reporter here |         # we are using reporter here | ||||||
|         client = Application(repository_id, configuration, report=True).repository.reporter |         client = Application(architecture, configuration, report=True, unsafe=unsafe).repository.reporter | ||||||
|         if args.ahriman: |         if args.ahriman: | ||||||
|             service_status = client.status_get() |             service_status = client.get_internal() | ||||||
|             StatusPrinter(service_status.status).print(verbose=args.info) |             StatusPrinter(service_status.status).print(verbose=args.info) | ||||||
|         if args.package: |         if args.package: | ||||||
|             packages: list[tuple[Package, BuildStatus]] = sum( |             packages: list[tuple[Package, BuildStatus]] = sum( | ||||||
|                 (client.package_get(base) for base in args.package), |                 (client.get(base) for base in args.package), | ||||||
|                 start=[]) |                 start=[]) | ||||||
|         else: |         else: | ||||||
|             packages = client.package_get(None) |             packages = client.get(None) | ||||||
|  |  | ||||||
|         Status.check_if_empty(args.exit_code, not packages) |         Status.check_if_empty(args.exit_code, not packages) | ||||||
|  |  | ||||||
|  | |||||||
| @ -23,7 +23,6 @@ from ahriman.application.application import Application | |||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.models.action import Action | from ahriman.models.action import Action | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class StatusUpdate(Handler): | class StatusUpdate(Handler): | ||||||
| @ -34,28 +33,28 @@ class StatusUpdate(Handler): | |||||||
|     ALLOW_AUTO_ARCHITECTURE_RUN = False |     ALLOW_AUTO_ARCHITECTURE_RUN = False | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         # we are using reporter here |         # we are using reporter here | ||||||
|         client = Application(repository_id, configuration, report=True).repository.reporter |         client = Application(architecture, configuration, report=True, unsafe=unsafe).repository.reporter | ||||||
|  |  | ||||||
|         match args.action: |         if args.action == Action.Update and args.package: | ||||||
|             case Action.Update if args.package: |  | ||||||
|             # update packages statuses |             # update packages statuses | ||||||
|             for package in args.package: |             for package in args.package: | ||||||
|                     client.package_update(package, args.status) |                 client.update(package, args.status) | ||||||
|             case Action.Update: |         elif args.action == Action.Update: | ||||||
|             # update service status |             # update service status | ||||||
|                 client.status_update(args.status) |             client.update_self(args.status) | ||||||
|             case Action.Remove: |         elif args.action == Action.Remove: | ||||||
|             for package in args.package: |             for package in args.package: | ||||||
|                     client.package_remove(package) |                 client.remove(package) | ||||||
|  | |||||||
| @ -22,9 +22,8 @@ import argparse | |||||||
| from ahriman.application.application import Application | from ahriman.application.application import Application | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.core.formatters import StringPrinter, TreePrinter | from ahriman.core.formatters import TreePrinter | ||||||
| from ahriman.core.tree import Tree | from ahriman.core.tree import Tree | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Structure(Handler): | class Structure(Handler): | ||||||
| @ -35,26 +34,21 @@ class Structure(Handler): | |||||||
|     ALLOW_AUTO_ARCHITECTURE_RUN = False |     ALLOW_AUTO_ARCHITECTURE_RUN = False | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         application = Application(repository_id, configuration, report=report) |         application = Application(architecture, configuration, report=report, unsafe=unsafe) | ||||||
|         partitions = Tree.partition(application.repository.packages(), count=args.partitions) |         packages = application.repository.packages() | ||||||
|  |  | ||||||
|         for partition_id, partition in enumerate(partitions): |         tree = Tree.resolve(packages) | ||||||
|             StringPrinter(f"partition #{partition_id}").print(verbose=False) |  | ||||||
|  |  | ||||||
|             tree = Tree.resolve(partition) |  | ||||||
|         for num, level in enumerate(tree): |         for num, level in enumerate(tree): | ||||||
|             TreePrinter(num, level).print(verbose=True, separator=" ") |             TreePrinter(num, level).print(verbose=True, separator=" ") | ||||||
|  |  | ||||||
|             # empty line |  | ||||||
|             StringPrinter("").print(verbose=False) |  | ||||||
|  | |||||||
| @ -22,7 +22,6 @@ import argparse | |||||||
| from ahriman.application.application import Application | from ahriman.application.application import Application | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
| from ahriman.models.result import Result | from ahriman.models.result import Result | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -32,20 +31,21 @@ class Triggers(Handler): | |||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         application = Application(repository_id, configuration, report=report) |         application = Application(architecture, configuration, report=report, unsafe=unsafe) | ||||||
|         if args.trigger: |         if args.trigger: | ||||||
|             loader = application.repository.triggers |             loader = application.repository.triggers | ||||||
|             loader.triggers = [loader.load_trigger(trigger, repository_id, configuration) for trigger in args.trigger] |             loader.triggers = [loader.load_trigger(trigger, architecture, configuration) for trigger in args.trigger] | ||||||
|         application.on_start() |         application.on_start() | ||||||
|         application.on_result(Result()) |         application.on_result(Result()) | ||||||
|  | |||||||
| @ -18,11 +18,11 @@ | |||||||
| # along with this program. If not, see <http://www.gnu.org/licenses/>. | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
| # | # | ||||||
| import argparse | import argparse | ||||||
|  | import shlex | ||||||
|  |  | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.core.formatters import StringPrinter | from ahriman.core.formatters import StringPrinter | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class UnsafeCommands(Handler): | class UnsafeCommands(Handler): | ||||||
| @ -33,27 +33,28 @@ class UnsafeCommands(Handler): | |||||||
|     ALLOW_AUTO_ARCHITECTURE_RUN = False  # it should be called only as "no-architecture" |     ALLOW_AUTO_ARCHITECTURE_RUN = False  # it should be called only as "no-architecture" | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         parser = args.parser() |         parser = args.parser() | ||||||
|         unsafe_commands = UnsafeCommands.get_unsafe_commands(parser) |         unsafe_commands = UnsafeCommands.get_unsafe_commands(parser) | ||||||
|         if args.command: |         if args.command is None: | ||||||
|             UnsafeCommands.check_unsafe(args.command, unsafe_commands, parser) |  | ||||||
|         else: |  | ||||||
|             for command in unsafe_commands: |             for command in unsafe_commands: | ||||||
|                 StringPrinter(command).print(verbose=True) |                 StringPrinter(command).print(verbose=True) | ||||||
|  |         else: | ||||||
|  |             UnsafeCommands.check_unsafe(args.command, unsafe_commands, parser) | ||||||
|  |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def check_unsafe(command: list[str], unsafe_commands: list[str], parser: argparse.ArgumentParser) -> None: |     def check_unsafe(command: str, unsafe_commands: list[str], parser: argparse.ArgumentParser) -> None: | ||||||
|         """ |         """ | ||||||
|         check if command is unsafe |         check if command is unsafe | ||||||
|  |  | ||||||
| @ -62,7 +63,7 @@ class UnsafeCommands(Handler): | |||||||
|             unsafe_commands(list[str]): list of unsafe commands |             unsafe_commands(list[str]): list of unsafe commands | ||||||
|             parser(argparse.ArgumentParser): generated argument parser |             parser(argparse.ArgumentParser): generated argument parser | ||||||
|         """ |         """ | ||||||
|         args = parser.parse_args(command) |         args = parser.parse_args(shlex.split(command)) | ||||||
|         UnsafeCommands.check_if_empty(True, args.command in unsafe_commands) |         UnsafeCommands.check_if_empty(True, args.command in unsafe_commands) | ||||||
|  |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
| @ -78,6 +79,5 @@ class UnsafeCommands(Handler): | |||||||
|         """ |         """ | ||||||
|         # should never fail |         # should never fail | ||||||
|         # pylint: disable=protected-access |         # pylint: disable=protected-access | ||||||
|         subparser = next((action for action in parser._actions if isinstance(action, argparse._SubParsersAction)), None) |         subparser = next(action for action in parser._actions if isinstance(action, argparse._SubParsersAction)) | ||||||
|         actions = subparser.choices if subparser is not None else {} |         return sorted(action_name for action_name, action in subparser.choices.items() if action.get_default("unsafe")) | ||||||
|         return sorted(action_name for action_name, action in actions.items() if action.get_default("unsafe")) |  | ||||||
|  | |||||||
| @ -24,8 +24,6 @@ from collections.abc import Callable | |||||||
| from ahriman.application.application import Application | from ahriman.application.application import Application | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.models.packagers import Packagers |  | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Update(Handler): | class Update(Handler): | ||||||
| @ -34,29 +32,29 @@ class Update(Handler): | |||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         application = Application(repository_id, configuration, report=report, refresh_pacman_database=args.refresh) |         application = Application(architecture, configuration, report=report, unsafe=unsafe, | ||||||
|  |                                   refresh_pacman_database=args.refresh) | ||||||
|         application.on_start() |         application.on_start() | ||||||
|         packages = application.updates(args.package, aur=args.aur, local=args.local, manual=args.manual, vcs=args.vcs) |         packages = application.updates(args.package, aur=args.aur, local=args.local, manual=args.manual, vcs=args.vcs, | ||||||
|  |                                        log_fn=Update.log_fn(application, args.dry_run)) | ||||||
|         Update.check_if_empty(args.exit_code, not packages) |         Update.check_if_empty(args.exit_code, not packages) | ||||||
|         if args.dry_run: |         if args.dry_run: | ||||||
|             return |             return | ||||||
|  |  | ||||||
|         packages = application.with_dependencies(packages, process_dependencies=args.dependencies) |         packages = application.with_dependencies(packages, process_dependencies=args.dependencies) | ||||||
|         packagers = Packagers(args.username, {package.base: package.packager for package in packages}) |         result = application.update(packages) | ||||||
|  |  | ||||||
|         application.print_updates(packages, log_fn=application.logger.info) |  | ||||||
|         result = application.update(packages, packagers, bump_pkgrel=args.increment) |  | ||||||
|         Update.check_if_empty(args.exit_code, result.is_empty) |         Update.check_if_empty(args.exit_code, result.is_empty) | ||||||
|  |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
|  | |||||||
| @ -20,13 +20,14 @@ | |||||||
| import argparse | import argparse | ||||||
| import getpass | import getpass | ||||||
|  |  | ||||||
|  | from pathlib import Path | ||||||
|  |  | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.core.database import SQLite | from ahriman.core.database import SQLite | ||||||
| from ahriman.core.exceptions import PasswordError | from ahriman.core.exceptions import PasswordError | ||||||
| from ahriman.core.formatters import UserPrinter | from ahriman.core.formatters import UserPrinter | ||||||
| from ahriman.models.action import Action | from ahriman.models.action import Action | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
| from ahriman.models.user import User | from ahriman.models.user import User | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -38,33 +39,101 @@ class Users(Handler): | |||||||
|     ALLOW_AUTO_ARCHITECTURE_RUN = False  # it should be called only as "no-architecture" |     ALLOW_AUTO_ARCHITECTURE_RUN = False  # it should be called only as "no-architecture" | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         database = SQLite.load(configuration) |         database = SQLite.load(configuration) | ||||||
|  |  | ||||||
|         match args.action: |         if args.action == Action.Update: | ||||||
|             case Action.Update: |             old_salt, salt = Users.get_salt(configuration) | ||||||
|             user = Users.user_create(args) |             user = Users.user_create(args) | ||||||
|                 # if password is left blank we are not going to require salt to be set |  | ||||||
|                 salt = configuration.get("auth", "salt", fallback="") if user.password else "" |             if old_salt is None: | ||||||
|  |                 auth_configuration = Users.configuration_get(configuration.include) | ||||||
|  |                 Users.configuration_create(auth_configuration, salt, args.secure) | ||||||
|  |  | ||||||
|             database.user_update(user.hash_password(salt)) |             database.user_update(user.hash_password(salt)) | ||||||
|             case Action.List: |         elif args.action == Action.List: | ||||||
|             users = database.user_list(args.username, args.role) |             users = database.user_list(args.username, args.role) | ||||||
|             Users.check_if_empty(args.exit_code, not users) |             Users.check_if_empty(args.exit_code, not users) | ||||||
|             for user in users: |             for user in users: | ||||||
|                 UserPrinter(user).print(verbose=True) |                 UserPrinter(user).print(verbose=True) | ||||||
|             case Action.Remove: |         elif args.action == Action.Remove: | ||||||
|             database.user_remove(args.username) |             database.user_remove(args.username) | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def configuration_create(configuration: Configuration, salt: str, secure: bool) -> None: | ||||||
|  |         """ | ||||||
|  |         enable configuration if it has been disabled | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             configuration(Configuration): configuration instance | ||||||
|  |             salt(str): password hash salt | ||||||
|  |             secure(bool): if true then set file permissions to 0o600 | ||||||
|  |         """ | ||||||
|  |         configuration.set_option("auth", "salt", salt) | ||||||
|  |         Users.configuration_write(configuration, secure) | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def configuration_get(include_path: Path) -> Configuration: | ||||||
|  |         """ | ||||||
|  |         create configuration instance | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             include_path(Path): path to directory with configuration includes | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             Configuration: configuration instance. In case if there are local settings they will be loaded | ||||||
|  |         """ | ||||||
|  |         target = include_path / "00-auth.ini" | ||||||
|  |         configuration = Configuration() | ||||||
|  |         configuration.load(target) | ||||||
|  |  | ||||||
|  |         configuration.architecture = ""  # not user anyway | ||||||
|  |  | ||||||
|  |         return configuration | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def configuration_write(configuration: Configuration, secure: bool) -> None: | ||||||
|  |         """ | ||||||
|  |         write configuration file | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             configuration(Configuration): configuration instance | ||||||
|  |             secure(bool): if true then set file permissions to 0o600 | ||||||
|  |         """ | ||||||
|  |         path, _ = configuration.check_loaded() | ||||||
|  |         with path.open("w") as ahriman_configuration: | ||||||
|  |             configuration.write(ahriman_configuration) | ||||||
|  |         if secure: | ||||||
|  |             path.chmod(0o600) | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def get_salt(configuration: Configuration, salt_length: int = 20) -> tuple[str | None, str]: | ||||||
|  |         """ | ||||||
|  |         get salt from configuration or create new string | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             configuration(Configuration): configuration instance | ||||||
|  |             salt_length(int, optional): salt length (Default value = 20) | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             tuple[str | None, str]: tuple containing salt from configuration if any and actual salt which must be | ||||||
|  |                 used for password hash | ||||||
|  |         """ | ||||||
|  |         if salt := configuration.get("auth", "salt", fallback=None): | ||||||
|  |             return salt, salt | ||||||
|  |         return None, User.generate_password(salt_length) | ||||||
|  |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def user_create(args: argparse.Namespace) -> User: |     def user_create(args: argparse.Namespace) -> User: | ||||||
|         """ |         """ | ||||||
| @ -75,9 +144,6 @@ class Users(Handler): | |||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             User: built user descriptor |             User: built user descriptor | ||||||
|  |  | ||||||
|         Raises: |  | ||||||
|             PasswordError: password input is invalid |  | ||||||
|         """ |         """ | ||||||
|         def read_password() -> str: |         def read_password() -> str: | ||||||
|             first_password = getpass.getpass() |             first_password = getpass.getpass() | ||||||
| @ -90,5 +156,4 @@ class Users(Handler): | |||||||
|         if password is None: |         if password is None: | ||||||
|             password = read_password() |             password = read_password() | ||||||
|  |  | ||||||
|         return User(username=args.username, password=password, access=args.role, |         return User(username=args.username, password=password, access=args.role) | ||||||
|                     packager_id=args.packager, key=args.key) |  | ||||||
|  | |||||||
| @ -29,7 +29,6 @@ from ahriman.core.configuration.validator import Validator | |||||||
| from ahriman.core.exceptions import ExtensionError | from ahriman.core.exceptions import ExtensionError | ||||||
| from ahriman.core.formatters import ValidationPrinter | from ahriman.core.formatters import ValidationPrinter | ||||||
| from ahriman.core.triggers import TriggerLoader | from ahriman.core.triggers import TriggerLoader | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Validate(Handler): | class Validate(Handler): | ||||||
| @ -40,18 +39,19 @@ class Validate(Handler): | |||||||
|     ALLOW_AUTO_ARCHITECTURE_RUN = False |     ALLOW_AUTO_ARCHITECTURE_RUN = False | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         schema = Validate.schema(repository_id, configuration) |         schema = Validate.schema(architecture, configuration) | ||||||
|         validator = Validator(configuration=configuration, schema=schema) |         validator = Validator(configuration=configuration, schema=schema) | ||||||
|  |  | ||||||
|         if validator.validate(configuration.dump()): |         if validator.validate(configuration.dump()): | ||||||
| @ -63,12 +63,12 @@ class Validate(Handler): | |||||||
|         Validate.check_if_empty(args.exit_code, True) |         Validate.check_if_empty(args.exit_code, True) | ||||||
|  |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def schema(repository_id: RepositoryId, configuration: Configuration) -> ConfigurationSchema: |     def schema(architecture: str, configuration: Configuration) -> ConfigurationSchema: | ||||||
|         """ |         """ | ||||||
|         get schema with triggers |         get schema with triggers | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
| @ -78,21 +78,19 @@ class Validate(Handler): | |||||||
|  |  | ||||||
|         # create trigger loader instance |         # create trigger loader instance | ||||||
|         loader = TriggerLoader() |         loader = TriggerLoader() | ||||||
|         triggers = loader.selected_triggers(configuration) + loader.known_triggers(configuration) |         for trigger in loader.selected_triggers(configuration): | ||||||
|  |  | ||||||
|         for trigger in set(triggers): |  | ||||||
|             try: |             try: | ||||||
|                 trigger_class = loader.load_trigger_class(trigger) |                 trigger_class = loader.load_trigger_class(trigger) | ||||||
|             except ExtensionError: |             except ExtensionError: | ||||||
|                 continue |                 continue | ||||||
|  |  | ||||||
|             # default settings if any |             # default settings if any | ||||||
|             for schema_name, schema in trigger_class.configuration_schema(repository_id, None).items(): |             for schema_name, schema in trigger_class.configuration_schema(architecture, None).items(): | ||||||
|                 erased = Validate.schema_erase_required(copy.deepcopy(schema)) |                 erased = Validate.schema_erase_required(copy.deepcopy(schema)) | ||||||
|                 root[schema_name] = Validate.schema_merge(root.get(schema_name, {}), erased) |                 root[schema_name] = Validate.schema_merge(root.get(schema_name, {}), erased) | ||||||
|  |  | ||||||
|             # settings according to enabled triggers |             # settings according to enabled triggers | ||||||
|             for schema_name, schema in trigger_class.configuration_schema(repository_id, configuration).items(): |             for schema_name, schema in trigger_class.configuration_schema(architecture, configuration).items(): | ||||||
|                 root[schema_name] = Validate.schema_merge(root.get(schema_name, {}), copy.deepcopy(schema)) |                 root[schema_name] = Validate.schema_merge(root.get(schema_name, {}), copy.deepcopy(schema)) | ||||||
|  |  | ||||||
|         return root |         return root | ||||||
|  | |||||||
| @ -24,11 +24,10 @@ import sys | |||||||
| from collections.abc import Generator | from collections.abc import Generator | ||||||
| from importlib import metadata | from importlib import metadata | ||||||
|  |  | ||||||
| from ahriman import __version__ | from ahriman import version | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.core.formatters import VersionPrinter | from ahriman.core.formatters import VersionPrinter | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Versions(Handler): | class Versions(Handler): | ||||||
| @ -43,18 +42,19 @@ class Versions(Handler): | |||||||
|     PEP423_PACKAGE_NAME = re.compile(r"^[A-Za-z0-9._-]+") |     PEP423_PACKAGE_NAME = re.compile(r"^[A-Za-z0-9._-]+") | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         VersionPrinter(f"Module version {__version__}", |         VersionPrinter(f"Module version {version.__version__}", | ||||||
|                        {"Python": sys.version}).print(verbose=False, separator=" ") |                        {"Python": sys.version}).print(verbose=False, separator=" ") | ||||||
|         packages = Versions.package_dependencies("ahriman") |         packages = Versions.package_dependencies("ahriman") | ||||||
|         VersionPrinter("Installed packages", dict(packages)).print(verbose=False, separator=" ") |         VersionPrinter("Installed packages", dict(packages)).print(verbose=False, separator=" ") | ||||||
|  | |||||||
| @ -19,12 +19,9 @@ | |||||||
| # | # | ||||||
| import argparse | import argparse | ||||||
|  |  | ||||||
| from collections.abc import Generator |  | ||||||
|  |  | ||||||
| from ahriman.application.handlers import Handler | from ahriman.application.handlers import Handler | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.core.spawn import Spawn | from ahriman.core.spawn import Spawn | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Web(Handler): | class Web(Handler): | ||||||
| @ -36,63 +33,27 @@ class Web(Handler): | |||||||
|     ALLOW_MULTI_ARCHITECTURE_RUN = False  # required to be able to spawn external processes |     ALLOW_MULTI_ARCHITECTURE_RUN = False  # required to be able to spawn external processes | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, |     def run(cls, args: argparse.Namespace, architecture: str, configuration: Configuration, *, | ||||||
|             report: bool) -> None: |             report: bool, unsafe: bool) -> None: | ||||||
|         """ |         """ | ||||||
|         callback for command line |         callback for command line | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             report(bool): force enable or disable reporting |             report(bool): force enable or disable reporting | ||||||
|  |             unsafe(bool): if set no user check will be performed before path creation | ||||||
|         """ |         """ | ||||||
|         # we are using local import for optional dependencies |         # we are using local import for optional dependencies | ||||||
|         from ahriman.web.web import run_server, setup_service |         from ahriman.web.web import run_server, setup_service | ||||||
|  |  | ||||||
|         spawner_args = Web.extract_arguments(args, repository_id, configuration) |         spawner = Spawn(args.parser(), architecture, configuration) | ||||||
|         spawner = Spawn(args.parser(), repository_id, list(spawner_args)) |  | ||||||
|         spawner.start() |         spawner.start() | ||||||
|  |  | ||||||
|         application = setup_service(repository_id, configuration, spawner) |         application = setup_service(architecture, configuration, spawner) | ||||||
|         run_server(application) |         run_server(application) | ||||||
|  |  | ||||||
|         # terminate spawn process at the last |         # terminate spawn process at the last | ||||||
|         spawner.stop() |         spawner.stop() | ||||||
|         spawner.join() |         spawner.join() | ||||||
|  |  | ||||||
|     @staticmethod |  | ||||||
|     def extract_arguments(args: argparse.Namespace, repository_id: RepositoryId, |  | ||||||
|                           configuration: Configuration) -> Generator[str, None, None]: |  | ||||||
|         """ |  | ||||||
|         extract list of arguments used for current command, except for command specific ones |  | ||||||
|  |  | ||||||
|         Args: |  | ||||||
|             args(argparse.Namespace): command line args |  | ||||||
|             repository_id(RepositoryId): repository unique identifier |  | ||||||
|             configuration(Configuration): configuration instance |  | ||||||
|  |  | ||||||
|         Returns: |  | ||||||
|             Generator[str, None, None]: command line arguments which were used for this specific command |  | ||||||
|         """ |  | ||||||
|         # read architecture from the same argument list |  | ||||||
|         yield from ["--architecture", repository_id.architecture] |  | ||||||
|         if repository_id.name is not None: |  | ||||||
|             yield from ["--repository", repository_id.name] |  | ||||||
|         # read configuration path from current settings |  | ||||||
|         if (configuration_path := configuration.path) is not None: |  | ||||||
|             yield from ["--configuration", str(configuration_path)] |  | ||||||
|  |  | ||||||
|         # arguments from command line |  | ||||||
|         if args.force: |  | ||||||
|             yield "--force" |  | ||||||
|         if args.log_handler is not None: |  | ||||||
|             yield from ["--log-handler", args.log_handler.value] |  | ||||||
|         if args.quiet: |  | ||||||
|             yield "--quiet" |  | ||||||
|         if args.unsafe: |  | ||||||
|             yield "--unsafe" |  | ||||||
|  |  | ||||||
|         # arguments from configuration |  | ||||||
|         if (wait_timeout := configuration.getint("web", "wait_timeout", fallback=None)) is not None: |  | ||||||
|             yield from ["--wait-timeout", str(wait_timeout)] |  | ||||||
|  | |||||||
| @ -19,19 +19,16 @@ | |||||||
| # | # | ||||||
| import argparse | import argparse | ||||||
|  |  | ||||||
| from pathlib import Path |  | ||||||
| from types import TracebackType | from types import TracebackType | ||||||
| from typing import Literal, Self | from typing import Literal, Self | ||||||
|  |  | ||||||
| from ahriman import __version__ | from ahriman import version | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.core.exceptions import DuplicateRunError | from ahriman.core.exceptions import DuplicateRunError | ||||||
| from ahriman.core.log import LazyLogging | from ahriman.core.log import LazyLogging | ||||||
| from ahriman.core.status.client import Client | from ahriman.core.status.client import Client | ||||||
| from ahriman.core.util import check_user | from ahriman.core.util import check_user | ||||||
| from ahriman.models.build_status import BuildStatusEnum | from ahriman.models.build_status import BuildStatusEnum | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
| from ahriman.models.waiter import Waiter |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Lock(LazyLogging): | class Lock(LazyLogging): | ||||||
| @ -44,39 +41,33 @@ class Lock(LazyLogging): | |||||||
|         reporter(Client): build status reporter instance |         reporter(Client): build status reporter instance | ||||||
|         paths(RepositoryPaths): repository paths instance |         paths(RepositoryPaths): repository paths instance | ||||||
|         unsafe(bool): skip user check |         unsafe(bool): skip user check | ||||||
|         wait_timeout(int): wait in seconds until lock will free |  | ||||||
|  |  | ||||||
|     Examples: |     Examples: | ||||||
|         Instance of this class except for controlling file-based lock is also required for basic applications checks. |         Instance of this class except for controlling file-based lock is also required for basic applications checks. | ||||||
|         The common flow is to create instance in ``with`` block and handle exceptions after all:: |         The common flow is to create instance in ``with`` block and handle exceptions after all:: | ||||||
|  |  | ||||||
|             >>> from ahriman.core.configuration import Configuration |             >>> from ahriman.core.configuration import Configuration | ||||||
|             >>> from ahriman.models.repository_id import RepositoryId |  | ||||||
|             >>> |             >>> | ||||||
|             >>> configuration = Configuration() |             >>> configuration = Configuration() | ||||||
|             >>> try: |             >>> try: | ||||||
|             >>>     with Lock(args, RepositoryId("x86_64", None), configuration): |             >>>     with Lock(args, "x86_64", configuration): | ||||||
|             >>>         perform_actions() |             >>>         perform_actions() | ||||||
|             >>> except Exception as exception: |             >>> except Exception as exception: | ||||||
|             >>>     handle_exceptions(exception) |             >>>     handle_exceptions(exception) | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     def __init__(self, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration) -> None: |     def __init__(self, args: argparse.Namespace, architecture: str, configuration: Configuration) -> None: | ||||||
|         """ |         """ | ||||||
|         default constructor |         default constructor | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             args(argparse.Namespace): command line args |             args(argparse.Namespace): command line args | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|         """ |         """ | ||||||
|         lock_suffix = f"{repository_id.name}_{repository_id.architecture}" if repository_id.name is not None else repository_id.architecture |         self.path = args.lock.with_stem(f"{args.lock.stem}_{architecture}") if args.lock is not None else None | ||||||
|         self.path: Path | None = \ |         self.force = args.force | ||||||
|             args.lock.with_stem(f"{args.lock.stem}_{lock_suffix}") if args.lock is not None else None |         self.unsafe = args.unsafe | ||||||
|  |  | ||||||
|         self.force: bool = args.force |  | ||||||
|         self.unsafe: bool = args.unsafe |  | ||||||
|         self.wait_timeout: int = args.wait_timeout |  | ||||||
|  |  | ||||||
|         self.paths = configuration.repository_paths |         self.paths = configuration.repository_paths | ||||||
|         self.reporter = Client.load(configuration, report=args.report) |         self.reporter = Client.load(configuration, report=args.report) | ||||||
| @ -85,17 +76,16 @@ class Lock(LazyLogging): | |||||||
|         """ |         """ | ||||||
|         check web server version |         check web server version | ||||||
|         """ |         """ | ||||||
|         status = self.reporter.status_get() |         status = self.reporter.get_internal() | ||||||
|         if status.version is not None and status.version != __version__: |         if status.version is not None and status.version != version.__version__: | ||||||
|             self.logger.warning("status watcher version mismatch, our %s, their %s", |             self.logger.warning("status watcher version mismatch, our %s, their %s", | ||||||
|                                 __version__, status.version) |                                 version.__version__, status.version) | ||||||
|  |  | ||||||
|     def check_user(self) -> None: |     def check_user(self) -> None: | ||||||
|         """ |         """ | ||||||
|         check if current user is actually owner of ahriman root |         check if current user is actually owner of ahriman root | ||||||
|         """ |         """ | ||||||
|         check_user(self.paths, unsafe=self.unsafe) |         check_user(self.paths, unsafe=self.unsafe) | ||||||
|         self.paths.tree_create() |  | ||||||
|  |  | ||||||
|     def clear(self) -> None: |     def clear(self) -> None: | ||||||
|         """ |         """ | ||||||
| @ -110,27 +100,14 @@ class Lock(LazyLogging): | |||||||
|         create lock file |         create lock file | ||||||
|  |  | ||||||
|         Raises: |         Raises: | ||||||
|             DuplicateRunError: if lock exists and no force flag supplied |             DuplicateRun: if lock exists and no force flag supplied | ||||||
|         """ |         """ | ||||||
|         if self.path is None: |         if self.path is None: | ||||||
|             return |             return | ||||||
|         try: |         try: | ||||||
|             self.path.touch(exist_ok=self.force) |             self.path.touch(exist_ok=self.force) | ||||||
|         except FileExistsError: |         except FileExistsError: | ||||||
|             raise DuplicateRunError from None |             raise DuplicateRunError() | ||||||
|  |  | ||||||
|     def watch(self) -> None: |  | ||||||
|         """ |  | ||||||
|         watch until lock disappear |  | ||||||
|         """ |  | ||||||
|         # there are reasons why we are not using inotify here. First of all, if we would use it, it would bring to |  | ||||||
|         # race conditions because multiple processes will be notified in the same time. Secondly, it is good library, |  | ||||||
|         # but platform-specific, and we only need to check if file exists |  | ||||||
|         if self.path is None: |  | ||||||
|             return |  | ||||||
|  |  | ||||||
|         waiter = Waiter(self.wait_timeout) |  | ||||||
|         waiter.wait(self.path.is_file) |  | ||||||
|  |  | ||||||
|     def __enter__(self) -> Self: |     def __enter__(self) -> Self: | ||||||
|         """ |         """ | ||||||
| @ -139,18 +116,16 @@ class Lock(LazyLogging): | |||||||
|             1. Check user UID |             1. Check user UID | ||||||
|             2. Check if there is lock file |             2. Check if there is lock file | ||||||
|             3. Check web status watcher status |             3. Check web status watcher status | ||||||
|             4. Wait for lock file to be free |             4. Create lock file | ||||||
|             5. Create lock file and directory tree |             5. Report to status page if enabled | ||||||
|             6. Report to status page if enabled |  | ||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             Self: always instance of self |             Self: always instance of self | ||||||
|         """ |         """ | ||||||
|         self.check_user() |         self.check_user() | ||||||
|         self.check_version() |         self.check_version() | ||||||
|         self.watch() |  | ||||||
|         self.create() |         self.create() | ||||||
|         self.reporter.status_update(BuildStatusEnum.Building) |         self.reporter.update_self(BuildStatusEnum.Building) | ||||||
|         return self |         return self | ||||||
|  |  | ||||||
|     def __exit__(self, exc_type: type[Exception] | None, exc_val: Exception | None, |     def __exit__(self, exc_type: type[Exception] | None, exc_val: Exception | None, | ||||||
| @ -168,5 +143,5 @@ class Lock(LazyLogging): | |||||||
|         """ |         """ | ||||||
|         self.clear() |         self.clear() | ||||||
|         status = BuildStatusEnum.Success if exc_val is None else BuildStatusEnum.Failed |         status = BuildStatusEnum.Success if exc_val is None else BuildStatusEnum.Failed | ||||||
|         self.reporter.status_update(status) |         self.reporter.update_self(status) | ||||||
|         return False |         return False | ||||||
|  | |||||||
| @ -20,54 +20,47 @@ | |||||||
| import shutil | import shutil | ||||||
|  |  | ||||||
| from collections.abc import Callable, Generator | from collections.abc import Callable, Generator | ||||||
| from functools import cached_property |  | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
| from pyalpm import DB, Handle, Package, SIG_PACKAGE, error as PyalpmError  # type: ignore[import] | from pyalpm import DB, Handle, Package, SIG_PACKAGE, error as PyalpmError  # type: ignore[import] | ||||||
| from string import Template | from typing import Any | ||||||
|  |  | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.core.log import LazyLogging | from ahriman.core.log import LazyLogging | ||||||
| from ahriman.core.util import trim_package | from ahriman.core.util import trim_package | ||||||
| from ahriman.models.pacman_synchronization import PacmanSynchronization | from ahriman.models.pacman_synchronization import PacmanSynchronization | ||||||
| from ahriman.models.repository_id import RepositoryId |  | ||||||
| from ahriman.models.repository_paths import RepositoryPaths | from ahriman.models.repository_paths import RepositoryPaths | ||||||
|  |  | ||||||
|  |  | ||||||
| class Pacman(LazyLogging): | class Pacman(LazyLogging): | ||||||
|     """ |     """ | ||||||
|     alpm wrapper |     alpm wrapper | ||||||
|  |  | ||||||
|  |     Attributes: | ||||||
|  |         handle(Handle): pyalpm root ``Handle`` | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     def __init__(self, repository_id: RepositoryId, configuration: Configuration, *, |     handle: Handle | ||||||
|  |  | ||||||
|  |     def __init__(self, architecture: str, configuration: Configuration, *, | ||||||
|                  refresh_database: PacmanSynchronization) -> None: |                  refresh_database: PacmanSynchronization) -> None: | ||||||
|         """ |         """ | ||||||
|         default constructor |         default constructor | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             refresh_database(PacmanSynchronization): synchronize local cache to remote |             refresh_database(PacmanSynchronization): synchronize local cache to remote | ||||||
|         """ |         """ | ||||||
|         self.__create_handle_fn: Callable[[], Handle] = lambda: self.__create_handle( |         self.__create_handle_fn: Callable[[], Handle] = lambda: self.__create_handle( | ||||||
|             repository_id, configuration, refresh_database=refresh_database) |             architecture, configuration, refresh_database=refresh_database) | ||||||
|  |  | ||||||
|     @cached_property |     def __create_handle(self, architecture: str, configuration: Configuration, *, | ||||||
|     def handle(self) -> Handle: |  | ||||||
|         """ |  | ||||||
|         pyalpm handle |  | ||||||
|  |  | ||||||
|         Returns: |  | ||||||
|             Handle: generated pyalpm handle instance |  | ||||||
|         """ |  | ||||||
|         return self.__create_handle_fn() |  | ||||||
|  |  | ||||||
|     def __create_handle(self, repository_id: RepositoryId, configuration: Configuration, *, |  | ||||||
|                         refresh_database: PacmanSynchronization) -> Handle: |                         refresh_database: PacmanSynchronization) -> Handle: | ||||||
|         """ |         """ | ||||||
|         create lazy handle function |         create lazy handle function | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             repository_id(RepositoryId): repository unique identifier |             architecture(str): repository architecture | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|             refresh_database(PacmanSynchronization): synchronize local cache to remote |             refresh_database(PacmanSynchronization): synchronize local cache to remote | ||||||
|  |  | ||||||
| @ -83,7 +76,7 @@ class Pacman(LazyLogging): | |||||||
|  |  | ||||||
|         handle = Handle(str(root), str(database_path)) |         handle = Handle(str(root), str(database_path)) | ||||||
|         for repository in configuration.getlist("alpm", "repositories"): |         for repository in configuration.getlist("alpm", "repositories"): | ||||||
|             database = self.database_init(handle, repository, mirror, repository_id.architecture) |             database = self.database_init(handle, repository, mirror, architecture) | ||||||
|             self.database_copy(handle, database, pacman_root, paths, use_ahriman_cache=use_ahriman_cache) |             self.database_copy(handle, database, pacman_root, paths, use_ahriman_cache=use_ahriman_cache) | ||||||
|  |  | ||||||
|         if use_ahriman_cache and refresh_database: |         if use_ahriman_cache and refresh_database: | ||||||
| @ -118,7 +111,7 @@ class Pacman(LazyLogging): | |||||||
|         src = repository_database(pacman_root) |         src = repository_database(pacman_root) | ||||||
|         if not src.is_file(): |         if not src.is_file(): | ||||||
|             self.logger.warning("repository %s is set to be used, however, no working copy was found", database.name) |             self.logger.warning("repository %s is set to be used, however, no working copy was found", database.name) | ||||||
|             return  # database for some reason deos not exist |             return  # database for some reasons deos not exist | ||||||
|         self.logger.info("copy pacman database from operating system root to ahriman's home") |         self.logger.info("copy pacman database from operating system root to ahriman's home") | ||||||
|         shutil.copy(src, dst) |         shutil.copy(src, dst) | ||||||
|         paths.chown(dst) |         paths.chown(dst) | ||||||
| @ -138,14 +131,8 @@ class Pacman(LazyLogging): | |||||||
|         """ |         """ | ||||||
|         self.logger.info("loading pacman database %s", repository) |         self.logger.info("loading pacman database %s", repository) | ||||||
|         database: DB = handle.register_syncdb(repository, SIG_PACKAGE) |         database: DB = handle.register_syncdb(repository, SIG_PACKAGE) | ||||||
|  |  | ||||||
|         # replace variables in mirror address |         # replace variables in mirror address | ||||||
|         variables = { |         database.servers = [mirror.replace("$repo", repository).replace("$arch", architecture)] | ||||||
|             "arch": architecture, |  | ||||||
|             "repo": repository, |  | ||||||
|         } |  | ||||||
|         database.servers = [Template(mirror).safe_substitute(variables)] |  | ||||||
|  |  | ||||||
|         return database |         return database | ||||||
|  |  | ||||||
|     def database_sync(self, handle: Handle, *, force: bool) -> None: |     def database_sync(self, handle: Handle, *, force: bool) -> None: | ||||||
| @ -197,3 +184,22 @@ class Pacman(LazyLogging): | |||||||
|                 result.update(trim_package(provides) for provides in package.provides) |                 result.update(trim_package(provides) for provides in package.provides) | ||||||
|  |  | ||||||
|         return result |         return result | ||||||
|  |  | ||||||
|  |     def __getattr__(self, item: str) -> Any: | ||||||
|  |         """ | ||||||
|  |         pacman handle extractor | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             item(str): property name | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             Any: attribute by its name | ||||||
|  |  | ||||||
|  |         Raises: | ||||||
|  |             AttributeError: in case if no such attribute found | ||||||
|  |         """ | ||||||
|  |         if item == "handle": | ||||||
|  |             handle = self.__create_handle_fn() | ||||||
|  |             setattr(self, item, handle) | ||||||
|  |             return handle | ||||||
|  |         return super().__getattr__(item)  # required for logging attribute | ||||||
|  | |||||||
| @ -17,11 +17,14 @@ | |||||||
| # You should have received a copy of the GNU General Public License | # You should have received a copy of the GNU General Public License | ||||||
| # along with this program. If not, see <http://www.gnu.org/licenses/>. | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
| # | # | ||||||
|  | import requests | ||||||
|  |  | ||||||
| from typing import Any | from typing import Any | ||||||
|  |  | ||||||
| from ahriman.core.alpm.pacman import Pacman | from ahriman.core.alpm.pacman import Pacman | ||||||
| from ahriman.core.alpm.remote import Remote | from ahriman.core.alpm.remote import Remote | ||||||
| from ahriman.core.exceptions import PackageInfoError, UnknownPackageError | from ahriman.core.exceptions import PackageInfoError, UnknownPackageError | ||||||
|  | from ahriman.core.util import exception_response_text | ||||||
| from ahriman.models.aur_package import AURPackage | from ahriman.models.aur_package import AURPackage | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -33,11 +36,13 @@ class AUR(Remote): | |||||||
|         DEFAULT_AUR_URL(str): (class attribute) default AUR url |         DEFAULT_AUR_URL(str): (class attribute) default AUR url | ||||||
|         DEFAULT_RPC_URL(str): (class attribute) default AUR RPC url |         DEFAULT_RPC_URL(str): (class attribute) default AUR RPC url | ||||||
|         DEFAULT_RPC_VERSION(str): (class attribute) default AUR RPC version |         DEFAULT_RPC_VERSION(str): (class attribute) default AUR RPC version | ||||||
|  |         DEFAULT_TIMEOUT(int): (class attribute) HTTP request timeout in seconds | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     DEFAULT_AUR_URL = "https://aur.archlinux.org" |     DEFAULT_AUR_URL = "https://aur.archlinux.org" | ||||||
|     DEFAULT_RPC_URL = f"{DEFAULT_AUR_URL}/rpc" |     DEFAULT_RPC_URL = f"{DEFAULT_AUR_URL}/rpc" | ||||||
|     DEFAULT_RPC_VERSION = "5" |     DEFAULT_RPC_VERSION = "5" | ||||||
|  |     DEFAULT_TIMEOUT = 30 | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def remote_git_url(cls, package_base: str, repository: str) -> str: |     def remote_git_url(cls, package_base: str, repository: str) -> str: | ||||||
| @ -78,7 +83,7 @@ class AUR(Remote): | |||||||
|             list[AURPackage]: list of parsed packages |             list[AURPackage]: list of parsed packages | ||||||
|  |  | ||||||
|         Raises: |         Raises: | ||||||
|             PackageInfoError: for error API response |             InvalidPackageInfo: for error API response | ||||||
|         """ |         """ | ||||||
|         response_type = response["type"] |         response_type = response["type"] | ||||||
|         if response_type == "error": |         if response_type == "error": | ||||||
| @ -86,7 +91,7 @@ class AUR(Remote): | |||||||
|             raise PackageInfoError(error_details) |             raise PackageInfoError(error_details) | ||||||
|         return [AURPackage.from_json(package) for package in response["results"]] |         return [AURPackage.from_json(package) for package in response["results"]] | ||||||
|  |  | ||||||
|     def aur_request(self, request_type: str, *args: str, **kwargs: str) -> list[AURPackage]: |     def make_request(self, request_type: str, *args: str, **kwargs: str) -> list[AURPackage]: | ||||||
|         """ |         """ | ||||||
|         perform request to AUR RPC |         perform request to AUR RPC | ||||||
|  |  | ||||||
| @ -98,20 +103,30 @@ class AUR(Remote): | |||||||
|         Returns: |         Returns: | ||||||
|             list[AURPackage]: response parsed to package list |             list[AURPackage]: response parsed to package list | ||||||
|         """ |         """ | ||||||
|         query: list[tuple[str, str]] = [ |         query: dict[str, Any] = { | ||||||
|             ("type", request_type), |             "type": request_type, | ||||||
|             ("v", self.DEFAULT_RPC_VERSION), |             "v": self.DEFAULT_RPC_VERSION | ||||||
|         ] |         } | ||||||
|  |  | ||||||
|         arg_query = "arg[]" if len(args) > 1 else "arg" |         arg_query = "arg[]" if len(args) > 1 else "arg" | ||||||
|         for arg in args: |         query[arg_query] = list(args) | ||||||
|             query.append((arg_query, arg)) |  | ||||||
|  |  | ||||||
|         for key, value in kwargs.items(): |         for key, value in kwargs.items(): | ||||||
|             query.append((key, value)) |             query[key] = value | ||||||
|  |  | ||||||
|         response = self.make_request("GET", self.DEFAULT_RPC_URL, params=query) |         try: | ||||||
|  |             response = requests.get(self.DEFAULT_RPC_URL, params=query, timeout=self.DEFAULT_TIMEOUT) | ||||||
|  |             response.raise_for_status() | ||||||
|             return self.parse_response(response.json()) |             return self.parse_response(response.json()) | ||||||
|  |         except requests.HTTPError as e: | ||||||
|  |             self.logger.exception( | ||||||
|  |                 "could not perform request by using type %s: %s", | ||||||
|  |                 request_type, | ||||||
|  |                 exception_response_text(e)) | ||||||
|  |             raise | ||||||
|  |         except Exception: | ||||||
|  |             self.logger.exception("could not perform request by using type %s", request_type) | ||||||
|  |             raise | ||||||
|  |  | ||||||
|     def package_info(self, package_name: str, *, pacman: Pacman) -> AURPackage: |     def package_info(self, package_name: str, *, pacman: Pacman) -> AURPackage: | ||||||
|         """ |         """ | ||||||
| @ -123,15 +138,12 @@ class AUR(Remote): | |||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             AURPackage: package which match the package name |             AURPackage: package which match the package name | ||||||
|  |  | ||||||
|         Raises: |  | ||||||
|             UnknownPackageError: package doesn't exist |  | ||||||
|         """ |         """ | ||||||
|         packages = self.aur_request("info", package_name) |         packages = self.make_request("info", package_name) | ||||||
|         try: |         try: | ||||||
|             return next(package for package in packages if package.name == package_name) |             return next(package for package in packages if package.name == package_name) | ||||||
|         except StopIteration: |         except StopIteration: | ||||||
|             raise UnknownPackageError(package_name) from None |             raise UnknownPackageError(package_name) | ||||||
|  |  | ||||||
|     def package_search(self, *keywords: str, pacman: Pacman) -> list[AURPackage]: |     def package_search(self, *keywords: str, pacman: Pacman) -> list[AURPackage]: | ||||||
|         """ |         """ | ||||||
| @ -144,4 +156,4 @@ class AUR(Remote): | |||||||
|         Returns: |         Returns: | ||||||
|             list[AURPackage]: list of packages which match the criteria |             list[AURPackage]: list of packages which match the criteria | ||||||
|         """ |         """ | ||||||
|         return self.aur_request("search", *keywords, by="name-desc") |         return self.make_request("search", *keywords, by="name-desc") | ||||||
|  | |||||||
| @ -17,11 +17,14 @@ | |||||||
| # You should have received a copy of the GNU General Public License | # You should have received a copy of the GNU General Public License | ||||||
| # along with this program. If not, see <http://www.gnu.org/licenses/>. | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
| # | # | ||||||
|  | import requests | ||||||
|  |  | ||||||
| from typing import Any | from typing import Any | ||||||
|  |  | ||||||
| from ahriman.core.alpm.pacman import Pacman | from ahriman.core.alpm.pacman import Pacman | ||||||
| from ahriman.core.alpm.remote import Remote | from ahriman.core.alpm.remote import Remote | ||||||
| from ahriman.core.exceptions import PackageInfoError, UnknownPackageError | from ahriman.core.exceptions import PackageInfoError, UnknownPackageError | ||||||
|  | from ahriman.core.util import exception_response_text | ||||||
| from ahriman.models.aur_package import AURPackage | from ahriman.models.aur_package import AURPackage | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -31,15 +34,15 @@ class Official(Remote): | |||||||
|  |  | ||||||
|     Attributes: |     Attributes: | ||||||
|         DEFAULT_ARCHLINUX_URL(str): (class attribute) default archlinux url |         DEFAULT_ARCHLINUX_URL(str): (class attribute) default archlinux url | ||||||
|         DEFAULT_ARCHLINUX_GIT_URL(str): (class attribute) default url for git packages |  | ||||||
|         DEFAULT_SEARCH_REPOSITORIES(list[str]): (class attribute) default list of repositories to search |         DEFAULT_SEARCH_REPOSITORIES(list[str]): (class attribute) default list of repositories to search | ||||||
|         DEFAULT_RPC_URL(str): (class attribute) default archlinux repositories RPC url |         DEFAULT_RPC_URL(str): (class attribute) default archlinux repositories RPC url | ||||||
|  |         DEFAULT_TIMEOUT(int): (class attribute) HTTP request timeout in seconds | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     DEFAULT_ARCHLINUX_GIT_URL = "https://gitlab.archlinux.org" |  | ||||||
|     DEFAULT_ARCHLINUX_URL = "https://archlinux.org" |     DEFAULT_ARCHLINUX_URL = "https://archlinux.org" | ||||||
|     DEFAULT_SEARCH_REPOSITORIES = ["Core", "Extra", "Multilib"] |     DEFAULT_SEARCH_REPOSITORIES = ["Core", "Extra", "Multilib", "Community"] | ||||||
|     DEFAULT_RPC_URL = "https://archlinux.org/packages/search/json" |     DEFAULT_RPC_URL = "https://archlinux.org/packages/search/json" | ||||||
|  |     DEFAULT_TIMEOUT = 30 | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def remote_git_url(cls, package_base: str, repository: str) -> str: |     def remote_git_url(cls, package_base: str, repository: str) -> str: | ||||||
| @ -53,7 +56,9 @@ class Official(Remote): | |||||||
|         Returns: |         Returns: | ||||||
|             str: git url for the specific base |             str: git url for the specific base | ||||||
|         """ |         """ | ||||||
|         return f"{Official.DEFAULT_ARCHLINUX_GIT_URL}/archlinux/packaging/packages/{package_base}.git" |         if repository.lower() in ("core", "extra", "testing", "kde-unstable"): | ||||||
|  |             return "https://github.com/archlinux/svntogit-packages.git"  # hardcoded, ok | ||||||
|  |         return "https://github.com/archlinux/svntogit-community.git" | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def remote_web_url(cls, package_base: str) -> str: |     def remote_web_url(cls, package_base: str) -> str: | ||||||
| @ -80,13 +85,13 @@ class Official(Remote): | |||||||
|             list[AURPackage]: list of parsed packages |             list[AURPackage]: list of parsed packages | ||||||
|  |  | ||||||
|         Raises: |         Raises: | ||||||
|             PackageInfoError: for error API response |             InvalidPackageInfo: for error API response | ||||||
|         """ |         """ | ||||||
|         if not response["valid"]: |         if not response["valid"]: | ||||||
|             raise PackageInfoError("API validation error") |             raise PackageInfoError("API validation error") | ||||||
|         return [AURPackage.from_repo(package) for package in response["results"]] |         return [AURPackage.from_repo(package) for package in response["results"]] | ||||||
|  |  | ||||||
|     def arch_request(self, *args: str, by: str) -> list[AURPackage]: |     def make_request(self, *args: str, by: str) -> list[AURPackage]: | ||||||
|         """ |         """ | ||||||
|         perform request to official repositories RPC |         perform request to official repositories RPC | ||||||
|  |  | ||||||
| @ -97,15 +102,19 @@ class Official(Remote): | |||||||
|         Returns: |         Returns: | ||||||
|             list[AURPackage]: response parsed to package list |             list[AURPackage]: response parsed to package list | ||||||
|         """ |         """ | ||||||
|         query: list[tuple[str, str]] = [ |         try: | ||||||
|             ("repo", repository) |             response = requests.get( | ||||||
|             for repository in self.DEFAULT_SEARCH_REPOSITORIES |                 self.DEFAULT_RPC_URL, | ||||||
|         ] |                 params={by: args, "repo": self.DEFAULT_SEARCH_REPOSITORIES}, | ||||||
|         for arg in args: |                 timeout=self.DEFAULT_TIMEOUT) | ||||||
|             query.append((by, arg)) |             response.raise_for_status() | ||||||
|  |  | ||||||
|         response = self.make_request("GET", self.DEFAULT_RPC_URL, params=query) |  | ||||||
|             return self.parse_response(response.json()) |             return self.parse_response(response.json()) | ||||||
|  |         except requests.HTTPError as e: | ||||||
|  |             self.logger.exception("could not perform request: %s", exception_response_text(e)) | ||||||
|  |             raise | ||||||
|  |         except Exception: | ||||||
|  |             self.logger.exception("could not perform request") | ||||||
|  |             raise | ||||||
|  |  | ||||||
|     def package_info(self, package_name: str, *, pacman: Pacman) -> AURPackage: |     def package_info(self, package_name: str, *, pacman: Pacman) -> AURPackage: | ||||||
|         """ |         """ | ||||||
| @ -117,15 +126,12 @@ class Official(Remote): | |||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             AURPackage: package which match the package name |             AURPackage: package which match the package name | ||||||
|  |  | ||||||
|         Raises: |  | ||||||
|             UnknownPackageError: package doesn't exist |  | ||||||
|         """ |         """ | ||||||
|         packages = self.arch_request(package_name, by="name") |         packages = self.make_request(package_name, by="name") | ||||||
|         try: |         try: | ||||||
|             return next(package for package in packages if package.name == package_name) |             return next(package for package in packages if package.name == package_name) | ||||||
|         except StopIteration: |         except StopIteration: | ||||||
|             raise UnknownPackageError(package_name) from None |             raise UnknownPackageError(package_name) | ||||||
|  |  | ||||||
|     def package_search(self, *keywords: str, pacman: Pacman) -> list[AURPackage]: |     def package_search(self, *keywords: str, pacman: Pacman) -> list[AURPackage]: | ||||||
|         """ |         """ | ||||||
| @ -138,4 +144,4 @@ class Official(Remote): | |||||||
|         Returns: |         Returns: | ||||||
|             list[AURPackage]: list of packages which match the criteria |             list[AURPackage]: list of packages which match the criteria | ||||||
|         """ |         """ | ||||||
|         return self.arch_request(*keywords, by="q") |         return self.make_request(*keywords, by="q") | ||||||
|  | |||||||
| @ -48,11 +48,8 @@ class OfficialSyncdb(Official): | |||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             AURPackage: package which match the package name |             AURPackage: package which match the package name | ||||||
|  |  | ||||||
|         Raises: |  | ||||||
|             UnknownPackageError: package doesn't exist |  | ||||||
|         """ |         """ | ||||||
|         try: |         try: | ||||||
|             return next(AURPackage.from_pacman(package) for package in pacman.package_get(package_name)) |             return next(AURPackage.from_pacman(package) for package in pacman.package_get(package_name)) | ||||||
|         except StopIteration: |         except StopIteration: | ||||||
|             raise UnknownPackageError(package_name) from None |             raise UnknownPackageError(package_name) | ||||||
|  | |||||||
| @ -18,11 +18,11 @@ | |||||||
| # along with this program. If not, see <http://www.gnu.org/licenses/>. | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
| # | # | ||||||
| from ahriman.core.alpm.pacman import Pacman | from ahriman.core.alpm.pacman import Pacman | ||||||
| from ahriman.core.http import SyncHttpClient | from ahriman.core.log import LazyLogging | ||||||
| from ahriman.models.aur_package import AURPackage | from ahriman.models.aur_package import AURPackage | ||||||
|  |  | ||||||
|  |  | ||||||
| class Remote(SyncHttpClient): | class Remote(LazyLogging): | ||||||
|     """ |     """ | ||||||
|     base class for remote package search |     base class for remote package search | ||||||
|  |  | ||||||
|  | |||||||
| @ -71,7 +71,7 @@ class Repo(LazyLogging): | |||||||
|         """ |         """ | ||||||
|         Repo._check_output( |         Repo._check_output( | ||||||
|             "repo-add", *self.sign_args, "-R", str(self.repo_path), str(path), |             "repo-add", *self.sign_args, "-R", str(self.repo_path), str(path), | ||||||
|             exception=BuildError.from_process(path.name), |             exception=BuildError(path.name), | ||||||
|             cwd=self.paths.repository, |             cwd=self.paths.repository, | ||||||
|             logger=self.logger, |             logger=self.logger, | ||||||
|             user=self.uid) |             user=self.uid) | ||||||
| @ -98,7 +98,7 @@ class Repo(LazyLogging): | |||||||
|         # remove package from registry |         # remove package from registry | ||||||
|         Repo._check_output( |         Repo._check_output( | ||||||
|             "repo-remove", *self.sign_args, str(self.repo_path), package, |             "repo-remove", *self.sign_args, str(self.repo_path), package, | ||||||
|             exception=BuildError.from_process(package), |             exception=BuildError(package), | ||||||
|             cwd=self.paths.repository, |             cwd=self.paths.repository, | ||||||
|             logger=self.logger, |             logger=self.logger, | ||||||
|             user=self.uid) |             user=self.uid) | ||||||
|  | |||||||
| @ -74,14 +74,13 @@ class Auth(LazyLogging): | |||||||
|         Returns: |         Returns: | ||||||
|             Auth: authorization module according to current settings |             Auth: authorization module according to current settings | ||||||
|         """ |         """ | ||||||
|         match AuthSettings.from_option(configuration.get("auth", "target", fallback="disabled")): |         provider = AuthSettings.from_option(configuration.get("auth", "target", fallback="disabled")) | ||||||
|             case AuthSettings.Configuration: |         if provider == AuthSettings.Configuration: | ||||||
|             from ahriman.core.auth.mapping import Mapping |             from ahriman.core.auth.mapping import Mapping | ||||||
|             return Mapping(configuration, database) |             return Mapping(configuration, database) | ||||||
|             case AuthSettings.OAuth: |         if provider == AuthSettings.OAuth: | ||||||
|             from ahriman.core.auth.oauth import OAuth |             from ahriman.core.auth.oauth import OAuth | ||||||
|             return OAuth(configuration, database) |             return OAuth(configuration, database) | ||||||
|             case _: |  | ||||||
|         return Auth(configuration) |         return Auth(configuration) | ||||||
|  |  | ||||||
|     async def check_credentials(self, username: str | None, password: str | None) -> bool: |     async def check_credentials(self, username: str | None, password: str | None) -> bool: | ||||||
|  | |||||||
| @ -46,7 +46,7 @@ class Mapping(Auth): | |||||||
|         """ |         """ | ||||||
|         Auth.__init__(self, configuration, provider) |         Auth.__init__(self, configuration, provider) | ||||||
|         self.database = database |         self.database = database | ||||||
|         self.salt = configuration.get("auth", "salt", fallback="") |         self.salt = configuration.get("auth", "salt") | ||||||
|  |  | ||||||
|     async def check_credentials(self, username: str | None, password: str | None) -> bool: |     async def check_credentials(self, username: str | None, password: str | None) -> bool: | ||||||
|         """ |         """ | ||||||
|  | |||||||
| @ -28,8 +28,8 @@ from ahriman.models.auth_settings import AuthSettings | |||||||
|  |  | ||||||
| class OAuth(Mapping): | class OAuth(Mapping): | ||||||
|     """ |     """ | ||||||
|     User authorization implementation via OAuth. It is required to create application first and put application |     OAuth's user authorization. | ||||||
|     credentials. |     It is required to create application first and put application credentials. | ||||||
|  |  | ||||||
|     Attributes: |     Attributes: | ||||||
|         client_id(str): application client id |         client_id(str): application client id | ||||||
| @ -81,7 +81,7 @@ class OAuth(Mapping): | |||||||
|             type[aioauth_client.OAuth2Client]: loaded provider type |             type[aioauth_client.OAuth2Client]: loaded provider type | ||||||
|  |  | ||||||
|         Raises: |         Raises: | ||||||
|             OptionError: in case if invalid OAuth provider name supplied |             InvalidOption: in case if invalid OAuth provider name supplied | ||||||
|         """ |         """ | ||||||
|         provider: type[aioauth_client.OAuth2Client] = getattr(aioauth_client, name) |         provider: type[aioauth_client.OAuth2Client] = getattr(aioauth_client, name) | ||||||
|         try: |         try: | ||||||
|  | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user