mirror of
				https://github.com/arcan1s/ahriman.git
				synced 2025-10-26 03:13:45 +00:00 
			
		
		
		
	Compare commits
	
		
			44 Commits
		
	
	
		
			2.18.3
			...
			feature/tr
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| f12786a0c6 | |||
| b6309caa32 | |||
| f3fef1daf5 | |||
| 0aaf096d73 | |||
| a1b6041ca8 | |||
| a36de5c4b9 | |||
| 1bcdce4e6e | |||
| 2983d7e61a | |||
| 43fb950a0a | |||
| a28589ec74 | |||
| dfab5f56b2 | |||
| 10798b9ba3 | |||
| 358e3dc4d2 | |||
| c13cd029bc | |||
| ae32cc8fbb | |||
| dff5b775a9 | |||
| db3f20546e | |||
| 53368468a4 | |||
| 228c2cce51 | |||
| f5aec4e5c1 | |||
| 9217c8c759 | |||
| 6392520e06 | |||
| c6306631e6 | |||
| 97b906c536 | |||
| 435375721d | |||
| 4c5caba6b7 | |||
| b83df9d2c5 | |||
| f2ea76aab9 | |||
| 471b1c1331 | |||
| bd770aac2f | |||
| 6abe35ef8c | |||
| fdc27a9ebf | |||
| b729096a25 | |||
| 390b9da29e | |||
| 256376df85 | |||
| 939a94d889 | |||
| 2b1b17a1a3 | |||
| 9e6705056a | |||
| b3a3a81f70 | |||
| 3e5dbbd6cd | |||
| f41e44895d | |||
| 765bbf486f | |||
| a3c54afb82 | |||
| 7f223ecc0a | 
							
								
								
									
										12
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							| @ -13,7 +13,15 @@ jobs: | |||||||
|  |  | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|  |     container: | ||||||
|  |       image: archlinux:base | ||||||
|  |       options: -w /build | ||||||
|  |       volumes: | ||||||
|  |         - ${{ github.workspace }}:/build | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|  |       - run: pacman --noconfirm -Syu base-devel git python-tox | ||||||
|  |  | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v4 | ||||||
|  |  | ||||||
|       - name: Extract version |       - name: Extract version | ||||||
| @ -27,10 +35,6 @@ jobs: | |||||||
|           token: ${{ secrets.GITHUB_TOKEN }} |           token: ${{ secrets.GITHUB_TOKEN }} | ||||||
|           filter: 'Release \d+\.\d+\.\d+' |           filter: 'Release \d+\.\d+\.\d+' | ||||||
|  |  | ||||||
|       - uses: ConorMacBride/install-package@v1.1.0 |  | ||||||
|         with: |  | ||||||
|           apt: tox |  | ||||||
|  |  | ||||||
|       - name: Create archive |       - name: Create archive | ||||||
|         run: tox -e archive |         run: tox -e archive | ||||||
|         env: |         env: | ||||||
|  | |||||||
| @ -1,10 +1,10 @@ | |||||||
| [tool.pylint.main] | [tool.pylint.main] | ||||||
| init-hook = "sys.path.append('pylint_plugins')" | init-hook = "sys.path.append('tools')" | ||||||
| load-plugins = [ | load-plugins = [ | ||||||
|     "pylint.extensions.docparams", |     "pylint.extensions.docparams", | ||||||
|     "pylint.extensions.bad_builtin", |     "pylint.extensions.bad_builtin", | ||||||
|     "definition_order", |     "pylint_plugins.definition_order", | ||||||
|     "import_order", |     "pylint_plugins.import_order", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [tool.pylint.classes] | [tool.pylint.classes] | ||||||
|  | |||||||
							
								
								
									
										5
									
								
								.pytest.ini
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								.pytest.ini
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,5 @@ | |||||||
|  | [pytest] | ||||||
|  | addopts = --cov=ahriman --cov-report=term-missing:skip-covered --no-cov-on-fail --cov-fail-under=100 --spec | ||||||
|  | asyncio_default_fixture_loop_scope = function | ||||||
|  | asyncio_mode = auto | ||||||
|  | spec_test_format = {result} {docstring_summary} | ||||||
| @ -165,6 +165,11 @@ Again, the most checks can be performed by `tox` command, though some additional | |||||||
|  |  | ||||||
|     # Blank line again and package imports |     # Blank line again and package imports | ||||||
|     from ahriman.core.configuration import Configuration |     from ahriman.core.configuration import Configuration | ||||||
|  |     # Multiline import example | ||||||
|  |     from ahriman.core.database.operations import ( | ||||||
|  |         AuthOperations, | ||||||
|  |         BuildOperations, | ||||||
|  |     ) | ||||||
|     ``` |     ``` | ||||||
|  |  | ||||||
| * 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. | ||||||
|  | |||||||
							
								
								
									
										9
									
								
								docs/_static/architecture.dot
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								docs/_static/architecture.dot
									
									
									
									
										vendored
									
									
								
							| @ -64,7 +64,7 @@ digraph G { | |||||||
|     ahriman_core_alpm_remote_aur [fillcolor="blue",fontcolor="white",label="ahriman\.\ncore\.\nalpm\.\nremote\.\naur",shape="box"]; |     ahriman_core_alpm_remote_aur [fillcolor="blue",fontcolor="white",label="ahriman\.\ncore\.\nalpm\.\nremote\.\naur",shape="box"]; | ||||||
|     ahriman_core_alpm_remote_official [fillcolor="blue",fontcolor="white",label="ahriman\.\ncore\.\nalpm\.\nremote\.\nofficial",shape="box"]; |     ahriman_core_alpm_remote_official [fillcolor="blue",fontcolor="white",label="ahriman\.\ncore\.\nalpm\.\nremote\.\nofficial",shape="box"]; | ||||||
|     ahriman_core_alpm_remote_official_syncdb [fillcolor="blue",fontcolor="white",label="ahriman\.\ncore\.\nalpm\.\nremote\.\nofficial_syncdb",shape="box"]; |     ahriman_core_alpm_remote_official_syncdb [fillcolor="blue",fontcolor="white",label="ahriman\.\ncore\.\nalpm\.\nremote\.\nofficial_syncdb",shape="box"]; | ||||||
|     ahriman_core_alpm_remote_remote [fillcolor="#ae441e",fontcolor="#ffffff",label="ahriman\.\ncore\.\nalpm\.\nremote\.\nremote"]; |     ahriman_core_alpm_remote_remote [fillcolor="#a5401d",fontcolor="#ffffff",label="ahriman\.\ncore\.\nalpm\.\nremote\.\nremote"]; | ||||||
|     ahriman_core_alpm_repo [fillcolor="#994d33",fontcolor="#ffffff",label="ahriman\.\ncore\.\nalpm\.\nrepo"]; |     ahriman_core_alpm_repo [fillcolor="#994d33",fontcolor="#ffffff",label="ahriman\.\ncore\.\nalpm\.\nrepo"]; | ||||||
|     ahriman_core_auth [fillcolor="blue",fontcolor="white",label="ahriman\.\ncore\.\nauth",shape="box"]; |     ahriman_core_auth [fillcolor="blue",fontcolor="white",label="ahriman\.\ncore\.\nauth",shape="box"]; | ||||||
|     ahriman_core_auth_auth [fillcolor="blue",fontcolor="white",label="ahriman\.\ncore\.\nauth\.\nauth",shape="box"]; |     ahriman_core_auth_auth [fillcolor="blue",fontcolor="white",label="ahriman\.\ncore\.\nauth\.\nauth",shape="box"]; | ||||||
| @ -509,9 +509,9 @@ digraph G { | |||||||
|     ahriman_core_alpm_remote_official -> ahriman_core_alpm_remote [fillcolor="blue",minlen="0",weight="4"]; |     ahriman_core_alpm_remote_official -> ahriman_core_alpm_remote [fillcolor="blue",minlen="0",weight="4"]; | ||||||
|     ahriman_core_alpm_remote_official -> ahriman_core_alpm_remote_official_syncdb [fillcolor="blue",minlen="0",weight="4"]; |     ahriman_core_alpm_remote_official -> ahriman_core_alpm_remote_official_syncdb [fillcolor="blue",minlen="0",weight="4"]; | ||||||
|     ahriman_core_alpm_remote_official_syncdb -> ahriman_core_alpm_remote [fillcolor="blue",minlen="0",weight="4"]; |     ahriman_core_alpm_remote_official_syncdb -> ahriman_core_alpm_remote [fillcolor="blue",minlen="0",weight="4"]; | ||||||
|     ahriman_core_alpm_remote_remote -> ahriman_core_alpm_remote [fillcolor="#ae441e",minlen="0",weight="4"]; |     ahriman_core_alpm_remote_remote -> ahriman_core_alpm_remote [fillcolor="#a5401d",minlen="0",weight="4"]; | ||||||
|     ahriman_core_alpm_remote_remote -> ahriman_core_alpm_remote_aur [fillcolor="#ae441e",minlen="0",weight="4"]; |     ahriman_core_alpm_remote_remote -> ahriman_core_alpm_remote_aur [fillcolor="#a5401d",minlen="0",weight="4"]; | ||||||
|     ahriman_core_alpm_remote_remote -> ahriman_core_alpm_remote_official [fillcolor="#ae441e",minlen="0",weight="4"]; |     ahriman_core_alpm_remote_remote -> ahriman_core_alpm_remote_official [fillcolor="#a5401d",minlen="0",weight="4"]; | ||||||
|     ahriman_core_alpm_repo -> ahriman_core_repository_repository_properties [fillcolor="#994d33",minlen="2",weight="2"]; |     ahriman_core_alpm_repo -> ahriman_core_repository_repository_properties [fillcolor="#994d33",minlen="2",weight="2"]; | ||||||
|     ahriman_core_auth -> ahriman_web_keys [fillcolor="blue",minlen="2"]; |     ahriman_core_auth -> ahriman_web_keys [fillcolor="blue",minlen="2"]; | ||||||
|     ahriman_core_auth -> ahriman_web_middlewares_auth_handler [fillcolor="blue",minlen="3"]; |     ahriman_core_auth -> ahriman_web_middlewares_auth_handler [fillcolor="blue",minlen="3"]; | ||||||
| @ -710,6 +710,7 @@ digraph G { | |||||||
|     ahriman_core_exceptions -> ahriman_core_alpm_remote_aur [fillcolor="#ef4306",minlen="2",weight="2"]; |     ahriman_core_exceptions -> ahriman_core_alpm_remote_aur [fillcolor="#ef4306",minlen="2",weight="2"]; | ||||||
|     ahriman_core_exceptions -> ahriman_core_alpm_remote_official [fillcolor="#ef4306",minlen="2",weight="2"]; |     ahriman_core_exceptions -> ahriman_core_alpm_remote_official [fillcolor="#ef4306",minlen="2",weight="2"]; | ||||||
|     ahriman_core_exceptions -> ahriman_core_alpm_remote_official_syncdb [fillcolor="#ef4306",minlen="2",weight="2"]; |     ahriman_core_exceptions -> ahriman_core_alpm_remote_official_syncdb [fillcolor="#ef4306",minlen="2",weight="2"]; | ||||||
|  |     ahriman_core_exceptions -> ahriman_core_alpm_remote_remote [fillcolor="#ef4306",minlen="2",weight="2"]; | ||||||
|     ahriman_core_exceptions -> ahriman_core_alpm_repo [fillcolor="#ef4306",minlen="2",weight="2"]; |     ahriman_core_exceptions -> ahriman_core_alpm_repo [fillcolor="#ef4306",minlen="2",weight="2"]; | ||||||
|     ahriman_core_exceptions -> ahriman_core_auth_oauth [fillcolor="#ef4306",minlen="2",weight="2"]; |     ahriman_core_exceptions -> ahriman_core_auth_oauth [fillcolor="#ef4306",minlen="2",weight="2"]; | ||||||
|     ahriman_core_exceptions -> ahriman_core_auth_pam [fillcolor="#ef4306",minlen="2",weight="2"]; |     ahriman_core_exceptions -> ahriman_core_auth_pam [fillcolor="#ef4306",minlen="2",weight="2"]; | ||||||
|  | |||||||
| @ -100,6 +100,14 @@ ahriman.application.handlers.rebuild module | |||||||
|    :no-undoc-members: |    :no-undoc-members: | ||||||
|    :show-inheritance: |    :show-inheritance: | ||||||
|  |  | ||||||
|  | ahriman.application.handlers.reload module | ||||||
|  | ------------------------------------------ | ||||||
|  |  | ||||||
|  | .. automodule:: ahriman.application.handlers.reload | ||||||
|  |    :members: | ||||||
|  |    :no-undoc-members: | ||||||
|  |    :show-inheritance: | ||||||
|  |  | ||||||
| ahriman.application.handlers.remove module | ahriman.application.handlers.remove module | ||||||
| ------------------------------------------ | ------------------------------------------ | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										29
									
								
								docs/ahriman.core.archive.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								docs/ahriman.core.archive.rst
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,29 @@ | |||||||
|  | ahriman.core.archive package | ||||||
|  | ============================ | ||||||
|  |  | ||||||
|  | Submodules | ||||||
|  | ---------- | ||||||
|  |  | ||||||
|  | ahriman.core.archive.archive\_tree module | ||||||
|  | ----------------------------------------- | ||||||
|  |  | ||||||
|  | .. automodule:: ahriman.core.archive.archive_tree | ||||||
|  |    :members: | ||||||
|  |    :no-undoc-members: | ||||||
|  |    :show-inheritance: | ||||||
|  |  | ||||||
|  | ahriman.core.archive.archive\_trigger module | ||||||
|  | -------------------------------------------- | ||||||
|  |  | ||||||
|  | .. automodule:: ahriman.core.archive.archive_trigger | ||||||
|  |    :members: | ||||||
|  |    :no-undoc-members: | ||||||
|  |    :show-inheritance: | ||||||
|  |  | ||||||
|  | Module contents | ||||||
|  | --------------- | ||||||
|  |  | ||||||
|  | .. automodule:: ahriman.core.archive | ||||||
|  |    :members: | ||||||
|  |    :no-undoc-members: | ||||||
|  |    :show-inheritance: | ||||||
| @ -132,6 +132,14 @@ ahriman.core.database.migrations.m015\_logs\_process\_id module | |||||||
|    :no-undoc-members: |    :no-undoc-members: | ||||||
|    :show-inheritance: |    :show-inheritance: | ||||||
|  |  | ||||||
|  | ahriman.core.database.migrations.m016\_archive module | ||||||
|  | ----------------------------------------------------- | ||||||
|  |  | ||||||
|  | .. automodule:: ahriman.core.database.migrations.m016_archive | ||||||
|  |    :members: | ||||||
|  |    :no-undoc-members: | ||||||
|  |    :show-inheritance: | ||||||
|  |  | ||||||
| Module contents | Module contents | ||||||
| --------------- | --------------- | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										29
									
								
								docs/ahriman.core.housekeeping.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								docs/ahriman.core.housekeeping.rst
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,29 @@ | |||||||
|  | ahriman.core.housekeeping package | ||||||
|  | ================================= | ||||||
|  |  | ||||||
|  | Submodules | ||||||
|  | ---------- | ||||||
|  |  | ||||||
|  | ahriman.core.housekeeping.archive\_rotation\_trigger module | ||||||
|  | ----------------------------------------------------------- | ||||||
|  |  | ||||||
|  | .. automodule:: ahriman.core.housekeeping.archive_rotation_trigger | ||||||
|  |    :members: | ||||||
|  |    :no-undoc-members: | ||||||
|  |    :show-inheritance: | ||||||
|  |  | ||||||
|  | ahriman.core.housekeeping.logs\_rotation\_trigger module | ||||||
|  | -------------------------------------------------------- | ||||||
|  |  | ||||||
|  | .. automodule:: ahriman.core.housekeeping.logs_rotation_trigger | ||||||
|  |    :members: | ||||||
|  |    :no-undoc-members: | ||||||
|  |    :show-inheritance: | ||||||
|  |  | ||||||
|  | Module contents | ||||||
|  | --------------- | ||||||
|  |  | ||||||
|  | .. automodule:: ahriman.core.housekeeping | ||||||
|  |    :members: | ||||||
|  |    :no-undoc-members: | ||||||
|  |    :show-inheritance: | ||||||
| @ -8,6 +8,7 @@ Subpackages | |||||||
|    :maxdepth: 4 |    :maxdepth: 4 | ||||||
|  |  | ||||||
|    ahriman.core.alpm |    ahriman.core.alpm | ||||||
|  |    ahriman.core.archive | ||||||
|    ahriman.core.auth |    ahriman.core.auth | ||||||
|    ahriman.core.build_tools |    ahriman.core.build_tools | ||||||
|    ahriman.core.configuration |    ahriman.core.configuration | ||||||
| @ -15,6 +16,7 @@ Subpackages | |||||||
|    ahriman.core.distributed |    ahriman.core.distributed | ||||||
|    ahriman.core.formatters |    ahriman.core.formatters | ||||||
|    ahriman.core.gitremote |    ahriman.core.gitremote | ||||||
|  |    ahriman.core.housekeeping | ||||||
|    ahriman.core.http |    ahriman.core.http | ||||||
|    ahriman.core.log |    ahriman.core.log | ||||||
|    ahriman.core.report |    ahriman.core.report | ||||||
|  | |||||||
| @ -44,6 +44,14 @@ ahriman.web.schemas.changes\_schema module | |||||||
|    :no-undoc-members: |    :no-undoc-members: | ||||||
|    :show-inheritance: |    :show-inheritance: | ||||||
|  |  | ||||||
|  | ahriman.web.schemas.configuration\_schema module | ||||||
|  | ------------------------------------------------ | ||||||
|  |  | ||||||
|  | .. automodule:: ahriman.web.schemas.configuration_schema | ||||||
|  |    :members: | ||||||
|  |    :no-undoc-members: | ||||||
|  |    :show-inheritance: | ||||||
|  |  | ||||||
| ahriman.web.schemas.counters\_schema module | ahriman.web.schemas.counters\_schema module | ||||||
| ------------------------------------------- | ------------------------------------------- | ||||||
|  |  | ||||||
| @ -140,6 +148,14 @@ ahriman.web.schemas.logs\_schema module | |||||||
|    :no-undoc-members: |    :no-undoc-members: | ||||||
|    :show-inheritance: |    :show-inheritance: | ||||||
|  |  | ||||||
|  | ahriman.web.schemas.logs\_search\_schema module | ||||||
|  | ----------------------------------------------- | ||||||
|  |  | ||||||
|  | .. automodule:: ahriman.web.schemas.logs_search_schema | ||||||
|  |    :members: | ||||||
|  |    :no-undoc-members: | ||||||
|  |    :show-inheritance: | ||||||
|  |  | ||||||
| ahriman.web.schemas.oauth2\_schema module | ahriman.web.schemas.oauth2\_schema module | ||||||
| ----------------------------------------- | ----------------------------------------- | ||||||
|  |  | ||||||
|  | |||||||
| @ -12,6 +12,14 @@ ahriman.web.views.v1.service.add module | |||||||
|    :no-undoc-members: |    :no-undoc-members: | ||||||
|    :show-inheritance: |    :show-inheritance: | ||||||
|  |  | ||||||
|  | ahriman.web.views.v1.service.config module | ||||||
|  | ------------------------------------------ | ||||||
|  |  | ||||||
|  | .. automodule:: ahriman.web.views.v1.service.config | ||||||
|  |    :members: | ||||||
|  |    :no-undoc-members: | ||||||
|  |    :show-inheritance: | ||||||
|  |  | ||||||
| ahriman.web.views.v1.service.logs module | ahriman.web.views.v1.service.logs module | ||||||
| ---------------------------------------- | ---------------------------------------- | ||||||
|  |  | ||||||
|  | |||||||
| @ -40,6 +40,7 @@ This package contains everything required for the most of application actions an | |||||||
| * ``ahriman.core.distributed`` package with triggers and helpers for distributed build system. | * ``ahriman.core.distributed`` package with triggers and helpers for distributed build system. | ||||||
| * ``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.housekeeping`` package provides few triggers for removing old data. | ||||||
| * ``ahriman.core.http`` package provides HTTP clients which can be used later by other classes. | * ``ahriman.core.http`` package provides HTTP clients which can be used later by other classes. | ||||||
| * ``ahriman.core.log`` is a log utils package. It includes logger loader class, custom HTTP based logger and some wrappers. | * ``ahriman.core.log`` is a log utils package. It includes logger loader class, custom HTTP based logger and some wrappers. | ||||||
| * ``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. | ||||||
|  | |||||||
| @ -65,6 +65,8 @@ will try to read value from ``SECRET`` environment variable. In case if the requ | |||||||
|  |  | ||||||
| will eventually lead ``key`` option in section ``section1`` to be set to the value of ``HOME`` environment variable (if available). | will eventually lead ``key`` option in section ``section1`` to be set to the value of ``HOME`` environment variable (if available). | ||||||
|  |  | ||||||
|  | Moreover, configuration can be read from environment variables directly by following the same naming convention, e.g. in the example above, one can have environment variable named ``section1:key`` (e.g. ``section1:key=$HOME``) and it will be substituted to the configuration with the highest priority. | ||||||
|  |  | ||||||
| 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 | ||||||
| @ -81,7 +83,6 @@ Base configuration settings. | |||||||
| * ``apply_migrations`` - perform database migrations on the application start, boolean, optional, default ``yes``. Useful if you are using git version. Note, however, that this option must be changed only if you know what to do and going to handle migrations manually. | * ``apply_migrations`` - perform database migrations on the application start, boolean, optional, default ``yes``. Useful if you are using git version. Note, however, that this option must be changed only if you know what to do and going to handle migrations manually. | ||||||
| * ``database`` - path to the application SQLite database, string, required. | * ``database`` - path to the application SQLite database, string, required. | ||||||
| * ``include`` - path to directory with configuration files overrides, string, optional. Files will be read in alphabetical order. | * ``include`` - path to directory with configuration files overrides, string, optional. Files will be read in alphabetical order. | ||||||
| * ``keep_last_logs`` - amount of build logs to be kept for each package, integer, optional ,default ``0``. Logs will be cleared at the end of each process. |  | ||||||
| * ``logging`` - path to logging configuration, string, required. Check ``logging.ini`` for reference. | * ``logging`` - path to logging configuration, string, required. Check ``logging.ini`` for reference. | ||||||
|  |  | ||||||
| ``alpm:*`` groups | ``alpm:*`` groups | ||||||
| @ -96,6 +97,13 @@ libalpm and AUR related configuration. Group name can refer to architecture, e.g | |||||||
| * ``sync_files_database`` - download files database from mirror, boolean, required. | * ``sync_files_database`` - download files database from mirror, boolean, required. | ||||||
| * ``use_ahriman_cache`` - use local pacman package cache instead of system one, boolean, required. With this option enabled you might want to refresh database periodically (available as additional flag for some subcommands). If set to ``no``, databases must be synchronized manually. | * ``use_ahriman_cache`` - use local pacman package cache instead of system one, boolean, required. With this option enabled you might want to refresh database periodically (available as additional flag for some subcommands). If set to ``no``, databases must be synchronized manually. | ||||||
|  |  | ||||||
|  | ``archive`` group | ||||||
|  | ----------------- | ||||||
|  |  | ||||||
|  | Describes settings for packages archives management extensions. | ||||||
|  |  | ||||||
|  | * ``keep_built_packages`` - keep this amount of built packages with different versions, integer, required. ``0`` (or negative number) will effectively disable archives removal. | ||||||
|  |  | ||||||
| ``auth`` group | ``auth`` group | ||||||
| -------------- | -------------- | ||||||
|  |  | ||||||
| @ -138,6 +146,8 @@ Build related configuration. Group name can refer to architecture, e.g. ``build: | |||||||
|  |  | ||||||
| Base repository settings. | Base repository settings. | ||||||
|  |  | ||||||
|  | * ``architecture`` - repository architecture, string. This field is read-only and generated automatically from run options if possible. | ||||||
|  | * ``name`` - repository name, string. This field is read-only and generated automatically from run options if possible. | ||||||
| * ``root`` - root path for application, string, required. | * ``root`` - root path for application, string, required. | ||||||
|  |  | ||||||
| ``sign:*`` groups | ``sign:*`` groups | ||||||
| @ -166,6 +176,7 @@ Reporting to web service related settings. In most cases there is fallback to we | |||||||
| Web server settings. This feature requires ``aiohttp`` libraries to be installed. | Web server settings. This feature requires ``aiohttp`` libraries to be installed. | ||||||
|  |  | ||||||
| * ``address`` - optional address in form ``proto://host:port`` (``port`` can be omitted in case of default ``proto`` ports), will be used instead of ``http://{host}:{port}`` in case if set, string, optional. This option is required in case if ``OAuth`` provider is used. | * ``address`` - optional address in form ``proto://host:port`` (``port`` can be omitted in case of default ``proto`` ports), will be used instead of ``http://{host}:{port}`` in case if set, string, optional. This option is required in case if ``OAuth`` provider is used. | ||||||
|  | * ``autorefresh_intervals`` - enable page auto refresh options, space separated list of integers, optional. The first defined interval will be used as default. If no intervals set, the auto refresh buttons will be disabled. If first element of the list equals ``0``, auto refresh will be disabled by default. | ||||||
| * ``enable_archive_upload`` - allow to upload packages via HTTP (i.e. call of ``/api/v1/service/upload`` uri), boolean, optional, default ``no``. | * ``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. | ||||||
| @ -179,7 +190,7 @@ Web server settings. This feature requires ``aiohttp`` libraries to be installed | |||||||
| * ``wait_timeout`` - wait timeout in seconds, maximum amount of time to be waited before lock will be free, integer, optional. | * ``wait_timeout`` - wait timeout in seconds, maximum amount of time to be waited before lock will be free, integer, optional. | ||||||
|  |  | ||||||
| ``keyring`` group | ``keyring`` group | ||||||
| -------------------- | ----------------- | ||||||
|  |  | ||||||
| Keyring package generator plugin. | Keyring package generator plugin. | ||||||
|  |  | ||||||
| @ -197,6 +208,13 @@ Keyring generator plugin | |||||||
| * ``revoked`` - list of revoked packagers keys, space separated list of strings, optional. | * ``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. | * ``trusted`` - list of master keys, space separated list of strings, optional, if not set, the ``key`` option from ``sign`` group will be used. | ||||||
|  |  | ||||||
|  | ``housekeeping`` group | ||||||
|  | ---------------------- | ||||||
|  |  | ||||||
|  | This section describes settings for the ``ahriman.core.housekeeping.LogsRotationTrigger`` plugin. | ||||||
|  |  | ||||||
|  | * ``keep_last_logs`` - amount of build logs to be kept for each package, integer, optional ,default ``0``. Logs will be cleared at the end of each process. | ||||||
|  |  | ||||||
| ``mirrorlist`` group | ``mirrorlist`` group | ||||||
| -------------------- | -------------------- | ||||||
|  |  | ||||||
|  | |||||||
| @ -2,7 +2,7 @@ | |||||||
|  |  | ||||||
| pkgbase='ahriman' | pkgbase='ahriman' | ||||||
| pkgname=('ahriman' 'ahriman-core' 'ahriman-triggers' 'ahriman-web') | pkgname=('ahriman' 'ahriman-core' 'ahriman-triggers' 'ahriman-web') | ||||||
| pkgver=2.18.3 | pkgver=2.19.0 | ||||||
| pkgrel=1 | pkgrel=1 | ||||||
| pkgdesc="ArcH linux ReposItory MANager" | pkgdesc="ArcH linux ReposItory MANager" | ||||||
| arch=('any') | arch=('any') | ||||||
| @ -40,6 +40,7 @@ package_ahriman-core() { | |||||||
|                 'rsync: sync by using rsync') |                 'rsync: sync by using rsync') | ||||||
|     install="$pkgbase.install" |     install="$pkgbase.install" | ||||||
|     backup=('etc/ahriman.ini' |     backup=('etc/ahriman.ini' | ||||||
|  |             'etc/ahriman.ini.d/00-housekeeping.ini' | ||||||
|             'etc/ahriman.ini.d/logging.ini') |             'etc/ahriman.ini.d/logging.ini') | ||||||
|  |  | ||||||
|     cd "$pkgbase-$pkgver" |     cd "$pkgbase-$pkgver" | ||||||
| @ -49,6 +50,7 @@ package_ahriman-core() { | |||||||
|  |  | ||||||
|     # keep usr/share configs as reference and copy them to /etc |     # keep usr/share configs as reference and copy them to /etc | ||||||
|     install -Dm644 "$pkgdir/usr/share/$pkgbase/settings/ahriman.ini" "$pkgdir/etc/ahriman.ini" |     install -Dm644 "$pkgdir/usr/share/$pkgbase/settings/ahriman.ini" "$pkgdir/etc/ahriman.ini" | ||||||
|  |     install -Dm644 "$pkgdir/usr/share/$pkgbase/settings/ahriman.ini.d/00-housekeeping.ini" "$pkgdir/etc/ahriman.ini.d/00-housekeeping.ini" | ||||||
|     install -Dm644 "$pkgdir/usr/share/$pkgbase/settings/ahriman.ini.d/logging.ini" "$pkgdir/etc/ahriman.ini.d/logging.ini" |     install -Dm644 "$pkgdir/usr/share/$pkgbase/settings/ahriman.ini.d/logging.ini" "$pkgdir/etc/ahriman.ini.d/logging.ini" | ||||||
|  |  | ||||||
|     install -Dm644 "$srcdir/$pkgbase.sysusers" "$pkgdir/usr/lib/sysusers.d/$pkgbase.conf" |     install -Dm644 "$srcdir/$pkgbase.sysusers" "$pkgdir/usr/lib/sysusers.d/$pkgbase.conf" | ||||||
|  | |||||||
| @ -5,6 +5,7 @@ After=network.target | |||||||
| [Service] | [Service] | ||||||
| Type=simple | Type=simple | ||||||
| ExecStart=/usr/bin/ahriman web | ExecStart=/usr/bin/ahriman web | ||||||
|  | ExecReload=/usr/bin/ahriman web-reload | ||||||
| User=ahriman | User=ahriman | ||||||
| Group=ahriman | Group=ahriman | ||||||
|  |  | ||||||
|  | |||||||
| @ -7,8 +7,6 @@ logging = ahriman.ini.d/logging.ini | |||||||
| ;apply_migrations = yes | ;apply_migrations = yes | ||||||
| ; Path to the application SQLite database. | ; Path to the application SQLite database. | ||||||
| database = ${repository:root}/ahriman.db | database = ${repository:root}/ahriman.db | ||||||
| ; Keep last build logs for each package |  | ||||||
| keep_last_logs = 5 |  | ||||||
|  |  | ||||||
| [alpm] | [alpm] | ||||||
| ; Path to pacman system database cache. | ; Path to pacman system database cache. | ||||||
| @ -45,9 +43,13 @@ triggers[] = ahriman.core.gitremote.RemotePullTrigger | |||||||
| triggers[] = ahriman.core.report.ReportTrigger | triggers[] = ahriman.core.report.ReportTrigger | ||||||
| triggers[] = ahriman.core.upload.UploadTrigger | triggers[] = ahriman.core.upload.UploadTrigger | ||||||
| triggers[] = ahriman.core.gitremote.RemotePushTrigger | triggers[] = ahriman.core.gitremote.RemotePushTrigger | ||||||
|  | triggers[] = ahriman.core.housekeeping.LogsRotationTrigger | ||||||
|  | triggers[] = ahriman.core.housekeeping.ArchiveRotationTrigger | ||||||
| ; List of well-known triggers. Used only for configuration purposes. | ; List of well-known triggers. Used only for configuration purposes. | ||||||
| triggers_known[] = ahriman.core.gitremote.RemotePullTrigger | triggers_known[] = ahriman.core.gitremote.RemotePullTrigger | ||||||
| triggers_known[] = ahriman.core.gitremote.RemotePushTrigger | triggers_known[] = ahriman.core.gitremote.RemotePushTrigger | ||||||
|  | triggers_known[] = ahriman.core.housekeeping.ArchiveRotationTrigger | ||||||
|  | triggers_known[] = ahriman.core.housekeeping.LogsRotationTrigger | ||||||
| triggers_known[] = ahriman.core.report.ReportTrigger | triggers_known[] = ahriman.core.report.ReportTrigger | ||||||
| triggers_known[] = ahriman.core.upload.UploadTrigger | triggers_known[] = ahriman.core.upload.UploadTrigger | ||||||
| ; Maximal age in seconds of the VCS packages before their version will be updated with its remote source. | ; Maximal age in seconds of the VCS packages before their version will be updated with its remote source. | ||||||
|  | |||||||
| @ -0,0 +1,7 @@ | |||||||
|  | [archive] | ||||||
|  | ; Keep amount of last built packages in archive. 0 means keep all packages | ||||||
|  | keep_built_packages = 1 | ||||||
|  |  | ||||||
|  | [logs-rotation] | ||||||
|  | ; Keep last build logs for each package | ||||||
|  | keep_last_logs = 5 | ||||||
| @ -1,5 +1,6 @@ | |||||||
| [build] | [build] | ||||||
| ; List of well-known triggers. Used only for configuration purposes. | ; List of well-known triggers. Used only for configuration purposes. | ||||||
|  | triggers_known[] = ahriman.core.archive.ArchiveTrigger | ||||||
| triggers_known[] = ahriman.core.distributed.WorkerLoaderTrigger | triggers_known[] = ahriman.core.distributed.WorkerLoaderTrigger | ||||||
| triggers_known[] = ahriman.core.distributed.WorkerTrigger | triggers_known[] = ahriman.core.distributed.WorkerTrigger | ||||||
| triggers_known[] = ahriman.core.support.KeyringTrigger | triggers_known[] = ahriman.core.support.KeyringTrigger | ||||||
|  | |||||||
| @ -28,6 +28,10 @@ allow_read_only = yes | |||||||
| ; External address of the web service. Will be used for some features like OAuth. If none set will be generated as | ; External address of the web service. Will be used for some features like OAuth. If none set will be generated as | ||||||
| ;     address = http://${web:host}:${web:port} | ;     address = http://${web:host}:${web:port} | ||||||
| ;address = http://${web:host}:${web:port} | ;address = http://${web:host}:${web:port} | ||||||
|  | ; Enable page auto refresh. Intervals are given in seconds. Default interval is always the first element of the list. | ||||||
|  | ; If no intervals set, auto refresh will be disabled. 0 can only be the first element and will disable auto refresh | ||||||
|  | ; by default. | ||||||
|  | autorefresh_intervals = 5 1 10 30 60 | ||||||
| ; Enable file upload endpoint used by some triggers. | ; Enable file upload endpoint used by some triggers. | ||||||
| ;enable_archive_upload = no | ;enable_archive_upload = no | ||||||
| ; Address to bind the server. | ; Address to bind the server. | ||||||
|  | |||||||
| @ -55,6 +55,11 @@ | |||||||
|                                 <i class="bi bi-play"></i> update |                                 <i class="bi bi-play"></i> update | ||||||
|                             </button> |                             </button> | ||||||
|                         </li> |                         </li> | ||||||
|  |                         <li> | ||||||
|  |                             <button id="update-repositories-button" class="btn dropdown-item" onclick="refreshDatabases()"> | ||||||
|  |                                 <i class="bi bi-arrow-down-circle"></i> update pacman databases | ||||||
|  |                             </button> | ||||||
|  |                         </li> | ||||||
|                         <li> |                         <li> | ||||||
|                             <button id="package-rebuild-button" class="btn dropdown-item" data-bs-toggle="modal" data-bs-target="#package-rebuild-modal"> |                             <button id="package-rebuild-button" class="btn dropdown-item" data-bs-toggle="modal" data-bs-target="#package-rebuild-modal"> | ||||||
|                                 <i class="bi bi-arrow-clockwise"></i> rebuild |                                 <i class="bi bi-arrow-clockwise"></i> rebuild | ||||||
| @ -75,10 +80,28 @@ | |||||||
|                 <button type="button" class="btn btn-secondary" onclick="reload()"> |                 <button type="button" class="btn btn-secondary" onclick="reload()"> | ||||||
|                     <i class="bi bi-arrow-clockwise"></i><span class="d-none d-sm-inline"> reload</span> |                     <i class="bi bi-arrow-clockwise"></i><span class="d-none d-sm-inline"> reload</span> | ||||||
|                 </button> |                 </button> | ||||||
|  |  | ||||||
|  |                 {% if autorefresh_intervals %} | ||||||
|  |                     <div class="btn-group"> | ||||||
|  |                         <input id="table-autoreload-button" type="checkbox" class="btn-check" autocomplete="off" onclick="toggleTableAutoReload()" checked> | ||||||
|  |                         <label for="table-autoreload-button" class="btn btn-outline-secondary" title="toggle auto reload"><i class="bi bi-clock"></i></label> | ||||||
|  |                         <button type="button" class="btn btn-outline-secondary dropdown-toggle dropdown-toggle-split" data-bs-toggle="dropdown" aria-expanded="false"> | ||||||
|  |                             <span class="visually-hidden">select interval</span> | ||||||
|  |                         </button> | ||||||
|  |                         <ul id="table-autoreload-input" class="dropdown-menu"> | ||||||
|  |                             {% for interval in autorefresh_intervals %} | ||||||
|  |                                 <li><a class="dropdown-item {{ "active" if interval.is_active }}" onclick="toggleTableAutoReload({{ interval.interval }})" data-interval="{{ interval.interval }}">{{ interval.text }}</a></li> | ||||||
|  |                             {% endfor %} | ||||||
|  |                         </ul> | ||||||
|  |                     </div> | ||||||
|  |                 {% endif %} | ||||||
|             </div> |             </div> | ||||||
|  |  | ||||||
|             <table id="packages" |             <table id="packages" | ||||||
|                    data-classes="table table-hover" |                    data-classes="table table-hover" | ||||||
|  |                    data-cookie="true" | ||||||
|  |                    data-cookie-id-table="ahriman-packages" | ||||||
|  |                    data-cookie-storage="localStorage" | ||||||
|                    data-export-options='{"fileName": "packages"}' |                    data-export-options='{"fileName": "packages"}' | ||||||
|                    data-filter-control="true" |                    data-filter-control="true" | ||||||
|                    data-filter-control-visible="false" |                    data-filter-control-visible="false" | ||||||
| @ -97,8 +120,8 @@ | |||||||
|                    data-sortable="true" |                    data-sortable="true" | ||||||
|                    data-sort-name="base" |                    data-sort-name="base" | ||||||
|                    data-sort-order="asc" |                    data-sort-order="asc" | ||||||
|                    data-toggle="table" |                    data-toolbar="#toolbar" | ||||||
|                    data-toolbar="#toolbar"> |                    data-unique-id="id"> | ||||||
|                 <thead class="table-primary"> |                 <thead class="table-primary"> | ||||||
|                     <tr> |                     <tr> | ||||||
|                         <th data-checkbox="true"></th> |                         <th data-checkbox="true"></th> | ||||||
|  | |||||||
| @ -3,7 +3,9 @@ | |||||||
|  |  | ||||||
|     function createAlert(title, message, clz, action, id) { |     function createAlert(title, message, clz, action, id) { | ||||||
|         id ??= md5(title + message); // MD5 id from the content |         id ??= md5(title + message); // MD5 id from the content | ||||||
|         if (alertPlaceholder.querySelector(`#alert-${id}`)) return; // check if there are duplicates |         if (alertPlaceholder.querySelector(`#alert-${id}`)) { | ||||||
|  |             return; // check if there are duplicates | ||||||
|  |         } | ||||||
|  |  | ||||||
|         const wrapper = document.createElement("div"); |         const wrapper = document.createElement("div"); | ||||||
|         wrapper.id = `alert-${id}`; |         wrapper.id = `alert-${id}`; | ||||||
|  | |||||||
| @ -51,6 +51,87 @@ | |||||||
|     const dashboardPackagesCountChartCanvas = document.getElementById("dashboard-packages-count-chart"); |     const dashboardPackagesCountChartCanvas = document.getElementById("dashboard-packages-count-chart"); | ||||||
|     let dashboardPackagesCountChart = null; |     let dashboardPackagesCountChart = null; | ||||||
|  |  | ||||||
|  |     function statusLoad() { | ||||||
|  |         const badgeClass = status => { | ||||||
|  |             if (status === "pending") return "btn-outline-warning"; | ||||||
|  |             if (status === "building") return "btn-outline-warning"; | ||||||
|  |             if (status === "failed") return "btn-outline-danger"; | ||||||
|  |             if (status === "success") return "btn-outline-success"; | ||||||
|  |             return "btn-outline-secondary"; | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         makeRequest( | ||||||
|  |             "/api/v1/status", | ||||||
|  |             { | ||||||
|  |                 query: { | ||||||
|  |                     architecture: repository.architecture, | ||||||
|  |                     repository: repository.repository, | ||||||
|  |                 }, | ||||||
|  |                 convert: response => response.json(), | ||||||
|  |             }, | ||||||
|  |             data => { | ||||||
|  |                 versionBadge.innerHTML = `<i class="bi bi-github"></i> ahriman ${safe(data.version)}`; | ||||||
|  |  | ||||||
|  |                 dashboardButton.classList.remove(...dashboardButton.classList); | ||||||
|  |                 dashboardButton.classList.add("btn"); | ||||||
|  |                 dashboardButton.classList.add(badgeClass(data.status.status)); | ||||||
|  |  | ||||||
|  |                 dashboardModalHeader.classList.remove(...dashboardModalHeader.classList); | ||||||
|  |                 dashboardModalHeader.classList.add("modal-header"); | ||||||
|  |                 headerClass(data.status.status).forEach(clz => dashboardModalHeader.classList.add(clz)); | ||||||
|  |  | ||||||
|  |                 dashboardName.textContent = data.repository; | ||||||
|  |                 dashboardArchitecture.textContent = data.architecture; | ||||||
|  |                 dashboardStatus.textContent = data.status.status; | ||||||
|  |                 dashboardStatusTimestamp.textContent = new Date(1000 * data.status.timestamp).toISOStringShort(); | ||||||
|  |  | ||||||
|  |                 if (dashboardPackagesStatusesChart) { | ||||||
|  |                     const labels = [ | ||||||
|  |                         "unknown", | ||||||
|  |                         "pending", | ||||||
|  |                         "building", | ||||||
|  |                         "failed", | ||||||
|  |                         "success", | ||||||
|  |                     ]; | ||||||
|  |                     dashboardPackagesStatusesChart.config.data = { | ||||||
|  |                         labels: labels, | ||||||
|  |                         datasets: [{ | ||||||
|  |                             label: "packages in status", | ||||||
|  |                             data: labels.map(label => data.packages[label]), | ||||||
|  |                             backgroundColor: [ | ||||||
|  |                                 "rgb(55, 58, 60)", | ||||||
|  |                                 "rgb(255, 117, 24)", | ||||||
|  |                                 "rgb(255, 117, 24)", | ||||||
|  |                                 "rgb(255, 0, 57)", | ||||||
|  |                                 "rgb(63, 182, 24)",  // copy-paste from current style | ||||||
|  |                             ], | ||||||
|  |                         }], | ||||||
|  |                     }; | ||||||
|  |                     dashboardPackagesStatusesChart.update(); | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 if (dashboardPackagesCountChart) { | ||||||
|  |                     dashboardPackagesCountChart.config.data = { | ||||||
|  |                         labels: ["packages"], | ||||||
|  |                         datasets: [ | ||||||
|  |                             { | ||||||
|  |                                 label: "archives", | ||||||
|  |                                 data: [data.stats.packages], | ||||||
|  |                             }, | ||||||
|  |                             { | ||||||
|  |                                 label: "bases", | ||||||
|  |                                 data: [data.stats.bases], | ||||||
|  |                             }, | ||||||
|  |                         ], | ||||||
|  |                     }; | ||||||
|  |                     dashboardPackagesCountChart.update(); | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 dashboardCanvas.hidden = data.status.total > 0; | ||||||
|  |             }, | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     ready(_ => { |     ready(_ => { | ||||||
|         dashboardPackagesStatusesChart = new Chart(dashboardPackagesStatusesChartCanvas, { |         dashboardPackagesStatusesChart = new Chart(dashboardPackagesStatusesChartCanvas, { | ||||||
|             type: "pie", |             type: "pie", | ||||||
|  | |||||||
| @ -24,6 +24,13 @@ | |||||||
|                             <datalist id="package-add-known-packages-dlist"></datalist> |                             <datalist id="package-add-known-packages-dlist"></datalist> | ||||||
|                         </div> |                         </div> | ||||||
|                     </div> |                     </div> | ||||||
|  |                     <div class="form-group row"> | ||||||
|  |                         <label class="col-3 col-form-label"></label> | ||||||
|  |                         <div class="col-9"> | ||||||
|  |                             <input id="package-add-refresh-input" type="checkbox" class="form-check-input" value="" checked> | ||||||
|  |                             <label for="package-add-refresh-input" class="form-check-label">update pacman databases</label> | ||||||
|  |                         </div> | ||||||
|  |                     </div> | ||||||
|                     <div class="form-group row"> |                     <div class="form-group row"> | ||||||
|                         <div class="col-12"> |                         <div class="col-12"> | ||||||
|                             <button id="package-add-variable-button" type="button" class="form-control btn btn-light rounded" onclick="packageAddVariableInputCreate()"><i class="bi bi-plus"></i> add environment variable </button> |                             <button id="package-add-variable-button" type="button" class="form-control btn btn-light rounded" onclick="packageAddVariableInputCreate()"><i class="bi bi-plus"></i> add environment variable </button> | ||||||
| @ -50,6 +57,8 @@ | |||||||
|  |  | ||||||
|     const packageAddVariablesDiv = document.getElementById("package-add-variables-div"); |     const packageAddVariablesDiv = document.getElementById("package-add-variables-div"); | ||||||
|  |  | ||||||
|  |     const packageAddRefreshInput = document.getElementById("package-add-refresh-input"); | ||||||
|  |  | ||||||
|     function packageAddVariableInputCreate() { |     function packageAddVariableInputCreate() { | ||||||
|         const variableInput = document.createElement("div"); |         const variableInput = document.createElement("div"); | ||||||
|         variableInput.classList.add("input-group"); |         variableInput.classList.add("input-group"); | ||||||
| @ -99,16 +108,18 @@ | |||||||
|         return {patches: patches}; |         return {patches: patches}; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     function packagesAdd(packages, patches, repository) { |     function packagesAdd(packages, patches, repository, data) { | ||||||
|         packages = packages ?? packageAddInput.value; |         packages = packages ?? packageAddInput.value; | ||||||
|         patches = patches ?? patchesParse(); |         patches = patches ?? patchesParse(); | ||||||
|         repository = repository ?? getRepositorySelector(packageAddRepositoryInput); |         repository = repository ?? getRepositorySelector(packageAddRepositoryInput); | ||||||
|  |         data = data ?? {refresh: packageAddRefreshInput.checked}; | ||||||
|  |  | ||||||
|         if (packages) { |         if (packages) { | ||||||
|             bootstrap.Modal.getOrCreateInstance(packageAddModal).hide(); |             bootstrap.Modal.getOrCreateInstance(packageAddModal).hide(); | ||||||
|             const onSuccess = update => `Packages ${update} have been added`; |             const onSuccess = update => `Packages ${update} have been added`; | ||||||
|             const onFailure = error => `Package addition failed: ${error}`; |             const onFailure = error => `Package addition failed: ${error}`; | ||||||
|             doPackageAction("/api/v1/service/add", [packages], repository, onSuccess, onFailure, patches); |             const parameters = Object.assign({}, data, patches); | ||||||
|  |             doPackageAction("/api/v1/service/add", [packages], repository, onSuccess, onFailure, parameters); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @ -137,8 +148,19 @@ | |||||||
|  |  | ||||||
|         packageAddInput.addEventListener("keyup", _ => { |         packageAddInput.addEventListener("keyup", _ => { | ||||||
|             clearTimeout(packageAddInput.requestTimeout); |             clearTimeout(packageAddInput.requestTimeout); | ||||||
|             packageAddInput.requestTimeout = setTimeout(_ => { |  | ||||||
|  |             // do not update datalist if search string didn't change yet | ||||||
|             const value = packageAddInput.value; |             const value = packageAddInput.value; | ||||||
|  |             const previousValue = packageAddInput.dataset.previousValue; | ||||||
|  |             if (value === previousValue) { | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // store current search string in attributes | ||||||
|  |             packageAddInput.dataset.previousValue = value; | ||||||
|  |  | ||||||
|  |             // perform data list update | ||||||
|  |             packageAddInput.requestTimeout = setTimeout(_ => { | ||||||
|  |  | ||||||
|                 if (value.length >= 3) { |                 if (value.length >= 3) { | ||||||
|                     makeRequest( |                     makeRequest( | ||||||
|  | |||||||
| @ -80,8 +80,7 @@ | |||||||
|                                data-classes="table table-hover" |                                data-classes="table table-hover" | ||||||
|                                data-sortable="true" |                                data-sortable="true" | ||||||
|                                data-sort-name="timestamp" |                                data-sort-name="timestamp" | ||||||
|                                data-sort-order="desc" |                                data-sort-order="desc"> | ||||||
|                                data-toggle="table"> |  | ||||||
|                             <thead class="table-primary"> |                             <thead class="table-primary"> | ||||||
|                                 <tr> |                                 <tr> | ||||||
|                                     <th data-align="right" data-field="timestamp">date</th> |                                     <th data-align="right" data-field="timestamp">date</th> | ||||||
| @ -95,10 +94,27 @@ | |||||||
|             </div> |             </div> | ||||||
|             <div class="modal-footer"> |             <div class="modal-footer"> | ||||||
|                 {% if not auth.enabled or auth.username is not none %} |                 {% if not auth.enabled or auth.username is not none %} | ||||||
|                     <button id="package-info-update-button" type="submit" class="btn btn-success" onclick="packageInfoUpdate()" data-bs-dismiss="modal"><i class="bi bi-play"></i><span class="d-none d-sm-inline"> update</span></button> |                     <input id="package-info-refresh-input" type="checkbox" class="form-check-input" value="" checked> | ||||||
|  |                     <label for="package-info-refresh-input" class="form-check-label">update pacman databases</label> | ||||||
|  |  | ||||||
|  |                     <button id="package-info-update-button" type="submit" class="btn btn-success" onclick="packageInfoUpdate()"><i class="bi bi-play"></i><span class="d-none d-sm-inline"> update</span></button> | ||||||
|                     <button id="package-info-remove-button" type="submit" class="btn btn-danger" onclick="packageInfoRemove()" data-bs-dismiss="modal"><i class="bi bi-trash"></i><span class="d-none d-sm-inline"> remove</span></button> |                     <button id="package-info-remove-button" type="submit" class="btn btn-danger" onclick="packageInfoRemove()" data-bs-dismiss="modal"><i class="bi bi-trash"></i><span class="d-none d-sm-inline"> remove</span></button> | ||||||
|                 {% endif %} |                 {% endif %} | ||||||
|  |                 {% if autorefresh_intervals %} | ||||||
|                     <button type="button" class="btn btn-secondary" onclick="showPackageInfo()"><i class="bi bi-arrow-clockwise"></i><span class="d-none d-sm-inline"> reload</span></button> |                     <button type="button" class="btn btn-secondary" onclick="showPackageInfo()"><i class="bi bi-arrow-clockwise"></i><span class="d-none d-sm-inline"> reload</span></button> | ||||||
|  |                     <div class="btn-group dropup"> | ||||||
|  |                         <input id="package-info-autoreload-button" type="checkbox" class="btn-check" autocomplete="off" onclick="togglePackageInfoAutoReload()" checked> | ||||||
|  |                         <label for="package-info-autoreload-button" class="btn btn-outline-secondary" title="toggle auto reload"><i class="bi bi-clock"></i></label> | ||||||
|  |                         <button type="button" class="btn btn-outline-secondary dropdown-toggle dropdown-toggle-split" data-bs-toggle="dropdown" aria-expanded="false"> | ||||||
|  |                             <span class="visually-hidden">select interval</span> | ||||||
|  |                         </button> | ||||||
|  |                         <ul id="package-info-autoreload-input" class="dropdown-menu"> | ||||||
|  |                             {% for interval in autorefresh_intervals %} | ||||||
|  |                                 <li><a class="dropdown-item {{ "active" if interval.is_active }}" onclick="togglePackageInfoAutoReload({{ interval.interval }})" data-interval="{{ interval.interval }}">{{ interval.text }}</a></li> | ||||||
|  |                             {% endfor %} | ||||||
|  |                         </ul> | ||||||
|  |                     </div> | ||||||
|  |                 {% endif %} | ||||||
|                 <button type="button" class="btn btn-primary" data-bs-dismiss="modal"><i class="bi bi-x"></i><span class="d-none d-sm-inline"> close</span></button> |                 <button type="button" class="btn btn-primary" data-bs-dismiss="modal"><i class="bi bi-x"></i><span class="d-none d-sm-inline"> close</span></button> | ||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
| @ -135,6 +151,12 @@ | |||||||
|     const packageInfoVariablesBlock = document.getElementById("package-info-variables-block"); |     const packageInfoVariablesBlock = document.getElementById("package-info-variables-block"); | ||||||
|     const packageInfoVariablesDiv = document.getElementById("package-info-variables-div"); |     const packageInfoVariablesDiv = document.getElementById("package-info-variables-div"); | ||||||
|  |  | ||||||
|  |     const packageInfoRefreshInput = document.getElementById("package-info-refresh-input"); | ||||||
|  |  | ||||||
|  |     const packageInfoAutoReloadButton = document.getElementById("package-info-autoreload-button"); | ||||||
|  |     const packageInfoAutoReloadInput = document.getElementById("package-info-autoreload-input"); | ||||||
|  |     let packageInfoAutoReloadTask = null; | ||||||
|  |  | ||||||
|     function clearChart() { |     function clearChart() { | ||||||
|         packageInfoEventsUpdateChartCanvas.hidden = true; |         packageInfoEventsUpdateChartCanvas.hidden = true; | ||||||
|         if (packageInfoEventsUpdateChart) { |         if (packageInfoEventsUpdateChart) { | ||||||
| @ -143,6 +165,13 @@ | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     function convertLogs(data, filter) { | ||||||
|  |         return data | ||||||
|  |             .filter((filter || Boolean)) | ||||||
|  |             .map(log_record => `[${new Date(1000 * log_record.created).toISOString()}] ${log_record.message}`) | ||||||
|  |             .join("\n"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     async function copyChanges() { |     async function copyChanges() { | ||||||
|         const changes = packageInfoChangesInput.textContent; |         const changes = packageInfoChangesInput.textContent; | ||||||
|         await copyToClipboard(changes, packageInfoChangesCopyButton); |         await copyToClipboard(changes, packageInfoChangesCopyButton); | ||||||
| @ -286,6 +315,69 @@ | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     function loadLogs(packageBase, onFailure) { |     function loadLogs(packageBase, onFailure) { | ||||||
|  |         const sortFn = (left, right) => left.process_id.localeCompare(right.process_id) || left.version.localeCompare(right.version); | ||||||
|  |         const compareFn = (left, right) => left.process_id === right.process_id && left.version === right.version; | ||||||
|  |  | ||||||
|  |         makeRequest( | ||||||
|  |             `/api/v2/packages/${packageBase}/logs`, | ||||||
|  |             { | ||||||
|  |                 query: { | ||||||
|  |                     architecture: repository.architecture, | ||||||
|  |                     head: true, | ||||||
|  |                     repository: repository.repository, | ||||||
|  |                 }, | ||||||
|  |                 convert: response => response.json(), | ||||||
|  |             }, | ||||||
|  |             data => { | ||||||
|  |                 const currentVersions = Array.from(packageInfoLogsVersions.children) | ||||||
|  |                     .map(el => { | ||||||
|  |                         return { | ||||||
|  |                             process_id: el.dataset.processId, | ||||||
|  |                             version: el.dataset.version, | ||||||
|  |                         }; | ||||||
|  |                     }) | ||||||
|  |                     .sort(sortFn); | ||||||
|  |                 const newVersions = data | ||||||
|  |                     .map(el => { | ||||||
|  |                         return { | ||||||
|  |                             process_id: el.process_id, | ||||||
|  |                             version: el.version, | ||||||
|  |                         }; | ||||||
|  |                     }) | ||||||
|  |                     .sort(sortFn); | ||||||
|  |  | ||||||
|  |                 if (currentVersions.equals(newVersions, compareFn)) | ||||||
|  |                     loadLogsActive(packageBase); | ||||||
|  |                 else | ||||||
|  |                     loadLogsAll(packageBase, onFailure); | ||||||
|  |             }, | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function loadLogsActive(packageBase) { | ||||||
|  |         const activeLogSelector = packageInfoLogsVersions.querySelector(".active"); | ||||||
|  |  | ||||||
|  |         if (activeLogSelector) { | ||||||
|  |             makeRequest( | ||||||
|  |                 `/api/v2/packages/${packageBase}/logs`, | ||||||
|  |                 { | ||||||
|  |                     query: { | ||||||
|  |                         architecture: repository.architecture, | ||||||
|  |                         repository: repository.repository, | ||||||
|  |                         version: activeLogSelector.dataset.version, | ||||||
|  |                         process_id: activeLogSelector.dataset.processId, | ||||||
|  |                     }, | ||||||
|  |                     convert: response => response.json(), | ||||||
|  |                 }, | ||||||
|  |                 data => { | ||||||
|  |                     activeLogSelector.dataset.logs = convertLogs(data); | ||||||
|  |                     activeLogSelector.click(); | ||||||
|  |                 }, | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function loadLogsAll(packageBase, onFailure) { | ||||||
|         makeRequest( |         makeRequest( | ||||||
|             `/api/v2/packages/${packageBase}/logs`, |             `/api/v2/packages/${packageBase}/logs`, | ||||||
|             { |             { | ||||||
| @ -314,15 +406,19 @@ | |||||||
|                         const link = document.createElement("a"); |                         const link = document.createElement("a"); | ||||||
|                         link.classList.add("dropdown-item"); |                         link.classList.add("dropdown-item"); | ||||||
|  |  | ||||||
|  |                         link.dataset.version = version.version; | ||||||
|  |                         link.dataset.processId = version.process_id; | ||||||
|  |                         link.dataset.logs = convertLogs(data, log_record => log_record.version === version.version && log_record.process_id === version.process_id); | ||||||
|  |  | ||||||
|                         link.textContent = new Date(1000 * version.created).toISOStringShort(); |                         link.textContent = new Date(1000 * version.created).toISOStringShort(); | ||||||
|                         link.href = "#"; |                         link.href = "#"; | ||||||
|                         link.onclick = _ => { |                         link.onclick = _ => { | ||||||
|                             const logs = data |                             // check if we are at the bottom of the code block | ||||||
|                                 .filter(log_record => log_record.version === version.version && log_record.process_id === version.process_id) |                             const isScrolledToBottom = packageInfoLogsInput.scrollTop + packageInfoLogsInput.clientHeight >= packageInfoLogsInput.scrollHeight; | ||||||
|                                 .map(log_record => `[${new Date(1000 * log_record.created).toISOString()}] ${log_record.message}`); |                             packageInfoLogsInput.textContent = link.dataset.logs; | ||||||
|  |  | ||||||
|                             packageInfoLogsInput.textContent = logs.join("\n"); |  | ||||||
|                             highlight(packageInfoLogsInput); |                             highlight(packageInfoLogsInput); | ||||||
|  |                             if (isScrolledToBottom) | ||||||
|  |                                 packageInfoLogsInput.scrollTop = packageInfoLogsInput.scrollHeight; // scroll to the new end | ||||||
|  |  | ||||||
|                             Array.from(packageInfoLogsVersions.children).forEach(el => el.classList.remove("active")); |                             Array.from(packageInfoLogsVersions.children).forEach(el => el.classList.remove("active")); | ||||||
|                             link.classList.add("active"); |                             link.classList.add("active"); | ||||||
| @ -398,23 +494,23 @@ | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     function packageInfoRemove() { |     function packageInfoRemove() { | ||||||
|         const packageBase = packageInfoModal.package; |         const packageBase = packageInfoModal.dataset.package; | ||||||
|         packagesRemove([packageBase]); |         packagesRemove([packageBase]); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     function packageInfoUpdate() { |     function packageInfoUpdate() { | ||||||
|         const packageBase = packageInfoModal.package; |         const packageBase = packageInfoModal.dataset.package; | ||||||
|         packagesAdd(packageBase, [], repository); |         packagesAdd(packageBase, [], repository, {refresh: packageInfoRefreshInput.checked}); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     function showPackageInfo(packageBase) { |     function showPackageInfo(packageBase) { | ||||||
|         const isPackageBaseSet = packageBase !== undefined; |         const isPackageBaseSet = packageBase !== undefined; | ||||||
|         if (isPackageBaseSet) { |         if (isPackageBaseSet) { | ||||||
|             // set package base as currently used |             // set package base as currently used | ||||||
|             packageInfoModal.package = packageBase; |             packageInfoModal.dataset.package = packageBase; | ||||||
|         } else { |         } else { | ||||||
|             // read package base from the current window attribute |             // read package base from the current window attribute | ||||||
|             packageBase = packageInfoModal.package; |             packageBase = packageInfoModal.dataset.package; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         const onFailure = error => { |         const onFailure = error => { | ||||||
| @ -433,10 +529,27 @@ | |||||||
|  |  | ||||||
|         if (isPackageBaseSet) { |         if (isPackageBaseSet) { | ||||||
|             bootstrap.Modal.getOrCreateInstance(packageInfoModal).show(); |             bootstrap.Modal.getOrCreateInstance(packageInfoModal).show(); | ||||||
|  |             {% if autorefresh_intervals %} | ||||||
|  |                 togglePackageInfoAutoReload(); | ||||||
|  |             {% endif %} | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     function togglePackageInfoAutoReload(interval) { | ||||||
|  |         clearInterval(packageInfoAutoReloadTask); | ||||||
|  |         packageInfoAutoReloadTask = toggleAutoReload(packageInfoAutoReloadButton, interval, packageInfoAutoReloadInput, _ => { | ||||||
|  |             if (!hasActiveSelection()) { | ||||||
|  |                 const packageBase = packageInfoModal.dataset.package; | ||||||
|  |                 // we only poll status and logs here | ||||||
|  |                 loadPackage(packageBase); | ||||||
|  |                 loadLogs(packageBase); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     ready(_ => { |     ready(_ => { | ||||||
|  |         packageInfoEventsTable.bootstrapTable({}); | ||||||
|  |  | ||||||
|         packageInfoEventsUpdateChart = new Chart(packageInfoEventsUpdateChartCanvas, { |         packageInfoEventsUpdateChart = new Chart(packageInfoEventsUpdateChartCanvas, { | ||||||
|             type: "line", |             type: "line", | ||||||
|             data: {}, |             data: {}, | ||||||
| @ -463,6 +576,11 @@ | |||||||
|             packageInfoChangesInput.textContent = ""; |             packageInfoChangesInput.textContent = ""; | ||||||
|             packageInfoEventsTable.bootstrapTable("load", []); |             packageInfoEventsTable.bootstrapTable("load", []); | ||||||
|             clearChart(); |             clearChart(); | ||||||
|  |  | ||||||
|  |             clearInterval(packageInfoAutoReloadTask); | ||||||
|  |             packageInfoAutoReloadTask = null; // not really required (?) but lets clear everything | ||||||
|         }); |         }); | ||||||
|  |  | ||||||
|  |         restoreAutoReloadSettings(packageInfoAutoReloadButton, packageInfoAutoReloadInput); | ||||||
|     }); |     }); | ||||||
| </script> | </script> | ||||||
|  | |||||||
| @ -10,6 +10,10 @@ | |||||||
|     const dashboardButton = document.getElementById("dashboard-button"); |     const dashboardButton = document.getElementById("dashboard-button"); | ||||||
|     const versionBadge = document.getElementById("badge-version"); |     const versionBadge = document.getElementById("badge-version"); | ||||||
|  |  | ||||||
|  |     const tableAutoReloadButton = document.getElementById("table-autoreload-button"); | ||||||
|  |     const tableAutoReloadInput = document.getElementById("table-autoreload-input"); | ||||||
|  |     let tableAutoReloadTask = null; | ||||||
|  |  | ||||||
|     function doPackageAction(uri, packages, repository, successText, failureText, data) { |     function doPackageAction(uri, packages, repository, successText, failureText, data) { | ||||||
|         makeRequest( |         makeRequest( | ||||||
|             uri, |             uri, | ||||||
| @ -55,6 +59,41 @@ | |||||||
|         return table.bootstrapTable("getSelections").map(row => row.id); |         return table.bootstrapTable("getSelections").map(row => row.id); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     function packagesLoad(onFailure) { | ||||||
|  |         makeRequest( | ||||||
|  |             "/api/v1/packages", | ||||||
|  |             { | ||||||
|  |                 query: { | ||||||
|  |                     architecture: repository.architecture, | ||||||
|  |                     repository: repository.repository, | ||||||
|  |                 }, | ||||||
|  |                 convert: response => response.json(), | ||||||
|  |             }, | ||||||
|  |             data => { | ||||||
|  |                 const payload = data | ||||||
|  |                     .map(description => { | ||||||
|  |                         const package_base = description.package.base; | ||||||
|  |                         const web_url = description.package.remote.web_url; | ||||||
|  |                         return { | ||||||
|  |                             id: package_base, | ||||||
|  |                             base: web_url ? safeLink(web_url, package_base, package_base).outerHTML : safe(package_base), | ||||||
|  |                             version: safe(description.package.version), | ||||||
|  |                             packager: description.package.packager ? safe(description.package.packager) : "", | ||||||
|  |                             packages: listToTable(Object.keys(description.package.packages)), | ||||||
|  |                             groups: listToTable(extractListProperties(description.package, "groups")), | ||||||
|  |                             licenses: listToTable(extractListProperties(description.package, "licenses")), | ||||||
|  |                             timestamp: new Date(1000 * description.status.timestamp).toISOStringShort(), | ||||||
|  |                             status: description.status.status, | ||||||
|  |                         }; | ||||||
|  |                     }); | ||||||
|  |  | ||||||
|  |                 updateTable(table, payload); | ||||||
|  |                 table.bootstrapTable("hideLoading"); | ||||||
|  |             }, | ||||||
|  |             onFailure, | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     function packagesRemove(packages) { |     function packagesRemove(packages) { | ||||||
|         packages = packages ?? getSelection(); |         packages = packages ?? getSelection(); | ||||||
|         const onSuccess = update => `Packages ${update} have been removed`; |         const onSuccess = update => `Packages ${update} have been removed`; | ||||||
| @ -73,48 +112,22 @@ | |||||||
|         doPackageAction(url, currentSelection, repository, onSuccess, onFailure); |         doPackageAction(url, currentSelection, repository, onSuccess, onFailure); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     function refreshDatabases() { | ||||||
|  |         const onSuccess = _ => "Pacman database update has been requested"; | ||||||
|  |         const onFailure = error => `Could not update pacman databases: ${error}`; | ||||||
|  |         const parameters = { | ||||||
|  |             refresh: true, | ||||||
|  |             aur: false, | ||||||
|  |             local: false, | ||||||
|  |             manual: false, | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         doPackageAction("/api/v1/service/update", [], repository, onSuccess, onFailure, parameters); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     function reload() { |     function reload() { | ||||||
|         table.bootstrapTable("showLoading"); |         table.bootstrapTable("showLoading"); | ||||||
|  |         const onFailure = error => { | ||||||
|         const badgeClass = status => { |  | ||||||
|             if (status === "pending") return "btn-outline-warning"; |  | ||||||
|             if (status === "building") return "btn-outline-warning"; |  | ||||||
|             if (status === "failed") return "btn-outline-danger"; |  | ||||||
|             if (status === "success") return "btn-outline-success"; |  | ||||||
|             return "btn-outline-secondary"; |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         makeRequest( |  | ||||||
|             "/api/v1/packages", |  | ||||||
|             { |  | ||||||
|                 query: { |  | ||||||
|                     architecture: repository.architecture, |  | ||||||
|                     repository: repository.repository, |  | ||||||
|                 }, |  | ||||||
|                 convert: response => response.json(), |  | ||||||
|             }, |  | ||||||
|             data => { |  | ||||||
|                 const payload = data.map(description => { |  | ||||||
|                     const package_base = description.package.base; |  | ||||||
|                     const web_url = description.package.remote.web_url; |  | ||||||
|                     return { |  | ||||||
|                         id: package_base, |  | ||||||
|                         base: web_url ? safeLink(web_url, package_base, package_base).outerHTML : safe(package_base), |  | ||||||
|                         version: safe(description.package.version), |  | ||||||
|                         packager: description.package.packager ? safe(description.package.packager) : "", |  | ||||||
|                         packages: listToTable(Object.keys(description.package.packages)), |  | ||||||
|                         groups: listToTable(extractListProperties(description.package, "groups")), |  | ||||||
|                         licenses: listToTable(extractListProperties(description.package, "licenses")), |  | ||||||
|                         timestamp: new Date(1000 * description.status.timestamp).toISOStringShort(), |  | ||||||
|                         status: description.status.status, |  | ||||||
|                     }; |  | ||||||
|                 }); |  | ||||||
|  |  | ||||||
|                 table.bootstrapTable("load", payload); |  | ||||||
|                 table.bootstrapTable("uncheckAll"); |  | ||||||
|                 table.bootstrapTable("hideLoading"); |  | ||||||
|             }, |  | ||||||
|             error => { |  | ||||||
|             if ((error.status === 401) || (error.status === 403)) { |             if ((error.status === 401) || (error.status === 403)) { | ||||||
|                 // authorization error |                 // authorization error | ||||||
|                 const text = "In order to see statuses you must login first."; |                 const text = "In order to see statuses you must login first."; | ||||||
| @ -126,79 +139,10 @@ | |||||||
|                 const message = details => `Could not load list of packages: ${details}`; |                 const message = details => `Could not load list of packages: ${details}`; | ||||||
|                 showFailure("Load failure", message, error); |                 showFailure("Load failure", message, error); | ||||||
|             } |             } | ||||||
|             }, |  | ||||||
|         ); |  | ||||||
|  |  | ||||||
|         makeRequest( |  | ||||||
|             "/api/v1/status", |  | ||||||
|             { |  | ||||||
|                 query: { |  | ||||||
|                     architecture: repository.architecture, |  | ||||||
|                     repository: repository.repository, |  | ||||||
|                 }, |  | ||||||
|                 convert: response => response.json(), |  | ||||||
|             }, |  | ||||||
|             data => { |  | ||||||
|                 versionBadge.innerHTML = `<i class="bi bi-github"></i> ahriman ${safe(data.version)}`; |  | ||||||
|  |  | ||||||
|                 dashboardButton.classList.remove(...dashboardButton.classList); |  | ||||||
|                 dashboardButton.classList.add("btn"); |  | ||||||
|                 dashboardButton.classList.add(badgeClass(data.status.status)); |  | ||||||
|  |  | ||||||
|                 dashboardModalHeader.classList.remove(...dashboardModalHeader.classList); |  | ||||||
|                 dashboardModalHeader.classList.add("modal-header"); |  | ||||||
|                 headerClass(data.status.status).forEach(clz => dashboardModalHeader.classList.add(clz)); |  | ||||||
|  |  | ||||||
|                 dashboardName.textContent = data.repository; |  | ||||||
|                 dashboardArchitecture.textContent = data.architecture; |  | ||||||
|                 dashboardStatus.textContent = data.status.status; |  | ||||||
|                 dashboardStatusTimestamp.textContent = new Date(1000 * data.status.timestamp).toISOStringShort(); |  | ||||||
|  |  | ||||||
|                 if (dashboardPackagesStatusesChart) { |  | ||||||
|                     const labels = [ |  | ||||||
|                         "unknown", |  | ||||||
|                         "pending", |  | ||||||
|                         "building", |  | ||||||
|                         "failed", |  | ||||||
|                         "success", |  | ||||||
|                     ]; |  | ||||||
|                     dashboardPackagesStatusesChart.config.data = { |  | ||||||
|                         labels: labels, |  | ||||||
|                         datasets: [{ |  | ||||||
|                             label: "packages in status", |  | ||||||
|                             data: labels.map(label => data.packages[label]), |  | ||||||
|                             backgroundColor: [ |  | ||||||
|                                 "rgb(55, 58, 60)", |  | ||||||
|                                 "rgb(255, 117, 24)", |  | ||||||
|                                 "rgb(255, 117, 24)", |  | ||||||
|                                 "rgb(255, 0, 57)", |  | ||||||
|                                 "rgb(63, 182, 24)",  // copy-paste from current style |  | ||||||
|                             ], |  | ||||||
|                         }], |  | ||||||
|         }; |         }; | ||||||
|                     dashboardPackagesStatusesChart.update(); |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 if (dashboardPackagesCountChart) { |         packagesLoad(onFailure); | ||||||
|                     dashboardPackagesCountChart.config.data = { |         statusLoad(); | ||||||
|                         labels: ["packages"], |  | ||||||
|                         datasets: [ |  | ||||||
|                             { |  | ||||||
|                                 label: "archives", |  | ||||||
|                                 data: [data.stats.packages], |  | ||||||
|                             }, |  | ||||||
|                             { |  | ||||||
|                                 label: "bases", |  | ||||||
|                                 data: [data.stats.bases], |  | ||||||
|                             }, |  | ||||||
|                         ], |  | ||||||
|                     }; |  | ||||||
|                     dashboardPackagesCountChart.update(); |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 dashboardCanvas.hidden = data.status.total > 0; |  | ||||||
|             }, |  | ||||||
|         ); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     function selectRepository() { |     function selectRepository() { | ||||||
| @ -217,7 +161,24 @@ | |||||||
|         return {classes: cellClass(value)}; |         return {classes: cellClass(value)}; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     function toggleTableAutoReload(interval) { | ||||||
|  |         clearInterval(tableAutoReloadTask); | ||||||
|  |         tableAutoReloadTask = toggleAutoReload(tableAutoReloadButton, interval, tableAutoReloadInput, _ => { | ||||||
|  |             if (!hasActiveModal() && | ||||||
|  |                 !hasActiveDropdown()) { | ||||||
|  |                 packagesLoad(); | ||||||
|  |                 statusLoad(); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     ready(_ => { |     ready(_ => { | ||||||
|  |         const onCheckFunction = function () { | ||||||
|  |             if (packageRemoveButton) { | ||||||
|  |                 packageRemoveButton.disabled = !getSelection().length; | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|  |  | ||||||
|         document.querySelectorAll("#repositories a").forEach(element => { |         document.querySelectorAll("#repositories a").forEach(element => { | ||||||
|             element.onclick = _ => { |             element.onclick = _ => { | ||||||
|                 repository = { |                 repository = { | ||||||
| @ -232,18 +193,16 @@ | |||||||
|             }; |             }; | ||||||
|         }); |         }); | ||||||
|  |  | ||||||
|         table.on("check.bs.table uncheck.bs.table check-all.bs.table uncheck-all.bs.table", _ => { |         table.bootstrapTable({ | ||||||
|             if (packageRemoveButton) { |             onCheck: onCheckFunction, | ||||||
|                 packageRemoveButton.disabled = !table.bootstrapTable("getSelections").length; |             onCheckAll: onCheckFunction, | ||||||
|             } |             onClickRow: (data, row, cell) => { | ||||||
|         }); |  | ||||||
|         table.on("click-row.bs.table", (self, data, row, cell) => { |  | ||||||
|                 if (0 === cell || "base" === cell) { |                 if (0 === cell || "base" === cell) { | ||||||
|                     const method = data[0] === true ? "uncheckBy" : "checkBy"; // fck javascript |                     const method = data[0] === true ? "uncheckBy" : "checkBy"; // fck javascript | ||||||
|                     table.bootstrapTable(method, {field: "id", values: [data.id]}); |                     table.bootstrapTable(method, {field: "id", values: [data.id]}); | ||||||
|                 } else showPackageInfo(data.id); |                 } else showPackageInfo(data.id); | ||||||
|         }); |             }, | ||||||
|         table.on("created-controls.bs.table", _ => { |             onCreatedControls: _ => { | ||||||
|                 new easepick.create({ |                 new easepick.create({ | ||||||
|                     element: document.querySelector(".bootstrap-table-filter-control-timestamp"), |                     element: document.querySelector(".bootstrap-table-filter-control-timestamp"), | ||||||
|                     css: [ |                     css: [ | ||||||
| @ -273,8 +232,16 @@ | |||||||
|                         }; |                         }; | ||||||
|                     }, |                     }, | ||||||
|                 }); |                 }); | ||||||
|  |             }, | ||||||
|  |             onUncheck: onCheckFunction, | ||||||
|  |             onUncheckAll: onCheckFunction, | ||||||
|         }); |         }); | ||||||
|  |  | ||||||
|  |         restoreAutoReloadSettings(tableAutoReloadButton, tableAutoReloadInput); | ||||||
|  |  | ||||||
|         selectRepository(); |         selectRepository(); | ||||||
|  |         {% if autorefresh_intervals %} | ||||||
|  |             toggleTableAutoReload(); | ||||||
|  |         {% endif %} | ||||||
|     }); |     }); | ||||||
| </script> | </script> | ||||||
|  | |||||||
| @ -53,8 +53,7 @@ SigLevel = Database{% if has_repo_signed %}Required{% else %}Never{% endif %} Pa | |||||||
|                    data-show-search-clear-button="true" |                    data-show-search-clear-button="true" | ||||||
|                    data-sortable="true" |                    data-sortable="true" | ||||||
|                    data-sort-name="base" |                    data-sort-name="base" | ||||||
|                    data-sort-order="asc" |                    data-sort-order="asc"> | ||||||
|                    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" data-field="name" data-filter-control="input" data-filter-control-placeholder="(any package)">package</th> | ||||||
| @ -128,7 +127,8 @@ SigLevel = Database{% if has_repo_signed %}Required{% else %}Never{% endif %} Pa | |||||||
|             } |             } | ||||||
|  |  | ||||||
|             ready(_ => { |             ready(_ => { | ||||||
|                 table.on("created-controls.bs.table", _ => { |                 table.bootstrapTable({ | ||||||
|  |                     onCreatedControls: _ => { | ||||||
|                         new easepick.create({ |                         new easepick.create({ | ||||||
|                             element: document.querySelector(".bootstrap-table-filter-control-timestamp"), |                             element: document.querySelector(".bootstrap-table-filter-control-timestamp"), | ||||||
|                             css: [ |                             css: [ | ||||||
| @ -158,6 +158,7 @@ SigLevel = Database{% if has_repo_signed %}Required{% else %}Never{% endif %} Pa | |||||||
|                                 }; |                                 }; | ||||||
|                             }, |                             }, | ||||||
|                         }); |                         }); | ||||||
|  |                     }, | ||||||
|                 }); |                 }); | ||||||
|             }); |             }); | ||||||
|         </script> |         </script> | ||||||
|  | |||||||
| @ -1,23 +1,24 @@ | |||||||
| <script src="https://cdn.jsdelivr.net/npm/jquery@3.7.1/dist/jquery.min.js" crossorigin="anonymous" type="application/javascript"></script> | <script src="https://cdn.jsdelivr.net/npm/jquery@3.7.1/dist/jquery.min.js" crossorigin="anonymous" type="application/javascript"></script> | ||||||
| <script src="https://cdn.jsdelivr.net/npm/js-md5@0.8.3/src/md5.min.js" crossorigin="anonymous" type="application/javascript"></script> | <script src="https://cdn.jsdelivr.net/npm/js-md5@0.8.3/src/md5.min.js" crossorigin="anonymous" type="application/javascript"></script> | ||||||
|  |  | ||||||
| <script src="https://cdn.jsdelivr.net/npm/tableexport.jquery.plugin@1.30.0/tableExport.min.js" crossorigin="anonymous" type="application/javascript"></script> | <script src="https://cdn.jsdelivr.net/npm/tableexport.jquery.plugin@1.33.0/tableExport.min.js" crossorigin="anonymous" type="application/javascript"></script> | ||||||
|  |  | ||||||
| <script src="https://cdn.jsdelivr.net/npm/jquery-resizable-columns@0.2.3/dist/jquery.resizableColumns.min.js" crossorigin="anonymous" type="application/javascript"></script> | <script src="https://cdn.jsdelivr.net/npm/jquery-resizable-columns@0.2.3/dist/jquery.resizableColumns.min.js" crossorigin="anonymous" type="application/javascript"></script> | ||||||
|  |  | ||||||
| <script src="https://cdn.jsdelivr.net/npm/@popperjs/core@2.11.8/dist/umd/popper.min.js" crossorigin="anonymous" type="application/javascript"></script> | <script src="https://cdn.jsdelivr.net/npm/@popperjs/core@2.11.8/dist/umd/popper.min.js" crossorigin="anonymous" type="application/javascript"></script> | ||||||
| <script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.3/dist/js/bootstrap.min.js" crossorigin="anonymous" type="application/javascript"></script> | <script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.7/dist/js/bootstrap.min.js" crossorigin="anonymous" type="application/javascript"></script> | ||||||
| <script src="https://cdn.jsdelivr.net/npm/bootstrap-table@1.23.2/dist/bootstrap-table.min.js" crossorigin="anonymous" type="application/javascript"></script> | <script src="https://cdn.jsdelivr.net/npm/bootstrap-table@1.24.1/dist/bootstrap-table.min.js" crossorigin="anonymous" type="application/javascript"></script> | ||||||
|  |  | ||||||
| <script src="https://cdn.jsdelivr.net/npm/bootstrap-table@1.23.2/dist/extensions/export/bootstrap-table-export.min.js" crossorigin="anonymous" type="application/javascript"></script> | <script src="https://cdn.jsdelivr.net/npm/bootstrap-table@1.24.1/dist/extensions/export/bootstrap-table-export.min.js" crossorigin="anonymous" type="application/javascript"></script> | ||||||
| <script src="https://cdn.jsdelivr.net/npm/bootstrap-table@1.23.2/dist/extensions/resizable/bootstrap-table-resizable.js" crossorigin="anonymous" type="application/javascript"></script> | <script src="https://cdn.jsdelivr.net/npm/bootstrap-table@1.24.1/dist/extensions/resizable/bootstrap-table-resizable.js" crossorigin="anonymous" type="application/javascript"></script> | ||||||
| <script src="https://cdn.jsdelivr.net/npm/bootstrap-table@1.23.2/dist/extensions/filter-control/bootstrap-table-filter-control.js" crossorigin="anonymous" type="application/javascript"></script> | <script src="https://cdn.jsdelivr.net/npm/bootstrap-table@1.24.1/dist/extensions/filter-control/bootstrap-table-filter-control.js" crossorigin="anonymous" type="application/javascript"></script> | ||||||
|  | <script src="https://cdn.jsdelivr.net/npm/bootstrap-table@1.24.1/dist/extensions/cookie/bootstrap-table-cookie.min.js" crossorigin="anonymous" type="application/javascript"></script> | ||||||
|  |  | ||||||
| <script src="https://cdn.jsdelivr.net/npm/@easepick/bundle@1.2.1/dist/index.umd.min.js" crossorigin="anonymous" type="application/javascript"></script> | <script src="https://cdn.jsdelivr.net/npm/@easepick/bundle@1.2.1/dist/index.umd.min.js" crossorigin="anonymous" type="application/javascript"></script> | ||||||
|  |  | ||||||
| <script src="https://cdn.jsdelivr.net/gh/highlightjs/cdn-release@11.10.0/build/highlight.min.js" crossorigin="anonymous" type="application/javascript"></script> | <script src="https://cdn.jsdelivr.net/gh/highlightjs/cdn-release@11.11.1/build/highlight.min.js" crossorigin="anonymous" type="application/javascript"></script> | ||||||
|  |  | ||||||
| <script src="https://cdn.jsdelivr.net/npm/chart.js@4.4.4/dist/chart.umd.min.js" crossorigin="anonymous" type="application/javascript"></script> | <script src="https://cdn.jsdelivr.net/npm/chart.js@4.5.0/dist/chart.umd.min.js" crossorigin="anonymous" type="application/javascript"></script> | ||||||
|  |  | ||||||
| <script> | <script> | ||||||
|     async function copyToClipboard(text, button) { |     async function copyToClipboard(text, button) { | ||||||
| @ -58,6 +59,20 @@ | |||||||
|         return value.includes(dataList[index].toLowerCase()); |         return value.includes(dataList[index].toLowerCase()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     function hasActiveSelection() { | ||||||
|  |         return !document.getSelection().isCollapsed; // not sure if it is a valid way, but I guess so | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function hasActiveDropdown() { | ||||||
|  |         return Array.from(document.querySelectorAll(".dropdown-menu")) | ||||||
|  |             .some(el => el.classList.contains("show")); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function hasActiveModal() { | ||||||
|  |         return Array.from(document.querySelectorAll(".modal")) | ||||||
|  |             .some(el => el.classList.contains("show")); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     function headerClass(status) { |     function headerClass(status) { | ||||||
|         if (status === "pending") return ["bg-warning"]; |         if (status === "pending") return ["bg-warning"]; | ||||||
|         if (status === "building") return ["bg-warning"]; |         if (status === "building") return ["bg-warning"]; | ||||||
| @ -106,6 +121,12 @@ | |||||||
|             .catch(error => onFailure && onFailure(error)); |             .catch(error => onFailure && onFailure(error)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     function readOptional(extractor, callback) { | ||||||
|  |         for (let value = extractor(); !!value; value = null) { | ||||||
|  |             callback(value); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     function ready(fn) { |     function ready(fn) { | ||||||
|         if (document.readyState === "complete" || document.readyState === "interactive") { |         if (document.readyState === "complete" || document.readyState === "interactive") { | ||||||
|             setTimeout(fn, 1); |             setTimeout(fn, 1); | ||||||
| @ -114,6 +135,11 @@ | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     function restoreAutoReloadSettings(toggle, intervalSelector) { | ||||||
|  |         readOptional(() => localStorage.getItem(`ahriman-${toggle.id}-refresh-enabled`), value => toggle.checked = value === "true"); | ||||||
|  |         readOptional(() => localStorage.getItem(`ahriman-${toggle.id}-refresh-interval`), value => toggleActiveElement(intervalSelector, "interval", value)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     function safe(string) { |     function safe(string) { | ||||||
|         return String(string) |         return String(string) | ||||||
|             .replace(/&/g, "&") |             .replace(/&/g, "&") | ||||||
| @ -133,7 +159,86 @@ | |||||||
|         return element; |         return element; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     Date.prototype.toISOStringShort = function() { |     function toggleActiveElement(selector, dataType, value) { | ||||||
|  |         const targetElement = selector.querySelector(`a[data-${dataType}="${value}"]`); | ||||||
|  |         if (targetElement?.classList?.contains("active")) { | ||||||
|  |             return; // element is already active, skip processing | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         Array.from(selector.children).forEach(il => { | ||||||
|  |             Array.from(il.children).forEach(el => el.classList.remove("active")); | ||||||
|  |         }); | ||||||
|  |         targetElement?.classList?.add("active"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function toggleAutoReload(toggle, interval, intervalSelector, callback) { | ||||||
|  |         if (interval) { | ||||||
|  |             toggle.checked = true; // toggle reload | ||||||
|  |         } else { | ||||||
|  |             interval = intervalSelector.querySelector(".active")?.dataset?.interval; // find active element | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         let intervalId = null; | ||||||
|  |         if (interval) { | ||||||
|  |             if (toggle.checked) { | ||||||
|  |                 // refresh UI | ||||||
|  |                 toggleActiveElement(intervalSelector, "interval", interval); | ||||||
|  |                 // finally create timer task | ||||||
|  |                 intervalId = setInterval(callback, interval); | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             toggle.checked = false; // no active interval found, disable toggle | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         localStorage.setItem(`ahriman-${toggle.id}-refresh-enabled`, toggle.checked); | ||||||
|  |         localStorage.setItem(`ahriman-${toggle.id}-refresh-interval`, interval); | ||||||
|  |         return intervalId; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function updateTable(table, rows) { | ||||||
|  |         // instead of using load method here, we just update rows manually to avoid table reinitialization | ||||||
|  |         const currentData = table.bootstrapTable("getData").reduce((accumulator, row) => { | ||||||
|  |             accumulator[row.id] = row["0"]; | ||||||
|  |             return accumulator; | ||||||
|  |         }, {}); | ||||||
|  |         // insert or update rows | ||||||
|  |         rows.forEach(row => { | ||||||
|  |             if (Object.hasOwn(currentData, row.id)) { | ||||||
|  |                 row["0"] = currentData[row.id]; // copy checkbox state | ||||||
|  |                 table.bootstrapTable("updateByUniqueId", { | ||||||
|  |                     id: row.id, | ||||||
|  |                     row: row, | ||||||
|  |                     replace: true, | ||||||
|  |                 }); | ||||||
|  |             } else { | ||||||
|  |                 table.bootstrapTable("insertRow", {index: 0, row: row}); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |         // remove old rows | ||||||
|  |         const newData = rows.map(value => value.id); | ||||||
|  |         Object.keys(currentData).forEach(id => { | ||||||
|  |             if (!newData.includes(id)) { | ||||||
|  |                 table.bootstrapTable("removeByUniqueId", id); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     Array.prototype.equals = function (right, comparator) { | ||||||
|  |         let index = this.length; | ||||||
|  |         if (index !== right.length) { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         while (index--) { | ||||||
|  |             if (!comparator(this[index], right[index])) { | ||||||
|  |                 return false; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     Date.prototype.toISOStringShort = function () { | ||||||
|         const pad = number => String(number).padStart(2, "0"); |         const pad = number => String(number).padStart(2, "0"); | ||||||
|         return `${this.getFullYear()}-${pad(this.getMonth() + 1)}-${pad(this.getDate())} ${pad(this.getHours())}:${pad(this.getMinutes())}:${pad(this.getSeconds())}`; |         return `${this.getFullYear()}-${pad(this.getMonth() + 1)}-${pad(this.getDate())} ${pad(this.getHours())}:${pad(this.getMinutes())}:${pad(this.getSeconds())}`; | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -1,15 +1,15 @@ | |||||||
| <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.3/dist/css/bootstrap.min.css" crossorigin="anonymous" type="text/css"> | <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.7/dist/css/bootstrap.min.css" crossorigin="anonymous" type="text/css"> | ||||||
| <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap-icons@1.11.1/font/bootstrap-icons.css" crossorigin="anonymous" type="text/css"> | <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap-icons@1.13.1/font/bootstrap-icons.css" crossorigin="anonymous" type="text/css"> | ||||||
|  |  | ||||||
| <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap-table@1.23.2/dist/bootstrap-table.min.css" crossorigin="anonymous" type="text/css"> | <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap-table@1.24.1/dist/bootstrap-table.min.css" crossorigin="anonymous" type="text/css"> | ||||||
|  |  | ||||||
| <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/jquery-resizable-columns@0.2.3/dist/jquery.resizableColumns.css" crossorigin="anonymous" type="text/css"> | <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/jquery-resizable-columns@0.2.3/dist/jquery.resizableColumns.css" crossorigin="anonymous" type="text/css"> | ||||||
|  |  | ||||||
| <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap-table@1.23.2/dist/extensions/filter-control/bootstrap-table-filter-control.css" crossorigin="anonymous" type="text/css"> | <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap-table@1.24.1/dist/extensions/filter-control/bootstrap-table-filter-control.css" crossorigin="anonymous" type="text/css"> | ||||||
|  |  | ||||||
| <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootswatch@5.3.3/dist/cosmo/bootstrap.min.css" crossorigin="anonymous" type="text/css"> | <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootswatch@5.3.7/dist/cosmo/bootstrap.min.css" crossorigin="anonymous" type="text/css"> | ||||||
|  |  | ||||||
| <link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/highlightjs/cdn-release@11.10.0/build/styles/github.min.css" crossorigin="anonymous" type="text/css"> | <link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/highlightjs/cdn-release@11.11.1/build/styles/github.min.css" crossorigin="anonymous" type="text/css"> | ||||||
|  |  | ||||||
| <style> | <style> | ||||||
|     .pre-scrollable { |     .pre-scrollable { | ||||||
|  | |||||||
| @ -1,6 +1,6 @@ | |||||||
| .TH AHRIMAN "1" "2025\-06\-16" "ahriman" "Generated Python Manual" | .TH AHRIMAN "1" "2025\-06\-29" "ahriman 2.19.0" "ArcH linux ReposItory MANager" | ||||||
| .SH NAME | .SH NAME | ||||||
| ahriman | ahriman \- ArcH linux ReposItory MANager | ||||||
| .SH SYNOPSIS | .SH SYNOPSIS | ||||||
| .B ahriman | .B ahriman | ||||||
| [-h] [-a ARCHITECTURE] [-c CONFIGURATION] [--force] [-l LOCK] [--log-handler {console,syslog,journald}] [-q] [--report | --no-report] [-r REPOSITORY] [--unsafe] [-V] [--wait-timeout WAIT_TIMEOUT] {add,aur-search,check,clean,config,config-validate,copy,daemon,help,help-commands-unsafe,help-updates,help-version,init,key-import,package-add,package-changes,package-changes-remove,package-copy,package-remove,package-status,package-status-remove,package-status-update,package-update,patch-add,patch-list,patch-remove,patch-set-add,rebuild,remove,remove-unknown,repo-backup,repo-check,repo-clean,repo-config,repo-config-validate,repo-create-keyring,repo-create-mirrorlist,repo-daemon,repo-init,repo-rebuild,repo-remove-unknown,repo-report,repo-restore,repo-setup,repo-sign,repo-statistics,repo-status-update,repo-sync,repo-tree,repo-triggers,repo-update,report,run,search,service-clean,service-config,service-config-validate,service-key-import,service-repositories,service-run,service-setup,service-shell,service-tree-migrate,setup,shell,sign,status,status-update,sync,update,user-add,user-list,user-remove,version,web} ... | [-h] [-a ARCHITECTURE] [-c CONFIGURATION] [--force] [-l LOCK] [--log-handler {console,syslog,journald}] [-q] [--report | --no-report] [-r REPOSITORY] [--unsafe] [-V] [--wait-timeout WAIT_TIMEOUT] {add,aur-search,check,clean,config,config-validate,copy,daemon,help,help-commands-unsafe,help-updates,help-version,init,key-import,package-add,package-changes,package-changes-remove,package-copy,package-remove,package-status,package-status-remove,package-status-update,package-update,patch-add,patch-list,patch-remove,patch-set-add,rebuild,remove,remove-unknown,repo-backup,repo-check,repo-clean,repo-config,repo-config-validate,repo-create-keyring,repo-create-mirrorlist,repo-daemon,repo-init,repo-rebuild,repo-remove-unknown,repo-report,repo-restore,repo-setup,repo-sign,repo-statistics,repo-status-update,repo-sync,repo-tree,repo-triggers,repo-update,report,run,search,service-clean,service-config,service-config-validate,service-key-import,service-repositories,service-run,service-setup,service-shell,service-tree-migrate,setup,shell,sign,status,status-update,sync,update,user-add,user-list,user-remove,version,web} ... | ||||||
|  | |||||||
| @ -58,23 +58,23 @@ web = [ | |||||||
|     "aiohttp_cors", |     "aiohttp_cors", | ||||||
|     "aiohttp_jinja2", |     "aiohttp_jinja2", | ||||||
| ] | ] | ||||||
| web_api-docs = [ | web-auth = [ | ||||||
|     "ahriman[web]", |  | ||||||
|     "aiohttp-apispec", |  | ||||||
|     "setuptools",  # required by aiohttp-apispec |  | ||||||
| ] |  | ||||||
| web_auth = [ |  | ||||||
|     "ahriman[web]", |     "ahriman[web]", | ||||||
|     "aiohttp_session", |     "aiohttp_session", | ||||||
|     "aiohttp_security", |     "aiohttp_security", | ||||||
|     "cryptography", |     "cryptography", | ||||||
| ] | ] | ||||||
| web_metrics = [ | web-docs = [ | ||||||
|  |     "ahriman[web]", | ||||||
|  |     "aiohttp-apispec", | ||||||
|  |     "setuptools",  # required by aiohttp-apispec | ||||||
|  | ] | ||||||
|  | web-metrics = [ | ||||||
|     "ahriman[web]", |     "ahriman[web]", | ||||||
|     "aiohttp-openmetrics", |     "aiohttp-openmetrics", | ||||||
| ] | ] | ||||||
| web_oauth2 = [ | web-oauth2 = [ | ||||||
|     "ahriman[web_auth]", |     "ahriman[web-auth]", | ||||||
|     "aioauth-client", |     "aioauth-client", | ||||||
| ] | ] | ||||||
|  |  | ||||||
|  | |||||||
| @ -8,7 +8,7 @@ services: | |||||||
|       AHRIMAN_OUTPUT: console |       AHRIMAN_OUTPUT: console | ||||||
|       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} |       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} | ||||||
|       AHRIMAN_PORT: 8080 |       AHRIMAN_PORT: 8080 | ||||||
|       AHRIMAN_PRESETUP_COMMAND: (cat /run/secrets/password; echo; cat /run/secrets/password) | sudo -u ahriman ahriman user-add demo -R full |       AHRIMAN_POSTSETUP_COMMAND: (cat /run/secrets/password; echo; cat /run/secrets/password) | sudo -u ahriman ahriman user-add demo -R full | ||||||
|       AHRIMAN_REPOSITORY: ahriman-demo |       AHRIMAN_REPOSITORY: ahriman-demo | ||||||
|       AHRIMAN_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock |       AHRIMAN_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock | ||||||
|  |  | ||||||
|  | |||||||
| @ -8,7 +8,7 @@ services: | |||||||
|       AHRIMAN_OUTPUT: console |       AHRIMAN_OUTPUT: console | ||||||
|       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} |       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} | ||||||
|       AHRIMAN_PORT: 8080 |       AHRIMAN_PORT: 8080 | ||||||
|       AHRIMAN_PRESETUP_COMMAND: (cat /run/secrets/password; echo; cat /run/secrets/password) | sudo -u ahriman ahriman user-add demo -R full |       AHRIMAN_POSTSETUP_COMMAND: (cat /run/secrets/password; echo; cat /run/secrets/password) | sudo -u ahriman ahriman user-add demo -R full | ||||||
|       AHRIMAN_REPOSITORY: ahriman-demo |       AHRIMAN_REPOSITORY: ahriman-demo | ||||||
|       AHRIMAN_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock |       AHRIMAN_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock | ||||||
|  |  | ||||||
|  | |||||||
| @ -8,7 +8,7 @@ services: | |||||||
|       AHRIMAN_OUTPUT: console |       AHRIMAN_OUTPUT: console | ||||||
|       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} |       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} | ||||||
|       AHRIMAN_PORT: 8080 |       AHRIMAN_PORT: 8080 | ||||||
|       AHRIMAN_PRESETUP_COMMAND: (cat /run/secrets/password; echo; cat /run/secrets/password) | sudo -u ahriman ahriman user-add demo -R full |       AHRIMAN_POSTSETUP_COMMAND: (cat /run/secrets/password; echo; cat /run/secrets/password) | sudo -u ahriman ahriman user-add demo -R full | ||||||
|       AHRIMAN_REPOSITORY: ahriman-demo |       AHRIMAN_REPOSITORY: ahriman-demo | ||||||
|       AHRIMAN_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock |       AHRIMAN_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock | ||||||
|  |  | ||||||
| @ -62,7 +62,7 @@ services: | |||||||
|       AHRIMAN_OUTPUT: console |       AHRIMAN_OUTPUT: console | ||||||
|       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} |       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} | ||||||
|       AHRIMAN_PORT: 8080 |       AHRIMAN_PORT: 8080 | ||||||
|       AHRIMAN_PRESETUP_COMMAND: (cat /run/secrets/password; echo; cat /run/secrets/password) | sudo -u ahriman ahriman user-add demo -R full |       AHRIMAN_POSTSETUP_COMMAND: (cat /run/secrets/password; echo; cat /run/secrets/password) | sudo -u ahriman ahriman user-add demo -R full | ||||||
|       AHRIMAN_REPOSITORY: ahriman-demo |       AHRIMAN_REPOSITORY: ahriman-demo | ||||||
|       AHRIMAN_REPOSITORY_SERVER: http://frontend/repo/$$repo/$$arch |       AHRIMAN_REPOSITORY_SERVER: http://frontend/repo/$$repo/$$arch | ||||||
|  |  | ||||||
|  | |||||||
| @ -12,7 +12,7 @@ services: | |||||||
|       AHRIMAN_PACMAN_MIRROR: https://de.mirror.archlinux32.org/$$arch/$$repo |       AHRIMAN_PACMAN_MIRROR: https://de.mirror.archlinux32.org/$$arch/$$repo | ||||||
|       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} |       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} | ||||||
|       AHRIMAN_PORT: 8080 |       AHRIMAN_PORT: 8080 | ||||||
|       AHRIMAN_PRESETUP_COMMAND: (cat /run/secrets/password; echo; cat /run/secrets/password) | sudo -u ahriman ahriman user-add demo -R full |       AHRIMAN_POSTSETUP_COMMAND: (cat /run/secrets/password; echo; cat /run/secrets/password) | sudo -u ahriman ahriman user-add demo -R full | ||||||
|       AHRIMAN_REPOSITORY: ahriman-demo |       AHRIMAN_REPOSITORY: ahriman-demo | ||||||
|       AHRIMAN_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock |       AHRIMAN_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock | ||||||
|  |  | ||||||
|  | |||||||
| @ -8,8 +8,8 @@ services: | |||||||
|       AHRIMAN_OUTPUT: console |       AHRIMAN_OUTPUT: console | ||||||
|       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} |       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} | ||||||
|       AHRIMAN_PORT: 8080 |       AHRIMAN_PORT: 8080 | ||||||
|       AHRIMAN_POSTSETUP_COMMAND: ahriman --architecture x86_64 --repository another-demo service-setup --build-as-user ahriman --packager 'ahriman bot <ahriman@example.com>' |       AHRIMAN_POSTSETUP_COMMAND: (cat /run/secrets/password; echo; cat /run/secrets/password) | sudo -u ahriman ahriman user-add demo -R full | ||||||
|       AHRIMAN_PRESETUP_COMMAND: (cat /run/secrets/password; echo; cat /run/secrets/password) | sudo -u ahriman ahriman user-add demo -R full |       AHRIMAN_PRESETUP_COMMAND: ahriman --architecture x86_64 --repository another-demo service-setup --build-as-user ahriman --packager 'ahriman bot <ahriman@example.com>' | ||||||
|       AHRIMAN_REPOSITORY: ahriman-demo |       AHRIMAN_REPOSITORY: ahriman-demo | ||||||
|       AHRIMAN_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock |       AHRIMAN_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock | ||||||
|  |  | ||||||
|  | |||||||
| @ -9,7 +9,7 @@ services: | |||||||
|       AHRIMAN_OAUTH_CLIENT_SECRET: ${AHRIMAN_OAUTH_CLIENT_SECRET} |       AHRIMAN_OAUTH_CLIENT_SECRET: ${AHRIMAN_OAUTH_CLIENT_SECRET} | ||||||
|       AHRIMAN_OUTPUT: console |       AHRIMAN_OUTPUT: console | ||||||
|       AHRIMAN_PORT: 8080 |       AHRIMAN_PORT: 8080 | ||||||
|       AHRIMAN_PRESETUP_COMMAND: sudo -u ahriman ahriman user-add ${AHRIMAN_OAUTH_USER} -R full -p "" |       AHRIMAN_POSTSETUP_COMMAND: sudo -u ahriman ahriman user-add ${AHRIMAN_OAUTH_USER} -R full -p "" | ||||||
|       AHRIMAN_REPOSITORY: ahriman-demo |       AHRIMAN_REPOSITORY: ahriman-demo | ||||||
|       AHRIMAN_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock |       AHRIMAN_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock | ||||||
|  |  | ||||||
|  | |||||||
| @ -6,7 +6,7 @@ services: | |||||||
|     environment: |     environment: | ||||||
|       AHRIMAN_DEBUG: yes |       AHRIMAN_DEBUG: yes | ||||||
|       AHRIMAN_OUTPUT: console |       AHRIMAN_OUTPUT: console | ||||||
|       AHRIMAN_PRESETUP_COMMAND: sudo -u ahriman gpg --import /run/secrets/key |       AHRIMAN_POSTSETUP_COMMAND: sudo -u ahriman gpg --import /run/secrets/key | ||||||
|       AHRIMAN_REPOSITORY: ahriman-demo |       AHRIMAN_REPOSITORY: ahriman-demo | ||||||
|  |  | ||||||
|     configs: |     configs: | ||||||
|  | |||||||
| @ -8,7 +8,7 @@ services: | |||||||
|       AHRIMAN_OUTPUT: console |       AHRIMAN_OUTPUT: console | ||||||
|       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} |       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} | ||||||
|       AHRIMAN_PORT: 8080 |       AHRIMAN_PORT: 8080 | ||||||
|       AHRIMAN_PRESETUP_COMMAND: (cat /run/secrets/password; echo; cat /run/secrets/password) | sudo -u ahriman ahriman user-add demo -R full |       AHRIMAN_POSTSETUP_COMMAND: (cat /run/secrets/password; echo; cat /run/secrets/password) | sudo -u ahriman ahriman user-add demo -R full | ||||||
|       AHRIMAN_REPOSITORY: ahriman-demo |       AHRIMAN_REPOSITORY: ahriman-demo | ||||||
|       AHRIMAN_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock |       AHRIMAN_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock | ||||||
|  |  | ||||||
|  | |||||||
| @ -17,4 +17,4 @@ | |||||||
| # 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.18.3" | __version__ = "2.19.0" | ||||||
|  | |||||||
| @ -133,18 +133,18 @@ class Application(ApplicationPackages, ApplicationRepository): | |||||||
|         if not process_dependencies or not packages: |         if not process_dependencies or not packages: | ||||||
|             return packages |             return packages | ||||||
|  |  | ||||||
|         def missing_dependencies(source: Iterable[Package]) -> dict[str, str | None]: |         def missing_dependencies(sources: Iterable[Package]) -> dict[str, str | None]: | ||||||
|             # append list of known packages with packages which are in current sources |             # append list of known packages with packages which are in current sources | ||||||
|             satisfied_packages = known_packages | { |             satisfied_packages = known_packages | { | ||||||
|                 single |                 single | ||||||
|                 for package in source |                 for source in sources | ||||||
|                 for single in package.packages_full |                 for single in source.packages_full | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             return { |             return { | ||||||
|                 dependency: package.packager |                 dependency: source.packager | ||||||
|                 for package in source |                 for source in sources | ||||||
|                 for dependency in package.depends_build |                 for dependency in source.depends_build | ||||||
|                 if dependency not in satisfied_packages |                 if dependency not in satisfied_packages | ||||||
|             } |             } | ||||||
|  |  | ||||||
| @ -156,7 +156,7 @@ class Application(ApplicationPackages, ApplicationRepository): | |||||||
|                         # there is local cache, load package from it |                         # there is local cache, load package from it | ||||||
|                         leaf = Package.from_build(source_dir, self.repository.architecture, packager) |                         leaf = Package.from_build(source_dir, self.repository.architecture, packager) | ||||||
|                     else: |                     else: | ||||||
|                         leaf = Package.from_aur(package_name, packager) |                         leaf = Package.from_aur(package_name, packager, include_provides=True) | ||||||
|                     portion[leaf.base] = leaf |                     portion[leaf.base] = leaf | ||||||
|  |  | ||||||
|                     # register package in the database |                     # register package in the database | ||||||
|  | |||||||
							
								
								
									
										70
									
								
								src/ahriman/application/handlers/reload.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								src/ahriman/application/handlers/reload.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,70 @@ | |||||||
|  | # | ||||||
|  | # Copyright (c) 2021-2025 ahriman team. | ||||||
|  | # | ||||||
|  | # This file is part of ahriman | ||||||
|  | # (see https://github.com/arcan1s/ahriman). | ||||||
|  | # | ||||||
|  | # This program is free software: you can redistribute it and/or modify | ||||||
|  | # it under the terms of the GNU General Public License as published by | ||||||
|  | # the Free Software Foundation, either version 3 of the License, or | ||||||
|  | # (at your option) any later version. | ||||||
|  | # | ||||||
|  | # This program is distributed in the hope that it will be useful, | ||||||
|  | # but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  | # GNU General Public License for more details. | ||||||
|  | # | ||||||
|  | # You should have received a copy of the GNU General Public License | ||||||
|  | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
|  | # | ||||||
|  | import argparse | ||||||
|  |  | ||||||
|  | from ahriman.application.application import Application | ||||||
|  | from ahriman.application.handlers.handler import Handler, SubParserAction | ||||||
|  | from ahriman.core.configuration import Configuration | ||||||
|  | from ahriman.models.repository_id import RepositoryId | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Reload(Handler): | ||||||
|  |     """ | ||||||
|  |     web server reload handler | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     ALLOW_MULTI_ARCHITECTURE_RUN = False  # system-wide action | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def run(cls, args: argparse.Namespace, repository_id: RepositoryId, configuration: Configuration, *, | ||||||
|  |             report: bool) -> None: | ||||||
|  |         """ | ||||||
|  |         callback for command line | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             args(argparse.Namespace): command line args | ||||||
|  |             repository_id(RepositoryId): repository unique identifier | ||||||
|  |             configuration(Configuration): configuration instance | ||||||
|  |             report(bool): force enable or disable reporting | ||||||
|  |         """ | ||||||
|  |         application = Application(repository_id, configuration, report=True) | ||||||
|  |         client = application.repository.reporter | ||||||
|  |         client.configuration_reload() | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def _set_web_reload_parser(root: SubParserAction) -> argparse.ArgumentParser: | ||||||
|  |         """ | ||||||
|  |         add parser for web reload subcommand | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             root(SubParserAction): subparsers for the commands | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             argparse.ArgumentParser: created argument parser | ||||||
|  |         """ | ||||||
|  |         parser = root.add_parser("web-reload", help="reload configuration", | ||||||
|  |                                  description="reload web server configuration", | ||||||
|  |                                  epilog="This method forces the web server to reload its configuration. " | ||||||
|  |                                         "Note, however, that this method does not apply all configuration changes " | ||||||
|  |                                         "(like ports, authentication, etc)") | ||||||
|  |         parser.set_defaults(architecture="", lock=None, quiet=True, report=False, repository="", unsafe=True) | ||||||
|  |         return parser | ||||||
|  |  | ||||||
|  |     arguments = [_set_web_reload_parser] | ||||||
| @ -28,6 +28,7 @@ from ahriman.core.alpm.remote import AUR, Official | |||||||
| from ahriman.core.configuration import Configuration | 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.core.types import Comparable | ||||||
| from ahriman.models.aur_package import AURPackage | from ahriman.models.aur_package import AURPackage | ||||||
| from ahriman.models.repository_id import RepositoryId | from ahriman.models.repository_id import RepositoryId | ||||||
|  |  | ||||||
| @ -115,7 +116,7 @@ class Search(Handler): | |||||||
|             raise OptionError(sort_by) |             raise OptionError(sort_by) | ||||||
|         # always sort by package name at the last |         # always sort by package name at the last | ||||||
|         # well technically it is not a string, but we can deal with it |         # well technically it is not a string, but we can deal with it | ||||||
|         comparator: Callable[[AURPackage], tuple[str, str]] =\ |         comparator: Callable[[AURPackage], Comparable] = \ | ||||||
|             lambda package: (getattr(package, sort_by), package.name) |             lambda package: (getattr(package, sort_by), package.name) | ||||||
|         return sorted(packages, key=comparator) |         return sorted(packages, key=comparator) | ||||||
|  |  | ||||||
|  | |||||||
| @ -72,6 +72,7 @@ class Setup(Handler): | |||||||
|  |  | ||||||
|         application = Application(repository_id, configuration, report=report) |         application = Application(repository_id, configuration, report=report) | ||||||
|  |  | ||||||
|  |         with application.repository.paths.preserve_owner(): | ||||||
|             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.executable_create(application.repository.paths, repository_id) | ||||||
|             repository_server = f"file://{application.repository.paths.repository}" if args.server is None else args.server |             repository_server = f"file://{application.repository.paths.repository}" if args.server is None else args.server | ||||||
| @ -280,6 +281,5 @@ class Setup(Handler): | |||||||
|         command = Setup.build_command(paths.root, repository_id) |         command = Setup.build_command(paths.root, repository_id) | ||||||
|         command.unlink(missing_ok=True) |         command.unlink(missing_ok=True) | ||||||
|         command.symlink_to(Setup.ARCHBUILD_COMMAND_PATH) |         command.symlink_to(Setup.ARCHBUILD_COMMAND_PATH) | ||||||
|         paths.chown(command)  # we would like to keep owner inside ahriman's home |  | ||||||
|  |  | ||||||
|     arguments = [_set_service_setup_parser] |     arguments = [_set_service_setup_parser] | ||||||
|  | |||||||
| @ -25,6 +25,7 @@ from ahriman.application.application import Application | |||||||
| from ahriman.application.handlers.handler import Handler, SubParserAction | from ahriman.application.handlers.handler import Handler, SubParserAction | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.core.formatters import PackagePrinter, StatusPrinter | from ahriman.core.formatters import PackagePrinter, StatusPrinter | ||||||
|  | from ahriman.core.types import Comparable | ||||||
| from ahriman.core.utils import enum_values | from ahriman.core.utils import enum_values | ||||||
| from ahriman.models.build_status import BuildStatus, BuildStatusEnum | from ahriman.models.build_status import BuildStatus, BuildStatusEnum | ||||||
| from ahriman.models.package import Package | from ahriman.models.package import Package | ||||||
| @ -64,8 +65,8 @@ class Status(Handler): | |||||||
|  |  | ||||||
|         Status.check_status(args.exit_code, packages) |         Status.check_status(args.exit_code, packages) | ||||||
|  |  | ||||||
|         comparator: Callable[[tuple[Package, BuildStatus]], str] = lambda item: item[0].base |         comparator: Callable[[tuple[Package, BuildStatus]], Comparable] = lambda item: item[0].base | ||||||
|         filter_fn: Callable[[tuple[Package, BuildStatus]], bool] =\ |         filter_fn: Callable[[tuple[Package, BuildStatus]], bool] = \ | ||||||
|             lambda item: args.status is None or item[1].status == args.status |             lambda item: args.status is None or item[1].status == args.status | ||||||
|         for package, package_status in sorted(filter(filter_fn, packages), key=comparator): |         for package, package_status in sorted(filter(filter_fn, packages), key=comparator): | ||||||
|             PackagePrinter(package, package_status)(verbose=args.info) |             PackagePrinter(package, package_status)(verbose=args.info) | ||||||
|  | |||||||
| @ -21,6 +21,7 @@ import argparse | |||||||
|  |  | ||||||
| from ahriman.application.handlers.handler import Handler, SubParserAction | from ahriman.application.handlers.handler import Handler, SubParserAction | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
|  | from ahriman.core.utils import walk | ||||||
| from ahriman.models.repository_id import RepositoryId | from ahriman.models.repository_id import RepositoryId | ||||||
| from ahriman.models.repository_paths import RepositoryPaths | from ahriman.models.repository_paths import RepositoryPaths | ||||||
|  |  | ||||||
| @ -49,6 +50,7 @@ class TreeMigrate(Handler): | |||||||
|         target_tree.tree_create() |         target_tree.tree_create() | ||||||
|         # perform migration |         # perform migration | ||||||
|         TreeMigrate.tree_move(current_tree, target_tree) |         TreeMigrate.tree_move(current_tree, target_tree) | ||||||
|  |         TreeMigrate.fix_symlinks(target_tree) | ||||||
|  |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def _set_service_tree_migrate_parser(root: SubParserAction) -> argparse.ArgumentParser: |     def _set_service_tree_migrate_parser(root: SubParserAction) -> argparse.ArgumentParser: | ||||||
| @ -66,6 +68,22 @@ class TreeMigrate(Handler): | |||||||
|         parser.set_defaults(lock=None, quiet=True, report=False) |         parser.set_defaults(lock=None, quiet=True, report=False) | ||||||
|         return parser |         return parser | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def fix_symlinks(paths: RepositoryPaths) -> None: | ||||||
|  |         """ | ||||||
|  |         fix packages archives symlinks | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             paths(RepositoryPaths): new repository paths | ||||||
|  |         """ | ||||||
|  |         archives = {path.name: path for path in walk(paths.archive)} | ||||||
|  |         for symlink in walk(paths.repository): | ||||||
|  |             if symlink.exists():  # no need to check for symlinks as we have just walked through the tree | ||||||
|  |                 continue | ||||||
|  |             if (source_archive := archives.get(symlink.name)) is not None: | ||||||
|  |                 symlink.unlink() | ||||||
|  |                 symlink.symlink_to(source_archive.relative_to(symlink.parent, walk_up=True)) | ||||||
|  |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def tree_move(from_tree: RepositoryPaths, to_tree: RepositoryPaths) -> None: |     def tree_move(from_tree: RepositoryPaths, to_tree: RepositoryPaths) -> None: | ||||||
|         """ |         """ | ||||||
|  | |||||||
| @ -52,7 +52,7 @@ class Validate(Handler): | |||||||
|         """ |         """ | ||||||
|         from ahriman.core.configuration.validator import Validator |         from ahriman.core.configuration.validator import Validator | ||||||
|  |  | ||||||
|         schema = Validate.schema(repository_id, configuration) |         schema = Validate.schema(configuration) | ||||||
|         validator = Validator(configuration=configuration, schema=schema) |         validator = Validator(configuration=configuration, schema=schema) | ||||||
|  |  | ||||||
|         if validator.validate(configuration.dump()): |         if validator.validate(configuration.dump()): | ||||||
| @ -83,12 +83,11 @@ class Validate(Handler): | |||||||
|         return parser |         return parser | ||||||
|  |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def schema(repository_id: RepositoryId, configuration: Configuration) -> ConfigurationSchema: |     def schema(configuration: Configuration) -> ConfigurationSchema: | ||||||
|         """ |         """ | ||||||
|         get schema with triggers |         get schema with triggers | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             repository_id(RepositoryId): repository unique identifier |  | ||||||
|             configuration(Configuration): configuration instance |             configuration(Configuration): configuration instance | ||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
| @ -107,12 +106,12 @@ class Validate(Handler): | |||||||
|                 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(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(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 | ||||||
|  | |||||||
| @ -130,8 +130,8 @@ class Pacman(LazyLogging): | |||||||
|             return  # database for some reason deos not exist |             return  # database for some reason deos not exist | ||||||
|  |  | ||||||
|         self.logger.info("copy pacman database %s from operating system root to ahriman's home %s", src, dst) |         self.logger.info("copy pacman database %s from operating system root to ahriman's home %s", src, dst) | ||||||
|  |         with self.repository_paths.preserve_owner(dst.parent): | ||||||
|             shutil.copy(src, dst) |             shutil.copy(src, dst) | ||||||
|         self.repository_paths.chown(dst) |  | ||||||
|  |  | ||||||
|     def database_init(self, handle: Handle, repository: str, architecture: str) -> DB: |     def database_init(self, handle: Handle, repository: str, architecture: str) -> DB: | ||||||
|         """ |         """ | ||||||
| @ -255,3 +255,20 @@ 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 provided_by(self, package_name: str) -> Generator[Package, None, None]: | ||||||
|  |         """ | ||||||
|  |         search through databases and emit packages which provides the ``package_name`` | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             package_name(str): package name to search | ||||||
|  |  | ||||||
|  |         Yields: | ||||||
|  |             Package: list of packages which were returned by the query | ||||||
|  |         """ | ||||||
|  |         def is_package_provided(package: Package) -> bool: | ||||||
|  |             provides = [trim_package(name) for name in package.provides] | ||||||
|  |             return package_name in provides | ||||||
|  |  | ||||||
|  |         for database in self.handle.get_syncdbs(): | ||||||
|  |             yield from filter(is_package_provided, database.search(package_name)) | ||||||
|  | |||||||
| @ -97,20 +97,17 @@ class AUR(Remote): | |||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             list[AURPackage]: response parsed to package list |             list[AURPackage]: response parsed to package list | ||||||
|  |  | ||||||
|  |         Raises: | ||||||
|  |             PackageInfoError: if multiple arguments are passed | ||||||
|         """ |         """ | ||||||
|         query: list[tuple[str, str]] = [ |         if len(args) != 1: | ||||||
|             ("type", request_type), |             raise PackageInfoError("AUR API requires exactly one argument to search") | ||||||
|             ("v", self.DEFAULT_RPC_VERSION), |  | ||||||
|         ] |  | ||||||
|  |  | ||||||
|         arg_query = "arg[]" if len(args) > 1 else "arg" |         url = f"{self.DEFAULT_RPC_URL}/v{self.DEFAULT_RPC_VERSION}/{request_type}/{args[0]}" | ||||||
|         for arg in args: |         query = list(kwargs.items()) | ||||||
|             query.append((arg_query, arg)) |  | ||||||
|  |  | ||||||
|         for key, value in kwargs.items(): |         response = self.make_request("GET", url, params=query) | ||||||
|             query.append((key, value)) |  | ||||||
|  |  | ||||||
|         response = self.make_request("GET", self.DEFAULT_RPC_URL, params=query) |  | ||||||
|         return self.parse_response(response.json()) |         return self.parse_response(response.json()) | ||||||
|  |  | ||||||
|     def package_info(self, package_name: str, *, pacman: Pacman | None) -> AURPackage: |     def package_info(self, package_name: str, *, pacman: Pacman | None) -> AURPackage: | ||||||
| @ -133,15 +130,36 @@ class AUR(Remote): | |||||||
|         except StopIteration: |         except StopIteration: | ||||||
|             raise UnknownPackageError(package_name) from None |             raise UnknownPackageError(package_name) from None | ||||||
|  |  | ||||||
|     def package_search(self, *keywords: str, pacman: Pacman | None) -> list[AURPackage]: |     def package_provided_by(self, package_name: str, *, pacman: Pacman | None) -> list[AURPackage]: | ||||||
|  |         """ | ||||||
|  |         get package list which provide the specified package name | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             package_name(str): package name to search | ||||||
|  |             pacman(Pacman | None): alpm wrapper instance, required for official repositories search | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             list[AURPackage]: list of packages which match the criteria | ||||||
|  |         """ | ||||||
|  |         return [ | ||||||
|  |             package | ||||||
|  |             # search api provides reduced models | ||||||
|  |             for stub in self.package_search(package_name, pacman=pacman, search_by="provides") | ||||||
|  |             # verity that found package actually provides it | ||||||
|  |             if package_name in (package := self.package_info(stub.name, pacman=pacman)).provides | ||||||
|  |         ] | ||||||
|  |  | ||||||
|  |     def package_search(self, *keywords: str, pacman: Pacman | None, search_by: str | None) -> list[AURPackage]: | ||||||
|         """ |         """ | ||||||
|         search package in AUR web |         search package in AUR web | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             *keywords(str): keywords to search |             *keywords(str): keywords to search | ||||||
|             pacman(Pacman | None): alpm wrapper instance, required for official repositories search |             pacman(Pacman | None): alpm wrapper instance, required for official repositories search | ||||||
|  |             search_by(str | None): search by keywords | ||||||
|  |  | ||||||
|         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") |         search_by = search_by or "name-desc" | ||||||
|  |         return self.aur_request("search", *keywords, by=search_by) | ||||||
|  | |||||||
| @ -127,15 +127,17 @@ class Official(Remote): | |||||||
|         except StopIteration: |         except StopIteration: | ||||||
|             raise UnknownPackageError(package_name) from None |             raise UnknownPackageError(package_name) from None | ||||||
|  |  | ||||||
|     def package_search(self, *keywords: str, pacman: Pacman | None) -> list[AURPackage]: |     def package_search(self, *keywords: str, pacman: Pacman | None, search_by: str | None) -> list[AURPackage]: | ||||||
|         """ |         """ | ||||||
|         search package in AUR web |         search package in AUR web | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             *keywords(str): keywords to search |             *keywords(str): keywords to search | ||||||
|             pacman(Pacman | None): alpm wrapper instance, required for official repositories search |             pacman(Pacman | None): alpm wrapper instance, required for official repositories search | ||||||
|  |             search_by(str | None): search by keywords | ||||||
|  |  | ||||||
|         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") |         search_by = search_by or "q" | ||||||
|  |         return self.arch_request(*keywords, by=search_by) | ||||||
|  | |||||||
| @ -59,3 +59,22 @@ class OfficialSyncdb(Official): | |||||||
|             return next(AURPackage.from_pacman(package) for package in pacman.package(package_name)) |             return next(AURPackage.from_pacman(package) for package in pacman.package(package_name)) | ||||||
|         except StopIteration: |         except StopIteration: | ||||||
|             raise UnknownPackageError(package_name) from None |             raise UnknownPackageError(package_name) from None | ||||||
|  |  | ||||||
|  |     def package_provided_by(self, package_name: str, *, pacman: Pacman | None) -> list[AURPackage]: | ||||||
|  |         """ | ||||||
|  |         get package list which provide the specified package name | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             package_name(str): package name to search | ||||||
|  |             pacman(Pacman | None): alpm wrapper instance, required for official repositories search | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             list[AURPackage]: list of packages which match the criteria | ||||||
|  |         """ | ||||||
|  |         if pacman is None: | ||||||
|  |             return [] | ||||||
|  |  | ||||||
|  |         return [ | ||||||
|  |             AURPackage.from_pacman(package) | ||||||
|  |             for package in pacman.provided_by(package_name) | ||||||
|  |         ] | ||||||
|  | |||||||
| @ -18,6 +18,7 @@ | |||||||
| # 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.exceptions import UnknownPackageError | ||||||
| from ahriman.core.http import SyncHttpClient | from ahriman.core.http import SyncHttpClient | ||||||
| from ahriman.models.aur_package import AURPackage | from ahriman.models.aur_package import AURPackage | ||||||
|  |  | ||||||
| @ -41,22 +42,36 @@ class Remote(SyncHttpClient): | |||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def info(cls, package_name: str, *, pacman: Pacman | None = None) -> AURPackage: |     def info(cls, package_name: str, *, pacman: Pacman | None = None, include_provides: bool = False) -> AURPackage: | ||||||
|         """ |         """ | ||||||
|         get package info by its name |         get package info by its name. If ``include_provides`` is set to ``True``, then, in addition, this method | ||||||
|  |         will perform search by :attr:`ahriman.models.aur_package.AURPackage.provides` and return first package found. | ||||||
|  |         Note, however, that in this case some implementation might not provide this method and search result will might | ||||||
|  |         not be stable | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             package_name(str): package name to search |             package_name(str): package name to search | ||||||
|             pacman(Pacman | None, optional): alpm wrapper instance, required for official repositories search |             pacman(Pacman | None, optional): alpm wrapper instance, required for official repositories search | ||||||
|                 (Default value = None) |                 (Default value = None) | ||||||
|  |             include_provides(bool, optional): search by provides if no exact match found (Default value = False) | ||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             AURPackage: package which match the package name |             AURPackage: package which match the package name | ||||||
|  |  | ||||||
|  |         Raises: | ||||||
|  |             UnknownPackageError: if requested package not found | ||||||
|         """ |         """ | ||||||
|         return cls().package_info(package_name, pacman=pacman) |         instance = cls() | ||||||
|  |         try: | ||||||
|  |             return instance.package_info(package_name, pacman=pacman) | ||||||
|  |         except UnknownPackageError: | ||||||
|  |             if include_provides and (provided_by := instance.package_provided_by(package_name, pacman=pacman)): | ||||||
|  |                 return next(iter(provided_by)) | ||||||
|  |             raise | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def multisearch(cls, *keywords: str, pacman: Pacman | None = None) -> list[AURPackage]: |     def multisearch(cls, *keywords: str, pacman: Pacman | None = None, | ||||||
|  |                     search_by: str | None = None) -> list[AURPackage]: | ||||||
|         """ |         """ | ||||||
|         search in remote repository by using API with multiple words. This method is required in order to handle |         search in remote repository by using API with multiple words. This method is required in order to handle | ||||||
|         https://bugs.archlinux.org/task/49133. In addition, short words will be dropped |         https://bugs.archlinux.org/task/49133. In addition, short words will be dropped | ||||||
| @ -65,6 +80,7 @@ class Remote(SyncHttpClient): | |||||||
|             *keywords(str): search terms, e.g. "ahriman", "is", "cool" |             *keywords(str): search terms, e.g. "ahriman", "is", "cool" | ||||||
|             pacman(Pacman | None, optional): alpm wrapper instance, required for official repositories search |             pacman(Pacman | None, optional): alpm wrapper instance, required for official repositories search | ||||||
|                 (Default value = None) |                 (Default value = None) | ||||||
|  |             search_by(str | None, optional): search by keywords (Default value = None) | ||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             list[AURPackage]: list of packages each of them matches all search terms |             list[AURPackage]: list of packages each of them matches all search terms | ||||||
| @ -72,12 +88,21 @@ class Remote(SyncHttpClient): | |||||||
|         instance = cls() |         instance = cls() | ||||||
|         packages: dict[str, AURPackage] = {} |         packages: dict[str, AURPackage] = {} | ||||||
|         for term in filter(lambda word: len(word) >= 3, keywords): |         for term in filter(lambda word: len(word) >= 3, keywords): | ||||||
|             portion = instance.search(term, pacman=pacman) |             portion = instance.package_search(term, pacman=pacman, search_by=search_by) | ||||||
|             packages = { |             packages = { | ||||||
|                 package.name: package  # not mistake to group them by name |                 package.name: package  # not mistake to group them by name | ||||||
|                 for package in portion |                 for package in portion | ||||||
|                 if package.name in packages or not packages |                 if package.name in packages or not packages | ||||||
|             } |             } | ||||||
|  |  | ||||||
|  |         # simple check for duplicates. This method will remove all packages under base if there is | ||||||
|  |         # a package named exactly as its base | ||||||
|  |         packages = { | ||||||
|  |             package.name: package | ||||||
|  |             for package in packages.values() | ||||||
|  |             if package.package_base not in packages or package.package_base == package.name | ||||||
|  |         } | ||||||
|  |  | ||||||
|         return list(packages.values()) |         return list(packages.values()) | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
| @ -114,7 +139,7 @@ class Remote(SyncHttpClient): | |||||||
|         raise NotImplementedError |         raise NotImplementedError | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def search(cls, *keywords: str, pacman: Pacman | None = None) -> list[AURPackage]: |     def search(cls, *keywords: str, pacman: Pacman | None = None, search_by: str | None = None) -> list[AURPackage]: | ||||||
|         """ |         """ | ||||||
|         search package in AUR web |         search package in AUR web | ||||||
|  |  | ||||||
| @ -122,11 +147,12 @@ class Remote(SyncHttpClient): | |||||||
|             *keywords(str): search terms, e.g. "ahriman", "is", "cool" |             *keywords(str): search terms, e.g. "ahriman", "is", "cool" | ||||||
|             pacman(Pacman | None, optional): alpm wrapper instance, required for official repositories search |             pacman(Pacman | None, optional): alpm wrapper instance, required for official repositories search | ||||||
|                 (Default value = None) |                 (Default value = None) | ||||||
|  |             search_by(str | None, optional): search by keywords (Default value = None) | ||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             list[AURPackage]: list of packages which match the criteria |             list[AURPackage]: list of packages which match the criteria | ||||||
|         """ |         """ | ||||||
|         return cls().package_search(*keywords, pacman=pacman) |         return cls().package_search(*keywords, pacman=pacman, search_by=search_by) | ||||||
|  |  | ||||||
|     def package_info(self, package_name: str, *, pacman: Pacman | None) -> AURPackage: |     def package_info(self, package_name: str, *, pacman: Pacman | None) -> AURPackage: | ||||||
|         """ |         """ | ||||||
| @ -144,13 +170,28 @@ class Remote(SyncHttpClient): | |||||||
|         """ |         """ | ||||||
|         raise NotImplementedError |         raise NotImplementedError | ||||||
|  |  | ||||||
|     def package_search(self, *keywords: str, pacman: Pacman | None) -> list[AURPackage]: |     def package_provided_by(self, package_name: str, *, pacman: Pacman | None) -> list[AURPackage]: | ||||||
|  |         """ | ||||||
|  |         get package list which provide the specified package name | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             package_name(str): package name to search | ||||||
|  |             pacman(Pacman | None): alpm wrapper instance, required for official repositories search | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             list[AURPackage]: list of packages which match the criteria | ||||||
|  |         """ | ||||||
|  |         del package_name, pacman | ||||||
|  |         return [] | ||||||
|  |  | ||||||
|  |     def package_search(self, *keywords: str, pacman: Pacman | None, search_by: str | None) -> list[AURPackage]: | ||||||
|         """ |         """ | ||||||
|         search package in AUR web |         search package in AUR web | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             *keywords(str): keywords to search |             *keywords(str): keywords to search | ||||||
|             pacman(Pacman | None): alpm wrapper instance, required for official repositories search |             pacman(Pacman | None): alpm wrapper instance, required for official repositories search | ||||||
|  |             search_by(str | None): search by keywords | ||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             list[AURPackage]: list of packages which match the criteria |             list[AURPackage]: list of packages which match the criteria | ||||||
|  | |||||||
| @ -31,20 +31,21 @@ class Repo(LazyLogging): | |||||||
|  |  | ||||||
|     Attributes: |     Attributes: | ||||||
|         name(str): repository name |         name(str): repository name | ||||||
|         paths(RepositoryPaths): repository paths instance |         root(Path): repository root | ||||||
|         sign_args(list[str]): additional args which have to be used to sign repository archive |         sign_args(list[str]): additional args which have to be used to sign repository archive | ||||||
|         uid(int): uid of the repository owner user |         uid(int): uid of the repository owner user | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     def __init__(self, name: str, paths: RepositoryPaths, sign_args: list[str]) -> None: |     def __init__(self, name: str, paths: RepositoryPaths, sign_args: list[str], root: Path | None = None) -> None: | ||||||
|         """ |         """ | ||||||
|         Args: |         Args: | ||||||
|             name(str): repository name |             name(str): repository name | ||||||
|             paths(RepositoryPaths): repository paths instance |             paths(RepositoryPaths): repository paths instance | ||||||
|             sign_args(list[str]): additional args which have to be used to sign repository archive |             sign_args(list[str]): additional args which have to be used to sign repository archive | ||||||
|  |             root(Path | None, optional): repository root. If none set, the default will be used (Default value = None) | ||||||
|         """ |         """ | ||||||
|         self.name = name |         self.name = name | ||||||
|         self.paths = paths |         self.root = root or paths.repository | ||||||
|         self.uid, _ = paths.root_owner |         self.uid, _ = paths.root_owner | ||||||
|         self.sign_args = sign_args |         self.sign_args = sign_args | ||||||
|  |  | ||||||
| @ -56,45 +57,56 @@ class Repo(LazyLogging): | |||||||
|         Returns: |         Returns: | ||||||
|             Path: path to repository database |             Path: path to repository database | ||||||
|         """ |         """ | ||||||
|         return self.paths.repository / f"{self.name}.db.tar.gz" |         return self.root / f"{self.name}.db.tar.gz" | ||||||
|  |  | ||||||
|     def add(self, path: Path) -> None: |     def add(self, path: Path, *, remove: bool = True) -> None: | ||||||
|         """ |         """ | ||||||
|         add new package to repository |         add new package to repository | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             path(Path): path to archive to add |             path(Path): path to archive to add | ||||||
|  |             remove(bool, optional): whether to remove old packages or not (Default value = True) | ||||||
|         """ |         """ | ||||||
|  |         command = ["repo-add", *self.sign_args] | ||||||
|  |         if remove: | ||||||
|  |             command.extend(["--remove"]) | ||||||
|  |         command.extend([str(self.repo_path), str(path)]) | ||||||
|  |  | ||||||
|  |         # add to repository | ||||||
|         check_output( |         check_output( | ||||||
|             "repo-add", *self.sign_args, "-R", str(self.repo_path), str(path), |             *command, | ||||||
|             exception=BuildError.from_process(path.name), |             exception=BuildError.from_process(path.name), | ||||||
|             cwd=self.paths.repository, |             cwd=self.root, | ||||||
|             logger=self.logger, |             logger=self.logger, | ||||||
|             user=self.uid) |             user=self.uid, | ||||||
|  |         ) | ||||||
|  |  | ||||||
|     def init(self) -> None: |     def init(self) -> None: | ||||||
|         """ |         """ | ||||||
|         create empty repository database. It just calls add with empty arguments |         create empty repository database. It just calls add with empty arguments | ||||||
|         """ |         """ | ||||||
|         check_output("repo-add", *self.sign_args, str(self.repo_path), |         check_output("repo-add", *self.sign_args, str(self.repo_path), | ||||||
|                      cwd=self.paths.repository, logger=self.logger, user=self.uid) |                      cwd=self.root, logger=self.logger, user=self.uid) | ||||||
|  |  | ||||||
|     def remove(self, package: str, filename: Path) -> None: |     def remove(self, package_name: str | None, filename: Path) -> None: | ||||||
|         """ |         """ | ||||||
|         remove package from repository |         remove package from repository | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             package(str): package name to remove |             package_name(str | None): package name to remove. If none set, it will be guessed from filename | ||||||
|             filename(Path): package filename to remove |             filename(Path): package filename to remove | ||||||
|         """ |         """ | ||||||
|  |         package_name = package_name or filename.name.rsplit("-", maxsplit=3)[0] | ||||||
|  |  | ||||||
|         # remove package and signature (if any) from filesystem |         # remove package and signature (if any) from filesystem | ||||||
|         for full_path in self.paths.repository.glob(f"{filename}*"): |         for full_path in self.root.glob(f"**/{filename.name}*"): | ||||||
|             full_path.unlink() |             full_path.unlink() | ||||||
|  |  | ||||||
|         # remove package from registry |         # remove package from registry | ||||||
|         check_output( |         check_output( | ||||||
|             "repo-remove", *self.sign_args, str(self.repo_path), package, |             "repo-remove", *self.sign_args, str(self.repo_path), package_name, | ||||||
|             exception=BuildError.from_process(package), |             exception=BuildError.from_process(package_name), | ||||||
|             cwd=self.paths.repository, |             cwd=self.root, | ||||||
|             logger=self.logger, |             logger=self.logger, | ||||||
|             user=self.uid) |             user=self.uid, | ||||||
|  |         ) | ||||||
|  | |||||||
							
								
								
									
										20
									
								
								src/ahriman/core/archive/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								src/ahriman/core/archive/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,20 @@ | |||||||
|  | # | ||||||
|  | # Copyright (c) 2021-2025 ahriman team. | ||||||
|  | # | ||||||
|  | # This file is part of ahriman | ||||||
|  | # (see https://github.com/arcan1s/ahriman). | ||||||
|  | # | ||||||
|  | # This program is free software: you can redistribute it and/or modify | ||||||
|  | # it under the terms of the GNU General Public License as published by | ||||||
|  | # the Free Software Foundation, either version 3 of the License, or | ||||||
|  | # (at your option) any later version. | ||||||
|  | # | ||||||
|  | # This program is distributed in the hope that it will be useful, | ||||||
|  | # but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  | # GNU General Public License for more details. | ||||||
|  | # | ||||||
|  | # You should have received a copy of the GNU General Public License | ||||||
|  | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
|  | # | ||||||
|  | from ahriman.core.archive.archive_trigger import ArchiveTrigger | ||||||
							
								
								
									
										130
									
								
								src/ahriman/core/archive/archive_tree.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										130
									
								
								src/ahriman/core/archive/archive_tree.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,130 @@ | |||||||
|  | # | ||||||
|  | # Copyright (c) 2021-2025 ahriman team. | ||||||
|  | # | ||||||
|  | # This file is part of ahriman | ||||||
|  | # (see https://github.com/arcan1s/ahriman). | ||||||
|  | # | ||||||
|  | # This program is free software: you can redistribute it and/or modify | ||||||
|  | # it under the terms of the GNU General Public License as published by | ||||||
|  | # the Free Software Foundation, either version 3 of the License, or | ||||||
|  | # (at your option) any later version. | ||||||
|  | # | ||||||
|  | # This program is distributed in the hope that it will be useful, | ||||||
|  | # but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  | # GNU General Public License for more details. | ||||||
|  | # | ||||||
|  | # You should have received a copy of the GNU General Public License | ||||||
|  | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
|  | # | ||||||
|  | import datetime | ||||||
|  |  | ||||||
|  | from pathlib import Path | ||||||
|  |  | ||||||
|  | from ahriman.core.alpm.repo import Repo | ||||||
|  | from ahriman.core.log import LazyLogging | ||||||
|  | from ahriman.core.utils import utcnow, walk | ||||||
|  | from ahriman.models.package import Package | ||||||
|  | from ahriman.models.repository_paths import RepositoryPaths | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ArchiveTree(LazyLogging): | ||||||
|  |     """ | ||||||
|  |     wrapper around archive tree | ||||||
|  |  | ||||||
|  |     Attributes: | ||||||
|  |         paths(RepositoryPaths): repository paths instance | ||||||
|  |         repository_id(RepositoryId): repository unique identifier | ||||||
|  |         sign_args(list[str]): additional args which have to be used to sign repository archive | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def __init__(self, repository_path: RepositoryPaths, sign_args: list[str]) -> None: | ||||||
|  |         """ | ||||||
|  |         Args: | ||||||
|  |             repository_path(RepositoryPaths): repository paths instance | ||||||
|  |             sign_args(list[str]): additional args which have to be used to sign repository archive | ||||||
|  |         """ | ||||||
|  |         self.paths = repository_path | ||||||
|  |         self.repository_id = repository_path.repository_id | ||||||
|  |         self.sign_args = sign_args | ||||||
|  |  | ||||||
|  |     def repository_for(self, date: datetime.date | None = None) -> Path: | ||||||
|  |         """ | ||||||
|  |         get full path to repository at the specified date | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             date(datetime.date | None, optional): date to generate path. If none supplied then today will be used | ||||||
|  |                 (Default value = None) | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             Path: path to the repository root | ||||||
|  |         """ | ||||||
|  |         date = date or utcnow().date() | ||||||
|  |         return ( | ||||||
|  |             self.paths.archive | ||||||
|  |             / "repos" | ||||||
|  |             / date.strftime("%Y") | ||||||
|  |             / date.strftime("%m") | ||||||
|  |             / date.strftime("%d") | ||||||
|  |             / self.repository_id.name | ||||||
|  |             / self.repository_id.architecture | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     def symlinks_create(self, packages: list[Package]) -> None: | ||||||
|  |         """ | ||||||
|  |         create symlinks for the specified packages in today's repository | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             packages(list[Package]): list of packages to be updated | ||||||
|  |         """ | ||||||
|  |         root = self.repository_for() | ||||||
|  |         repo = Repo(self.repository_id.name, self.paths, self.sign_args, root) | ||||||
|  |  | ||||||
|  |         for package in packages: | ||||||
|  |             archive = self.paths.archive_for(package.base) | ||||||
|  |  | ||||||
|  |             for package_name, single in package.packages.items(): | ||||||
|  |                 if single.filename is None: | ||||||
|  |                     self.logger.warning("received empty package filename for %s", package_name) | ||||||
|  |                     continue | ||||||
|  |  | ||||||
|  |                 has_file = False | ||||||
|  |                 for file in archive.glob(f"{single.filename}*"): | ||||||
|  |                     symlink = root / file.name | ||||||
|  |                     if symlink.exists(): | ||||||
|  |                         continue  # symlink is already created, skip processing | ||||||
|  |                     has_file = True | ||||||
|  |                     symlink.symlink_to(file.relative_to(symlink.parent, walk_up=True)) | ||||||
|  |  | ||||||
|  |                 if has_file: | ||||||
|  |                     repo.add(root / single.filename) | ||||||
|  |  | ||||||
|  |     def symlinks_fix(self) -> None: | ||||||
|  |         """ | ||||||
|  |         remove broken symlinks across repositories for all dates | ||||||
|  |         """ | ||||||
|  |         for path in walk(self.paths.archive / "repos"): | ||||||
|  |             root = path.parent | ||||||
|  |             *_, name, architecture = root.parts | ||||||
|  |             if self.repository_id.name != name or self.repository_id.architecture != architecture: | ||||||
|  |                 continue  # we only process same name repositories | ||||||
|  |  | ||||||
|  |             if not path.is_symlink(): | ||||||
|  |                 continue  # find symlinks only | ||||||
|  |             if path.exists(): | ||||||
|  |                 continue  # filter out not broken symlinks | ||||||
|  |  | ||||||
|  |             Repo(self.repository_id.name, self.paths, self.sign_args, root).remove(None, path) | ||||||
|  |  | ||||||
|  |     def tree_create(self) -> None: | ||||||
|  |         """ | ||||||
|  |         create repository tree for current repository | ||||||
|  |         """ | ||||||
|  |         root = self.repository_for() | ||||||
|  |         if root.exists(): | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         with self.paths.preserve_owner(self.paths.archive): | ||||||
|  |             root.mkdir(0o755, parents=True) | ||||||
|  |             # init empty repository here | ||||||
|  |             Repo(self.repository_id.name, self.paths, self.sign_args, root).init() | ||||||
							
								
								
									
										69
									
								
								src/ahriman/core/archive/archive_trigger.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								src/ahriman/core/archive/archive_trigger.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,69 @@ | |||||||
|  | # | ||||||
|  | # Copyright (c) 2021-2025 ahriman team. | ||||||
|  | # | ||||||
|  | # This file is part of ahriman | ||||||
|  | # (see https://github.com/arcan1s/ahriman). | ||||||
|  | # | ||||||
|  | # This program is free software: you can redistribute it and/or modify | ||||||
|  | # it under the terms of the GNU General Public License as published by | ||||||
|  | # the Free Software Foundation, either version 3 of the License, or | ||||||
|  | # (at your option) any later version. | ||||||
|  | # | ||||||
|  | # This program is distributed in the hope that it will be useful, | ||||||
|  | # but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  | # GNU General Public License for more details. | ||||||
|  | # | ||||||
|  | # You should have received a copy of the GNU General Public License | ||||||
|  | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
|  | # | ||||||
|  | from ahriman.core.archive.archive_tree import ArchiveTree | ||||||
|  | from ahriman.core.configuration import Configuration | ||||||
|  | from ahriman.core.sign.gpg import GPG | ||||||
|  | from ahriman.core.triggers import Trigger | ||||||
|  | from ahriman.models.package import Package | ||||||
|  | from ahriman.models.repository_id import RepositoryId | ||||||
|  | from ahriman.models.result import Result | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ArchiveTrigger(Trigger): | ||||||
|  |     """ | ||||||
|  |     archive repository extension | ||||||
|  |  | ||||||
|  |     Attributes: | ||||||
|  |         paths(RepositoryPaths): repository paths instance | ||||||
|  |         tree(ArchiveTree): archive tree wrapper | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def __init__(self, repository_id: RepositoryId, configuration: Configuration) -> None: | ||||||
|  |         """ | ||||||
|  |         Args: | ||||||
|  |             repository_id(RepositoryId): repository unique identifier | ||||||
|  |             configuration(Configuration): configuration instance | ||||||
|  |         """ | ||||||
|  |         Trigger.__init__(self, repository_id, configuration) | ||||||
|  |  | ||||||
|  |         self.paths = configuration.repository_paths | ||||||
|  |         self.tree = ArchiveTree(self.paths, GPG(configuration).repository_sign_args) | ||||||
|  |  | ||||||
|  |     def on_result(self, result: Result, packages: list[Package]) -> None: | ||||||
|  |         """ | ||||||
|  |         run trigger | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             result(Result): build result | ||||||
|  |             packages(list[Package]): list of all available packages | ||||||
|  |         """ | ||||||
|  |         self.tree.symlinks_create(packages) | ||||||
|  |  | ||||||
|  |     def on_start(self) -> None: | ||||||
|  |         """ | ||||||
|  |         trigger action which will be called at the start of the application | ||||||
|  |         """ | ||||||
|  |         self.tree.tree_create() | ||||||
|  |  | ||||||
|  |     def on_stop(self) -> None: | ||||||
|  |         """ | ||||||
|  |         trigger action which will be called before the stop of the application | ||||||
|  |         """ | ||||||
|  |         self.tree.symlinks_fix() | ||||||
| @ -17,7 +17,9 @@ | |||||||
| # 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/>. | ||||||
| # | # | ||||||
|  | # pylint: disable=too-many-public-methods | ||||||
| import configparser | import configparser | ||||||
|  | import os | ||||||
| import shlex | import shlex | ||||||
| import sys | import sys | ||||||
|  |  | ||||||
| @ -41,7 +43,6 @@ class Configuration(configparser.RawConfigParser): | |||||||
|         SYSTEM_CONFIGURATION_PATH(Path): (class attribute) default system configuration path distributed by package |         SYSTEM_CONFIGURATION_PATH(Path): (class attribute) default system configuration path distributed by package | ||||||
|         includes(list[Path]): list of includes which were read |         includes(list[Path]): list of includes which were read | ||||||
|         path(Path | None): path to root configuration file |         path(Path | None): path to root configuration file | ||||||
|         repository_id(RepositoryId | None): repository unique identifier |  | ||||||
|  |  | ||||||
|     Examples: |     Examples: | ||||||
|         Configuration class provides additional method in order to handle application configuration. Since this class is |         Configuration class provides additional method in order to handle application configuration. Since this class is | ||||||
| @ -85,13 +86,14 @@ class Configuration(configparser.RawConfigParser): | |||||||
|             empty_lines_in_values=not allow_multi_key, |             empty_lines_in_values=not allow_multi_key, | ||||||
|             interpolation=ShellInterpolator(), |             interpolation=ShellInterpolator(), | ||||||
|             converters={ |             converters={ | ||||||
|  |                 "intlist": lambda value: list(map(int, shlex.split(value))), | ||||||
|                 "list": shlex.split, |                 "list": shlex.split, | ||||||
|                 "path": self._convert_path, |                 "path": self._convert_path, | ||||||
|                 "pathlist": lambda value: [self._convert_path(element) for element in shlex.split(value)], |                 "pathlist": lambda value: list(map(self._convert_path, shlex.split(value))), | ||||||
|             }, |             }, | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         self.repository_id: RepositoryId | None = None |         self._repository_id: RepositoryId | None = None | ||||||
|         self.path: Path | None = None |         self.path: Path | None = None | ||||||
|         self.includes: list[Path] = [] |         self.includes: list[Path] = [] | ||||||
|  |  | ||||||
| @ -126,6 +128,32 @@ class Configuration(configparser.RawConfigParser): | |||||||
|         """ |         """ | ||||||
|         return self.getpath("settings", "logging") |         return self.getpath("settings", "logging") | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def repository_id(self) -> RepositoryId | None: | ||||||
|  |         """ | ||||||
|  |         repository identifier | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             RepositoryId: repository unique identifier | ||||||
|  |         """ | ||||||
|  |         return self._repository_id | ||||||
|  |  | ||||||
|  |     @repository_id.setter | ||||||
|  |     def repository_id(self, repository_id: RepositoryId | None) -> None: | ||||||
|  |         """ | ||||||
|  |         setter for repository identifier | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             repository_id(RepositoryId | None): repository unique identifier | ||||||
|  |         """ | ||||||
|  |         self._repository_id = repository_id | ||||||
|  |         if repository_id is None or repository_id.is_empty: | ||||||
|  |             self.remove_option("repository", "name") | ||||||
|  |             self.remove_option("repository", "architecture") | ||||||
|  |         else: | ||||||
|  |             self.set_option("repository", "name", repository_id.name) | ||||||
|  |             self.set_option("repository", "architecture", repository_id.architecture) | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def repository_name(self) -> str: |     def repository_name(self) -> str: | ||||||
|         """ |         """ | ||||||
| @ -162,6 +190,7 @@ class Configuration(configparser.RawConfigParser): | |||||||
|         """ |         """ | ||||||
|         configuration = cls() |         configuration = cls() | ||||||
|         configuration.load(path) |         configuration.load(path) | ||||||
|  |         configuration.load_environment() | ||||||
|         configuration.merge_sections(repository_id) |         configuration.merge_sections(repository_id) | ||||||
|         return configuration |         return configuration | ||||||
|  |  | ||||||
| @ -236,6 +265,8 @@ class Configuration(configparser.RawConfigParser): | |||||||
|  |  | ||||||
|     # pylint and mypy are too stupid to find these methods |     # pylint and mypy are too stupid to find these methods | ||||||
|     # pylint: disable=missing-function-docstring,unused-argument |     # pylint: disable=missing-function-docstring,unused-argument | ||||||
|  |     def getintlist(self, *args: Any, **kwargs: Any) -> list[int]: ...  # type: ignore[empty-body] | ||||||
|  |  | ||||||
|     def getlist(self, *args: Any, **kwargs: Any) -> list[str]: ...  # type: ignore[empty-body] |     def getlist(self, *args: Any, **kwargs: Any) -> list[str]: ...  # type: ignore[empty-body] | ||||||
|  |  | ||||||
|     def getpath(self, *args: Any, **kwargs: Any) -> Path: ...  # type: ignore[empty-body] |     def getpath(self, *args: Any, **kwargs: Any) -> Path: ...  # type: ignore[empty-body] | ||||||
| @ -284,6 +315,16 @@ class Configuration(configparser.RawConfigParser): | |||||||
|         self.read(self.path) |         self.read(self.path) | ||||||
|         self.load_includes()  # load includes |         self.load_includes()  # load includes | ||||||
|  |  | ||||||
|  |     def load_environment(self) -> None: | ||||||
|  |         """ | ||||||
|  |         load environment variables into configuration | ||||||
|  |         """ | ||||||
|  |         for name, value in os.environ.items(): | ||||||
|  |             if ":" not in name: | ||||||
|  |                 continue | ||||||
|  |             section, key = name.rsplit(":", maxsplit=1) | ||||||
|  |             self.set_option(section, key, value) | ||||||
|  |  | ||||||
|     def load_includes(self, path: Path | None = None) -> None: |     def load_includes(self, path: Path | None = None) -> None: | ||||||
|         """ |         """ | ||||||
|         load configuration includes from specified path |         load configuration includes from specified path | ||||||
| @ -352,11 +393,16 @@ class Configuration(configparser.RawConfigParser): | |||||||
|         """ |         """ | ||||||
|         reload configuration if possible or raise exception otherwise |         reload configuration if possible or raise exception otherwise | ||||||
|         """ |         """ | ||||||
|  |         # get current properties and validate input | ||||||
|         path, repository_id = self.check_loaded() |         path, repository_id = self.check_loaded() | ||||||
|         for section in self.sections():  # clear current content |  | ||||||
|  |         # clear current content | ||||||
|  |         for section in self.sections(): | ||||||
|             self.remove_section(section) |             self.remove_section(section) | ||||||
|         self.load(path) |  | ||||||
|         self.merge_sections(repository_id) |         # create another instance and copy values from there | ||||||
|  |         instance = self.from_path(path, repository_id) | ||||||
|  |         self.copy_from(instance) | ||||||
|  |  | ||||||
|     def set_option(self, section: str, option: str, value: str) -> None: |     def set_option(self, section: str, option: str, value: str) -> None: | ||||||
|         """ |         """ | ||||||
|  | |||||||
| @ -45,11 +45,6 @@ CONFIGURATION_SCHEMA: ConfigurationSchema = { | |||||||
|                 "path_exists": True, |                 "path_exists": True, | ||||||
|                 "path_type": "dir", |                 "path_type": "dir", | ||||||
|             }, |             }, | ||||||
|             "keep_last_logs": { |  | ||||||
|                 "type": "integer", |  | ||||||
|                 "coerce": "integer", |  | ||||||
|                 "min": 0, |  | ||||||
|             }, |  | ||||||
|             "logging": { |             "logging": { | ||||||
|                 "type": "path", |                 "type": "path", | ||||||
|                 "coerce": "absolute_path", |                 "coerce": "absolute_path", | ||||||
| @ -254,6 +249,10 @@ CONFIGURATION_SCHEMA: ConfigurationSchema = { | |||||||
|     "repository": { |     "repository": { | ||||||
|         "type": "dict", |         "type": "dict", | ||||||
|         "schema": { |         "schema": { | ||||||
|  |             "architecture": { | ||||||
|  |                 "type": "string", | ||||||
|  |                 "empty": False, | ||||||
|  |             }, | ||||||
|             "name": { |             "name": { | ||||||
|                 "type": "string", |                 "type": "string", | ||||||
|                 "empty": False, |                 "empty": False, | ||||||
| @ -324,6 +323,15 @@ CONFIGURATION_SCHEMA: ConfigurationSchema = { | |||||||
|                 "empty": False, |                 "empty": False, | ||||||
|                 "is_url": ["http", "https"], |                 "is_url": ["http", "https"], | ||||||
|             }, |             }, | ||||||
|  |             "autorefresh_intervals": { | ||||||
|  |                 "type": "list", | ||||||
|  |                 "coerce": "list", | ||||||
|  |                 "schema": { | ||||||
|  |                     "type": "integer", | ||||||
|  |                     "coerce": "integer", | ||||||
|  |                     "min": 0, | ||||||
|  |                 }, | ||||||
|  |             }, | ||||||
|             "enable_archive_upload": { |             "enable_archive_upload": { | ||||||
|                 "type": "boolean", |                 "type": "boolean", | ||||||
|                 "coerce": "boolean", |                 "coerce": "boolean", | ||||||
|  | |||||||
							
								
								
									
										84
									
								
								src/ahriman/core/database/migrations/m016_archive.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								src/ahriman/core/database/migrations/m016_archive.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,84 @@ | |||||||
|  | # | ||||||
|  | # Copyright (c) 2021-2025 ahriman team. | ||||||
|  | # | ||||||
|  | # This file is part of ahriman | ||||||
|  | # (see https://github.com/arcan1s/ahriman). | ||||||
|  | # | ||||||
|  | # This program is free software: you can redistribute it and/or modify | ||||||
|  | # it under the terms of the GNU General Public License as published by | ||||||
|  | # the Free Software Foundation, either version 3 of the License, or | ||||||
|  | # (at your option) any later version. | ||||||
|  | # | ||||||
|  | # This program is distributed in the hope that it will be useful, | ||||||
|  | # but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  | # GNU General Public License for more details. | ||||||
|  | # | ||||||
|  | # You should have received a copy of the GNU General Public License | ||||||
|  | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
|  | # | ||||||
|  | import argparse | ||||||
|  |  | ||||||
|  | from dataclasses import replace | ||||||
|  | from sqlite3 import Connection | ||||||
|  |  | ||||||
|  | from ahriman.application.handlers.handler import Handler | ||||||
|  | from ahriman.core.alpm.pacman import Pacman | ||||||
|  | from ahriman.core.configuration import Configuration | ||||||
|  | from ahriman.models.package import Package | ||||||
|  | from ahriman.models.pacman_synchronization import PacmanSynchronization | ||||||
|  | from ahriman.models.repository_paths import RepositoryPaths | ||||||
|  |  | ||||||
|  |  | ||||||
|  | __all__ = ["migrate_data"] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def migrate_data(connection: Connection, configuration: Configuration) -> None: | ||||||
|  |     """ | ||||||
|  |     perform data migration | ||||||
|  |  | ||||||
|  |     Args: | ||||||
|  |         connection(Connection): database connection | ||||||
|  |         configuration(Configuration): configuration instance | ||||||
|  |     """ | ||||||
|  |     del connection | ||||||
|  |  | ||||||
|  |     config_path, _ = configuration.check_loaded() | ||||||
|  |     args = argparse.Namespace(configuration=config_path, architecture=None, repository=None, repository_id=None) | ||||||
|  |  | ||||||
|  |     for repository_id in Handler.repositories_extract(args): | ||||||
|  |         paths = replace(configuration.repository_paths, repository_id=repository_id) | ||||||
|  |         pacman = Pacman(repository_id, configuration, refresh_database=PacmanSynchronization.Disabled) | ||||||
|  |  | ||||||
|  |         # create archive directory if required | ||||||
|  |         if not paths.archive.is_dir(): | ||||||
|  |             with paths.preserve_owner(paths.archive): | ||||||
|  |                 paths.archive.mkdir(mode=0o755, parents=True) | ||||||
|  |  | ||||||
|  |         move_packages(paths, pacman) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def move_packages(repository_paths: RepositoryPaths, pacman: Pacman) -> None: | ||||||
|  |     """ | ||||||
|  |     move packages from repository to archive and create symbolic links | ||||||
|  |  | ||||||
|  |     Args: | ||||||
|  |         repository_paths(RepositoryPaths): repository paths instance | ||||||
|  |         pacman(Pacman): alpm wrapper instance | ||||||
|  |     """ | ||||||
|  |     for source in repository_paths.repository.iterdir(): | ||||||
|  |         if not source.is_file(follow_symlinks=False): | ||||||
|  |             continue  # skip symbolic links if any | ||||||
|  |  | ||||||
|  |         filename = source.name | ||||||
|  |         if filename.startswith(".") or ".pkg." not in filename: | ||||||
|  |             # we don't use package_like method here, because it also filters out signatures | ||||||
|  |             continue | ||||||
|  |         package = Package.from_archive(source, pacman) | ||||||
|  |  | ||||||
|  |         # move package to the archive directory | ||||||
|  |         target = repository_paths.archive_for(package.base) / filename | ||||||
|  |         source.rename(target) | ||||||
|  |  | ||||||
|  |         # create symlink to the archive | ||||||
|  |         source.symlink_to(target.relative_to(source.parent, walk_up=True)) | ||||||
| @ -29,13 +29,15 @@ class LogsOperations(Operations): | |||||||
|     logs operations |     logs operations | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     def logs_get(self, package_base: str, limit: int = -1, offset: int = 0, |     def logs_get(self, package_base: str, version: str | None = None, process_id: str | None = None, | ||||||
|                  repository_id: RepositoryId | None = None) -> list[LogRecord]: |                  limit: int = -1, offset: int = 0, repository_id: RepositoryId | None = None) -> list[LogRecord]: | ||||||
|         """ |         """ | ||||||
|         extract logs for specified package base |         extract logs for specified package base | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             package_base(str): package base to extract logs |             package_base(str): package base to extract logs | ||||||
|  |             version(str | None, optional): package version to filter (Default value = None) | ||||||
|  |             process_id(str | None, optional): process identifier to filter (Default value = None) | ||||||
|             limit(int, optional): limit records to the specified count, -1 means unlimited (Default value = -1) |             limit(int, optional): limit records to the specified count, -1 means unlimited (Default value = -1) | ||||||
|             offset(int, optional): records offset (Default value = 0) |             offset(int, optional): records offset (Default value = 0) | ||||||
|             repository_id(RepositoryId, optional): repository unique identifier override (Default value = None) |             repository_id(RepositoryId, optional): repository unique identifier override (Default value = None) | ||||||
| @ -52,12 +54,17 @@ class LogsOperations(Operations): | |||||||
|                     """ |                     """ | ||||||
|                     select created, message, version, process_id from ( |                     select created, message, version, process_id from ( | ||||||
|                         select * from logs |                         select * from logs | ||||||
|                         where package_base = :package_base and repository = :repository |                         where package_base = :package_base | ||||||
|  |                           and repository = :repository | ||||||
|  |                           and (:version is null or version = :version) | ||||||
|  |                           and (:process_id is null or process_id = :process_id) | ||||||
|                         order by created desc limit :limit offset :offset |                         order by created desc limit :limit offset :offset | ||||||
|                     ) order by created asc |                     ) order by created asc | ||||||
|                     """, |                     """, | ||||||
|                     { |                     { | ||||||
|                         "package_base": package_base, |                         "package_base": package_base, | ||||||
|  |                         "version": version, | ||||||
|  |                         "process_id": process_id, | ||||||
|                         "repository": repository_id.id, |                         "repository": repository_id.id, | ||||||
|                         "limit": limit, |                         "limit": limit, | ||||||
|                         "offset": offset, |                         "offset": offset, | ||||||
|  | |||||||
| @ -25,8 +25,16 @@ from typing import Self | |||||||
|  |  | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.core.database.migrations import Migrations | from ahriman.core.database.migrations import Migrations | ||||||
| from ahriman.core.database.operations import AuthOperations, BuildOperations, ChangesOperations, \ | from ahriman.core.database.operations import ( | ||||||
|     DependenciesOperations, EventOperations, LogsOperations, PackageOperations, PatchOperations |     AuthOperations, | ||||||
|  |     BuildOperations, | ||||||
|  |     ChangesOperations, | ||||||
|  |     DependenciesOperations, | ||||||
|  |     EventOperations, | ||||||
|  |     LogsOperations, | ||||||
|  |     PackageOperations, | ||||||
|  |     PatchOperations, | ||||||
|  | ) | ||||||
| from ahriman.models.repository_id import RepositoryId | from ahriman.models.repository_id import RepositoryId | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -94,9 +102,13 @@ class SQLite( | |||||||
|         sqlite3.register_adapter(list, json.dumps) |         sqlite3.register_adapter(list, json.dumps) | ||||||
|         sqlite3.register_converter("json", json.loads) |         sqlite3.register_converter("json", json.loads) | ||||||
|  |  | ||||||
|         if self._configuration.getboolean("settings", "apply_migrations", fallback=True): |         if not self._configuration.getboolean("settings", "apply_migrations", fallback=True): | ||||||
|  |             return | ||||||
|  |         if self._repository_id.is_empty: | ||||||
|  |             return  # do not perform migration on empty repository identifier (e.g. multirepo command) | ||||||
|  |  | ||||||
|  |         with self._repository_paths.preserve_owner(): | ||||||
|             self.with_connection(lambda connection: Migrations.migrate(connection, self._configuration)) |             self.with_connection(lambda connection: Migrations.migrate(connection, self._configuration)) | ||||||
|         self._repository_paths.chown(self.path) |  | ||||||
|  |  | ||||||
|     def package_clear(self, package_base: str, repository_id: RepositoryId | None = None) -> None: |     def package_clear(self, package_base: str, repository_id: RepositoryId | None = None) -> None: | ||||||
|         """ |         """ | ||||||
|  | |||||||
							
								
								
									
										21
									
								
								src/ahriman/core/housekeeping/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								src/ahriman/core/housekeeping/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,21 @@ | |||||||
|  | # | ||||||
|  | # Copyright (c) 2021-2025 ahriman team. | ||||||
|  | # | ||||||
|  | # This file is part of ahriman | ||||||
|  | # (see https://github.com/arcan1s/ahriman). | ||||||
|  | # | ||||||
|  | # This program is free software: you can redistribute it and/or modify | ||||||
|  | # it under the terms of the GNU General Public License as published by | ||||||
|  | # the Free Software Foundation, either version 3 of the License, or | ||||||
|  | # (at your option) any later version. | ||||||
|  | # | ||||||
|  | # This program is distributed in the hope that it will be useful, | ||||||
|  | # but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  | # GNU General Public License for more details. | ||||||
|  | # | ||||||
|  | # You should have received a copy of the GNU General Public License | ||||||
|  | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
|  | # | ||||||
|  | from ahriman.core.housekeeping.archive_rotation_trigger import ArchiveRotationTrigger | ||||||
|  | from ahriman.core.housekeeping.logs_rotation_trigger import LogsRotationTrigger | ||||||
							
								
								
									
										115
									
								
								src/ahriman/core/housekeeping/archive_rotation_trigger.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										115
									
								
								src/ahriman/core/housekeeping/archive_rotation_trigger.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,115 @@ | |||||||
|  | # | ||||||
|  | # Copyright (c) 2021-2025 ahriman team. | ||||||
|  | # | ||||||
|  | # This file is part of ahriman | ||||||
|  | # (see https://github.com/arcan1s/ahriman). | ||||||
|  | # | ||||||
|  | # This program is free software: you can redistribute it and/or modify | ||||||
|  | # it under the terms of the GNU General Public License as published by | ||||||
|  | # the Free Software Foundation, either version 3 of the License, or | ||||||
|  | # (at your option) any later version. | ||||||
|  | # | ||||||
|  | # This program is distributed in the hope that it will be useful, | ||||||
|  | # but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  | # GNU General Public License for more details. | ||||||
|  | # | ||||||
|  | # You should have received a copy of the GNU General Public License | ||||||
|  | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
|  | # | ||||||
|  | from collections.abc import Callable | ||||||
|  | from functools import cmp_to_key | ||||||
|  |  | ||||||
|  | from ahriman.core import context | ||||||
|  | from ahriman.core.alpm.pacman import Pacman | ||||||
|  | from ahriman.core.configuration import Configuration | ||||||
|  | from ahriman.core.triggers import Trigger | ||||||
|  | from ahriman.core.utils import package_like | ||||||
|  | from ahriman.models.package import Package | ||||||
|  | from ahriman.models.repository_id import RepositoryId | ||||||
|  | from ahriman.models.result import Result | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ArchiveRotationTrigger(Trigger): | ||||||
|  |     """ | ||||||
|  |     remove packages from archive | ||||||
|  |  | ||||||
|  |     Attributes: | ||||||
|  |         keep_built_packages(int): number of last packages to keep | ||||||
|  |         paths(RepositoryPaths): repository paths instance | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     CONFIGURATION_SCHEMA = { | ||||||
|  |         "archive": { | ||||||
|  |             "type": "dict", | ||||||
|  |             "schema": { | ||||||
|  |                 "keep_built_packages": { | ||||||
|  |                     "type": "integer", | ||||||
|  |                     "required": True, | ||||||
|  |                     "coerce": "integer", | ||||||
|  |                     "min": 0, | ||||||
|  |                 }, | ||||||
|  |             }, | ||||||
|  |         }, | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     def __init__(self, repository_id: RepositoryId, configuration: Configuration) -> None: | ||||||
|  |         """ | ||||||
|  |         Args: | ||||||
|  |             repository_id(RepositoryId): repository unique identifier | ||||||
|  |             configuration(Configuration): configuration instance | ||||||
|  |         """ | ||||||
|  |         Trigger.__init__(self, repository_id, configuration) | ||||||
|  |  | ||||||
|  |         section = next(iter(self.configuration_sections(configuration))) | ||||||
|  |         self.keep_built_packages = max(configuration.getint(section, "keep_built_packages"), 0) | ||||||
|  |         self.paths = configuration.repository_paths | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def configuration_sections(cls, configuration: Configuration) -> list[str]: | ||||||
|  |         """ | ||||||
|  |         extract configuration sections from configuration | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             configuration(Configuration): configuration instance | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             list[str]: read configuration sections belong to this trigger | ||||||
|  |         """ | ||||||
|  |         return list(cls.CONFIGURATION_SCHEMA.keys()) | ||||||
|  |  | ||||||
|  |     def archives_remove(self, package: Package, pacman: Pacman) -> None: | ||||||
|  |         """ | ||||||
|  |         remove older versions of the specified package | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             package(Package): package which has been updated to check for older versions | ||||||
|  |             pacman(Pacman): alpm wrapper instance | ||||||
|  |         """ | ||||||
|  |         packages: dict[tuple[str, str], Package] = {} | ||||||
|  |         # we can't use here load_archives, because it ignores versions | ||||||
|  |         for full_path in filter(package_like, self.paths.archive_for(package.base).iterdir()): | ||||||
|  |             local = Package.from_archive(full_path, pacman) | ||||||
|  |             packages.setdefault((local.base, local.version), local).packages.update(local.packages) | ||||||
|  |  | ||||||
|  |         comparator: Callable[[Package, Package], int] = lambda left, right: left.vercmp(right.version) | ||||||
|  |         to_remove = sorted(packages.values(), key=cmp_to_key(comparator)) | ||||||
|  |         for single in to_remove[:-self.keep_built_packages]: | ||||||
|  |             self.logger.info("removing version %s of package %s", single.version, single.base) | ||||||
|  |             for archive in single.packages.values(): | ||||||
|  |                 for path in self.paths.archive_for(single.base).glob(f"{archive.filename}*"): | ||||||
|  |                     path.unlink() | ||||||
|  |  | ||||||
|  |     def on_result(self, result: Result, packages: list[Package]) -> None: | ||||||
|  |         """ | ||||||
|  |         run trigger | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             result(Result): build result | ||||||
|  |             packages(list[Package]): list of all available packages | ||||||
|  |         """ | ||||||
|  |         ctx = context.get() | ||||||
|  |         pacman = ctx.get(Pacman) | ||||||
|  |  | ||||||
|  |         for package in result.success: | ||||||
|  |             self.archives_remove(package, pacman) | ||||||
							
								
								
									
										87
									
								
								src/ahriman/core/housekeeping/logs_rotation_trigger.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								src/ahriman/core/housekeeping/logs_rotation_trigger.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,87 @@ | |||||||
|  | # | ||||||
|  | # Copyright (c) 2021-2025 ahriman team. | ||||||
|  | # | ||||||
|  | # This file is part of ahriman | ||||||
|  | # (see https://github.com/arcan1s/ahriman). | ||||||
|  | # | ||||||
|  | # This program is free software: you can redistribute it and/or modify | ||||||
|  | # it under the terms of the GNU General Public License as published by | ||||||
|  | # the Free Software Foundation, either version 3 of the License, or | ||||||
|  | # (at your option) any later version. | ||||||
|  | # | ||||||
|  | # This program is distributed in the hope that it will be useful, | ||||||
|  | # but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  | # GNU General Public License for more details. | ||||||
|  | # | ||||||
|  | # You should have received a copy of the GNU General Public License | ||||||
|  | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
|  | # | ||||||
|  | from ahriman.core import context | ||||||
|  | from ahriman.core.configuration import Configuration | ||||||
|  | from ahriman.core.status import Client | ||||||
|  | from ahriman.core.triggers import Trigger | ||||||
|  | from ahriman.models.package import Package | ||||||
|  | from ahriman.models.repository_id import RepositoryId | ||||||
|  | from ahriman.models.result import Result | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class LogsRotationTrigger(Trigger): | ||||||
|  |     """ | ||||||
|  |     rotate logs after build processes | ||||||
|  |  | ||||||
|  |     Attributes: | ||||||
|  |         keep_last_records(int): number of last records to keep | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     CONFIGURATION_SCHEMA = { | ||||||
|  |         "logs-rotation": { | ||||||
|  |             "type": "dict", | ||||||
|  |             "schema": { | ||||||
|  |                 "keep_last_logs": { | ||||||
|  |                     "type": "integer", | ||||||
|  |                     "required": True, | ||||||
|  |                     "coerce": "integer", | ||||||
|  |                     "min": 0, | ||||||
|  |                 }, | ||||||
|  |             }, | ||||||
|  |         }, | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     def __init__(self, repository_id: RepositoryId, configuration: Configuration) -> None: | ||||||
|  |         """ | ||||||
|  |         Args: | ||||||
|  |             repository_id(RepositoryId): repository unique identifier | ||||||
|  |             configuration(Configuration): configuration instance | ||||||
|  |         """ | ||||||
|  |         Trigger.__init__(self, repository_id, configuration) | ||||||
|  |  | ||||||
|  |         section = next(iter(self.configuration_sections(configuration))) | ||||||
|  |         self.keep_last_records = configuration.getint(  # read old-style first and then fallback to new style | ||||||
|  |             "settings", "keep_last_logs", | ||||||
|  |             fallback=configuration.getint(section, "keep_last_logs")) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def configuration_sections(cls, configuration: Configuration) -> list[str]: | ||||||
|  |         """ | ||||||
|  |         extract configuration sections from configuration | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             configuration(Configuration): configuration instance | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             list[str]: read configuration sections belong to this trigger | ||||||
|  |         """ | ||||||
|  |         return list(cls.CONFIGURATION_SCHEMA.keys()) | ||||||
|  |  | ||||||
|  |     def on_result(self, result: Result, packages: list[Package]) -> None: | ||||||
|  |         """ | ||||||
|  |         run trigger | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             result(Result): build result | ||||||
|  |             packages(list[Package]): list of all available packages | ||||||
|  |         """ | ||||||
|  |         ctx = context.get() | ||||||
|  |         reporter = ctx.get(Client) | ||||||
|  |         reporter.logs_rotate(self.keep_last_records) | ||||||
| @ -18,6 +18,7 @@ | |||||||
| # 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 | import requests | ||||||
|  | import sys | ||||||
|  |  | ||||||
| from functools import cached_property | from functools import cached_property | ||||||
| from typing import Any, IO, Literal | from typing import Any, IO, Literal | ||||||
| @ -70,7 +71,10 @@ class SyncHttpClient(LazyLogging): | |||||||
|             request.Session: created session object |             request.Session: created session object | ||||||
|         """ |         """ | ||||||
|         session = requests.Session() |         session = requests.Session() | ||||||
|         session.headers["User-Agent"] = f"ahriman/{__version__}" |         python_version = ".".join(map(str, sys.version_info[:3]))  # just major.minor.patch | ||||||
|  |         session.headers["User-Agent"] = f"ahriman/{__version__} " \ | ||||||
|  |             f"{requests.utils.default_user_agent()} " \ | ||||||
|  |             f"python/{python_version}" | ||||||
|  |  | ||||||
|         return session |         return session | ||||||
|  |  | ||||||
|  | |||||||
| @ -17,7 +17,6 @@ | |||||||
| # 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 atexit |  | ||||||
| import logging | import logging | ||||||
| import uuid | import uuid | ||||||
|  |  | ||||||
| @ -37,7 +36,6 @@ class HttpLogHandler(logging.Handler): | |||||||
|     method |     method | ||||||
|  |  | ||||||
|     Attributes: |     Attributes: | ||||||
|         keep_last_records(int): number of last records to keep |  | ||||||
|         reporter(Client): build status reporter instance |         reporter(Client): build status reporter instance | ||||||
|         suppress_errors(bool): suppress logging errors (e.g. if no web server available) |         suppress_errors(bool): suppress logging errors (e.g. if no web server available) | ||||||
|     """ |     """ | ||||||
| @ -56,7 +54,6 @@ class HttpLogHandler(logging.Handler): | |||||||
|  |  | ||||||
|         self.reporter = Client.load(repository_id, configuration, report=report) |         self.reporter = Client.load(repository_id, configuration, report=report) | ||||||
|         self.suppress_errors = suppress_errors |         self.suppress_errors = suppress_errors | ||||||
|         self.keep_last_records = configuration.getint("settings", "keep_last_logs", fallback=0) |  | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def load(cls, repository_id: RepositoryId, configuration: Configuration, *, report: bool) -> Self: |     def load(cls, repository_id: RepositoryId, configuration: Configuration, *, report: bool) -> Self: | ||||||
| @ -83,7 +80,6 @@ class HttpLogHandler(logging.Handler): | |||||||
|         root.addHandler(handler) |         root.addHandler(handler) | ||||||
|  |  | ||||||
|         LogRecordId.DEFAULT_PROCESS_ID = str(uuid.uuid4())  # assign default process identifier for log records |         LogRecordId.DEFAULT_PROCESS_ID = str(uuid.uuid4())  # assign default process identifier for log records | ||||||
|         atexit.register(handler.rotate) |  | ||||||
|  |  | ||||||
|         return handler |         return handler | ||||||
|  |  | ||||||
| @ -104,9 +100,3 @@ class HttpLogHandler(logging.Handler): | |||||||
|             if self.suppress_errors: |             if self.suppress_errors: | ||||||
|                 return |                 return | ||||||
|             self.handleError(record) |             self.handleError(record) | ||||||
|  |  | ||||||
|     def rotate(self) -> None: |  | ||||||
|         """ |  | ||||||
|         rotate log records, removing older ones |  | ||||||
|         """ |  | ||||||
|         self.reporter.logs_rotate(self.keep_last_records) |  | ||||||
|  | |||||||
| @ -26,6 +26,7 @@ from typing import Any | |||||||
|  |  | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.core.sign.gpg import GPG | from ahriman.core.sign.gpg import GPG | ||||||
|  | from ahriman.core.types import Comparable | ||||||
| from ahriman.core.utils import pretty_datetime, pretty_size, utcnow | from ahriman.core.utils import pretty_datetime, pretty_size, utcnow | ||||||
| from ahriman.models.repository_id import RepositoryId | from ahriman.models.repository_id import RepositoryId | ||||||
| from ahriman.models.result import Result | from ahriman.models.result import Result | ||||||
| @ -111,7 +112,7 @@ class JinjaTemplate: | |||||||
|         Returns: |         Returns: | ||||||
|             list[dict[str, str]]: sorted content according to comparator defined |             list[dict[str, str]]: sorted content according to comparator defined | ||||||
|         """ |         """ | ||||||
|         comparator: Callable[[dict[str, str]], str] = lambda item: item["filename"] |         comparator: Callable[[dict[str, str]], Comparable] = lambda item: item["filename"] | ||||||
|         return sorted(content, key=comparator) |         return sorted(content, key=comparator) | ||||||
|  |  | ||||||
|     def make_html(self, result: Result, template_name: Path | str) -> str: |     def make_html(self, result: Result, template_name: Path | str) -> str: | ||||||
|  | |||||||
| @ -28,6 +28,7 @@ from ahriman.core.configuration import Configuration | |||||||
| from ahriman.core.report.jinja_template import JinjaTemplate | from ahriman.core.report.jinja_template import JinjaTemplate | ||||||
| from ahriman.core.report.report import Report | from ahriman.core.report.report import Report | ||||||
| from ahriman.core.status import Client | from ahriman.core.status import Client | ||||||
|  | from ahriman.core.types import Comparable | ||||||
| from ahriman.models.event import EventType | from ahriman.models.event import EventType | ||||||
| from ahriman.models.package import Package | from ahriman.models.package import Package | ||||||
| from ahriman.models.repository_id import RepositoryId | from ahriman.models.repository_id import RepositoryId | ||||||
| @ -86,7 +87,7 @@ class RSS(Report, JinjaTemplate): | |||||||
|         Returns: |         Returns: | ||||||
|             list[dict[str, str]]: sorted content according to comparator defined |             list[dict[str, str]]: sorted content according to comparator defined | ||||||
|         """ |         """ | ||||||
|         comparator: Callable[[dict[str, str]], datetime.datetime] = \ |         comparator: Callable[[dict[str, str]], Comparable] = \ | ||||||
|             lambda item: parsedate_to_datetime(item["build_date"]) |             lambda item: parsedate_to_datetime(item["build_date"]) | ||||||
|         return sorted(content, key=comparator, reverse=True) |         return sorted(content, key=comparator, reverse=True) | ||||||
|  |  | ||||||
|  | |||||||
| @ -19,7 +19,7 @@ | |||||||
| # | # | ||||||
| import shutil | import shutil | ||||||
|  |  | ||||||
| from collections.abc import Iterable | from collections.abc import Generator, Iterable | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
| from tempfile import TemporaryDirectory | from tempfile import TemporaryDirectory | ||||||
|  |  | ||||||
| @ -27,7 +27,7 @@ from ahriman.core.build_tools.package_archive import PackageArchive | |||||||
| from ahriman.core.build_tools.task import Task | from ahriman.core.build_tools.task import Task | ||||||
| from ahriman.core.repository.cleaner import Cleaner | from ahriman.core.repository.cleaner import Cleaner | ||||||
| from ahriman.core.repository.package_info import PackageInfo | from ahriman.core.repository.package_info import PackageInfo | ||||||
| from ahriman.core.utils import safe_filename | from ahriman.core.utils import atomic_move, filelock, package_like, safe_filename | ||||||
| from ahriman.models.changes import Changes | from ahriman.models.changes import Changes | ||||||
| from ahriman.models.event import EventType | from ahriman.models.event import EventType | ||||||
| from ahriman.models.package import Package | from ahriman.models.package import Package | ||||||
| @ -41,6 +41,141 @@ class Executor(PackageInfo, Cleaner): | |||||||
|     trait for common repository update processes |     trait for common repository update processes | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|  |     def _archive_lookup(self, package: Package) -> Generator[Path, None, None]: | ||||||
|  |         """ | ||||||
|  |         check if there is a rebuilt package already | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             package(Package): package to check | ||||||
|  |  | ||||||
|  |         Yields: | ||||||
|  |             Path: list of built packages and signatures if available, empty list otherwise | ||||||
|  |         """ | ||||||
|  |         archive = self.paths.archive_for(package.base) | ||||||
|  |  | ||||||
|  |         # find all packages which have same version | ||||||
|  |         same_version = [ | ||||||
|  |             built | ||||||
|  |             for path in filter(package_like, archive.iterdir()) | ||||||
|  |             if (built := Package.from_archive(path, self.pacman)).version == package.version | ||||||
|  |         ] | ||||||
|  |         # no packages of the same version found | ||||||
|  |         if not same_version: | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         packages = [single for built in same_version for single in built.packages.values()] | ||||||
|  |         # all packages must be either any or same architecture | ||||||
|  |         if not all(single.architecture in ("any", self.architecture) for single in packages): | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         for single in packages: | ||||||
|  |             yield from archive.glob(f"{single.filename}*") | ||||||
|  |  | ||||||
|  |     def _archive_rename(self, description: PackageDescription, package_base: str) -> None: | ||||||
|  |         """ | ||||||
|  |         rename package archive removing special symbols | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             description(PackageDescription): package description | ||||||
|  |             package_base(str): package base name | ||||||
|  |         """ | ||||||
|  |         if description.filename is None: | ||||||
|  |             self.logger.warning("received empty package filename for base %s", package_base) | ||||||
|  |             return  # suppress type checking, it never can be none actually | ||||||
|  |  | ||||||
|  |         if (safe := safe_filename(description.filename)) != description.filename: | ||||||
|  |             atomic_move(self.paths.packages / description.filename, self.paths.packages / safe) | ||||||
|  |             description.filename = safe | ||||||
|  |  | ||||||
|  |     def _package_build(self, package: Package, path: Path, packager: str | None, | ||||||
|  |                        local_version: str | None) -> str | None: | ||||||
|  |         """ | ||||||
|  |         build single package | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             package(Package): package to build | ||||||
|  |             path(Path): path to directory with package files | ||||||
|  |             packager(str | None): packager identifier used for this package | ||||||
|  |             local_version(str | None): local version of the package | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             str | None: current commit sha if available | ||||||
|  |         """ | ||||||
|  |         self.reporter.set_building(package.base) | ||||||
|  |  | ||||||
|  |         task = Task(package, self.configuration, self.architecture, self.paths) | ||||||
|  |         patches = self.reporter.package_patches_get(package.base, None) | ||||||
|  |         commit_sha = task.init(path, patches, local_version) | ||||||
|  |  | ||||||
|  |         loaded_package = Package.from_build(path, self.architecture, None) | ||||||
|  |         if prebuilt := list(self._archive_lookup(loaded_package)): | ||||||
|  |             self.logger.info("using prebuilt packages for %s-%s", loaded_package.base, loaded_package.version) | ||||||
|  |             built = [] | ||||||
|  |             for artefact in prebuilt: | ||||||
|  |                 with filelock(artefact): | ||||||
|  |                     shutil.copy(artefact, path) | ||||||
|  |                 built.append(path / artefact.name) | ||||||
|  |         else: | ||||||
|  |             built = task.build(path, PACKAGER=packager) | ||||||
|  |  | ||||||
|  |         package.with_packages(built, self.pacman) | ||||||
|  |         for src in built: | ||||||
|  |             dst = self.paths.packages / src.name | ||||||
|  |             atomic_move(src, dst) | ||||||
|  |  | ||||||
|  |         return commit_sha | ||||||
|  |  | ||||||
|  |     def _package_remove(self, package_name: str, path: Path) -> None: | ||||||
|  |         """ | ||||||
|  |         remove single package from repository | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             package_name(str): package name | ||||||
|  |             path(Path): path to package archive | ||||||
|  |         """ | ||||||
|  |         try: | ||||||
|  |             self.repo.remove(package_name, path) | ||||||
|  |         except Exception: | ||||||
|  |             self.logger.exception("could not remove %s", package_name) | ||||||
|  |  | ||||||
|  |     def _package_remove_base(self, package_base: str) -> None: | ||||||
|  |         """ | ||||||
|  |         remove package base from repository | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             package_base(str): package base name: | ||||||
|  |         """ | ||||||
|  |         try: | ||||||
|  |             with self.in_event(package_base, EventType.PackageRemoved): | ||||||
|  |                 self.reporter.package_remove(package_base) | ||||||
|  |         except Exception: | ||||||
|  |             self.logger.exception("could not remove base %s", package_base) | ||||||
|  |  | ||||||
|  |     def _package_update(self, filename: str | None, package_base: str, packager_key: str | None) -> None: | ||||||
|  |         """ | ||||||
|  |         update built package in repository database | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             filename(str | None): archive filename | ||||||
|  |             package_base(str): package base name | ||||||
|  |             packager_key(str | None): packager key identifier | ||||||
|  |         """ | ||||||
|  |         if filename is None: | ||||||
|  |             self.logger.warning("received empty package filename for base %s", package_base) | ||||||
|  |             return  # suppress type checking, it never can be none actually | ||||||
|  |  | ||||||
|  |         # in theory, it might be NOT packages directory, but we suppose it is | ||||||
|  |         full_path = self.paths.packages / filename | ||||||
|  |         files = self.sign.process_sign_package(full_path, packager_key) | ||||||
|  |  | ||||||
|  |         for src in files: | ||||||
|  |             dst = self.paths.archive_for(package_base) / src.name | ||||||
|  |             atomic_move(src, dst)  # move package to archive directory | ||||||
|  |             if not (symlink := self.paths.repository / dst.name).exists(): | ||||||
|  |                 symlink.symlink_to(dst.relative_to(symlink.parent, walk_up=True))  # create link to archive | ||||||
|  |  | ||||||
|  |         self.repo.add(self.paths.repository / filename) | ||||||
|  |  | ||||||
|     def process_build(self, updates: Iterable[Package], packagers: Packagers | None = None, *, |     def process_build(self, updates: Iterable[Package], packagers: Packagers | None = None, *, | ||||||
|                       bump_pkgrel: bool = False) -> Result: |                       bump_pkgrel: bool = False) -> Result: | ||||||
|         """ |         """ | ||||||
| @ -55,21 +190,6 @@ class Executor(PackageInfo, Cleaner): | |||||||
|         Returns: |         Returns: | ||||||
|             Result: build result |             Result: build result | ||||||
|         """ |         """ | ||||||
|         def build_single(package: Package, local_path: Path, packager_id: str | None) -> str | None: |  | ||||||
|             self.reporter.set_building(package.base) |  | ||||||
|             task = Task(package, self.configuration, self.architecture, self.paths) |  | ||||||
|             local_version = local_versions.get(package.base) if bump_pkgrel else None |  | ||||||
|             patches = self.reporter.package_patches_get(package.base, None) |  | ||||||
|             commit_sha = task.init(local_path, patches, local_version) |  | ||||||
|             built = task.build(local_path, PACKAGER=packager_id) |  | ||||||
|  |  | ||||||
|             package.with_packages(built, self.pacman) |  | ||||||
|             for src in built: |  | ||||||
|                 dst = self.paths.packages / src.name |  | ||||||
|                 shutil.move(src, dst) |  | ||||||
|  |  | ||||||
|             return commit_sha |  | ||||||
|  |  | ||||||
|         packagers = packagers or Packagers() |         packagers = packagers or Packagers() | ||||||
|         local_versions = {package.base: package.version for package in self.packages()} |         local_versions = {package.base: package.version for package in self.packages()} | ||||||
|  |  | ||||||
| @ -80,16 +200,21 @@ class Executor(PackageInfo, Cleaner): | |||||||
|                 try: |                 try: | ||||||
|                     with self.in_event(single.base, EventType.PackageUpdated, failure=EventType.PackageUpdateFailed): |                     with self.in_event(single.base, EventType.PackageUpdated, failure=EventType.PackageUpdateFailed): | ||||||
|                         packager = self.packager(packagers, single.base) |                         packager = self.packager(packagers, single.base) | ||||||
|                         last_commit_sha = build_single(single, Path(dir_name), packager.packager_id) |                         local_version = local_versions.get(single.base) if bump_pkgrel else None | ||||||
|  |                         commit_sha = self._package_build(single, Path(dir_name), packager.packager_id, local_version) | ||||||
|  |  | ||||||
|                         # update commit hash for changes keeping current diff if there is any |                         # update commit hash for changes keeping current diff if there is any | ||||||
|                         changes = self.reporter.package_changes_get(single.base) |                         changes = self.reporter.package_changes_get(single.base) | ||||||
|                         self.reporter.package_changes_update(single.base, Changes(last_commit_sha, changes.changes)) |                         self.reporter.package_changes_update(single.base, Changes(commit_sha, changes.changes)) | ||||||
|  |  | ||||||
|                         # update dependencies list |                         # update dependencies list | ||||||
|                         package_archive = PackageArchive(self.paths.build_root, single, self.pacman, self.scan_paths) |                         package_archive = PackageArchive(self.paths.build_root, single, self.pacman, self.scan_paths) | ||||||
|                         dependencies = package_archive.depends_on() |                         dependencies = package_archive.depends_on() | ||||||
|                         self.reporter.package_dependencies_update(single.base, dependencies) |                         self.reporter.package_dependencies_update(single.base, dependencies) | ||||||
|  |  | ||||||
|                         # update result set |                         # update result set | ||||||
|                         result.add_updated(single) |                         result.add_updated(single) | ||||||
|  |  | ||||||
|                 except Exception: |                 except Exception: | ||||||
|                     self.reporter.set_failed(single.base) |                     self.reporter.set_failed(single.base) | ||||||
|                     result.add_failed(single) |                     result.add_failed(single) | ||||||
| @ -107,19 +232,6 @@ class Executor(PackageInfo, Cleaner): | |||||||
|         Returns: |         Returns: | ||||||
|             Result: remove result |             Result: remove result | ||||||
|         """ |         """ | ||||||
|         def remove_base(package_base: str) -> None: |  | ||||||
|             try: |  | ||||||
|                 with self.in_event(package_base, EventType.PackageRemoved): |  | ||||||
|                     self.reporter.package_remove(package_base) |  | ||||||
|             except Exception: |  | ||||||
|                 self.logger.exception("could not remove base %s", package_base) |  | ||||||
|  |  | ||||||
|         def remove_package(package: str, archive_path: Path) -> None: |  | ||||||
|             try: |  | ||||||
|                 self.repo.remove(package, archive_path)  # remove the package itself |  | ||||||
|             except Exception: |  | ||||||
|                 self.logger.exception("could not remove %s", package) |  | ||||||
|  |  | ||||||
|         packages_to_remove: dict[str, Path] = {} |         packages_to_remove: dict[str, Path] = {} | ||||||
|         bases_to_remove: list[str] = [] |         bases_to_remove: list[str] = [] | ||||||
|  |  | ||||||
| @ -136,6 +248,7 @@ class Executor(PackageInfo, Cleaner): | |||||||
|                 }) |                 }) | ||||||
|                 bases_to_remove.append(local.base) |                 bases_to_remove.append(local.base) | ||||||
|                 result.add_removed(local) |                 result.add_removed(local) | ||||||
|  |  | ||||||
|             elif requested.intersection(local.packages.keys()): |             elif requested.intersection(local.packages.keys()): | ||||||
|                 packages_to_remove.update({ |                 packages_to_remove.update({ | ||||||
|                     package: properties.filepath |                     package: properties.filepath | ||||||
| @ -152,11 +265,11 @@ class Executor(PackageInfo, Cleaner): | |||||||
|  |  | ||||||
|         # remove packages from repository files |         # remove packages from repository files | ||||||
|         for package, filename in packages_to_remove.items(): |         for package, filename in packages_to_remove.items(): | ||||||
|             remove_package(package, filename) |             self._package_remove(package, filename) | ||||||
|  |  | ||||||
|         # remove bases from registered |         # remove bases from registered | ||||||
|         for package in bases_to_remove: |         for package in bases_to_remove: | ||||||
|             remove_base(package) |             self._package_remove_base(package) | ||||||
|  |  | ||||||
|         return result |         return result | ||||||
|  |  | ||||||
| @ -172,27 +285,6 @@ class Executor(PackageInfo, Cleaner): | |||||||
|         Returns: |         Returns: | ||||||
|             Result: path to repository database |             Result: path to repository database | ||||||
|         """ |         """ | ||||||
|         def rename(archive: PackageDescription, package_base: str) -> None: |  | ||||||
|             if archive.filename is None: |  | ||||||
|                 self.logger.warning("received empty package name for base %s", package_base) |  | ||||||
|                 return  # suppress type checking, it never can be none actually |  | ||||||
|             if (safe := safe_filename(archive.filename)) != archive.filename: |  | ||||||
|                 shutil.move(self.paths.packages / archive.filename, self.paths.packages / safe) |  | ||||||
|                 archive.filename = safe |  | ||||||
|  |  | ||||||
|         def update_single(name: str | None, package_base: str, packager_key: str | None) -> None: |  | ||||||
|             if name is None: |  | ||||||
|                 self.logger.warning("received empty package name for base %s", package_base) |  | ||||||
|                 return  # suppress type checking, it never can be none actually |  | ||||||
|             # in theory, it might be NOT packages directory, but we suppose it is |  | ||||||
|             full_path = self.paths.packages / name |  | ||||||
|             files = self.sign.process_sign_package(full_path, packager_key) |  | ||||||
|             for src in files: |  | ||||||
|                 dst = self.paths.repository / safe_filename(src.name) |  | ||||||
|                 shutil.move(src, dst) |  | ||||||
|             package_path = self.paths.repository / safe_filename(name) |  | ||||||
|             self.repo.add(package_path) |  | ||||||
|  |  | ||||||
|         current_packages = {package.base: package for package in self.packages()} |         current_packages = {package.base: package for package in self.packages()} | ||||||
|         local_versions = {package_base: package.version for package_base, package in current_packages.items()} |         local_versions = {package_base: package.version for package_base, package in current_packages.items()} | ||||||
|  |  | ||||||
| @ -207,8 +299,8 @@ class Executor(PackageInfo, Cleaner): | |||||||
|                     packager = self.packager(packagers, local.base) |                     packager = self.packager(packagers, local.base) | ||||||
|  |  | ||||||
|                     for description in local.packages.values(): |                     for description in local.packages.values(): | ||||||
|                         rename(description, local.base) |                         self._archive_rename(description, local.base) | ||||||
|                         update_single(description.filename, local.base, packager.key) |                         self._package_update(description.filename, local.base, packager.key) | ||||||
|                     self.reporter.set_success(local) |                     self.reporter.set_success(local) | ||||||
|                     result.add_updated(local) |                     result.add_updated(local) | ||||||
|  |  | ||||||
| @ -216,12 +308,13 @@ class Executor(PackageInfo, Cleaner): | |||||||
|                     if local.base in current_packages: |                     if local.base in current_packages: | ||||||
|                         current_package_archives = set(current_packages[local.base].packages.keys()) |                         current_package_archives = set(current_packages[local.base].packages.keys()) | ||||||
|                     removed_packages.extend(current_package_archives.difference(local.packages)) |                     removed_packages.extend(current_package_archives.difference(local.packages)) | ||||||
|  |  | ||||||
|                 except Exception: |                 except Exception: | ||||||
|                     self.reporter.set_failed(local.base) |                     self.reporter.set_failed(local.base) | ||||||
|                     result.add_failed(local) |                     result.add_failed(local) | ||||||
|                     self.logger.exception("could not process %s", local.base) |                     self.logger.exception("could not process %s", local.base) | ||||||
|         self.clear_packages() |  | ||||||
|  |  | ||||||
|  |         self.clear_packages() | ||||||
|         self.process_remove(removed_packages) |         self.process_remove(removed_packages) | ||||||
|  |  | ||||||
|         return result |         return result | ||||||
|  | |||||||
| @ -81,6 +81,11 @@ class Client: | |||||||
|  |  | ||||||
|         return make_local_client() |         return make_local_client() | ||||||
|  |  | ||||||
|  |     def configuration_reload(self) -> None: | ||||||
|  |         """ | ||||||
|  |         reload configuration | ||||||
|  |         """ | ||||||
|  |  | ||||||
|     def event_add(self, event: Event) -> None: |     def event_add(self, event: Event) -> None: | ||||||
|         """ |         """ | ||||||
|         create new event |         create new event | ||||||
| @ -203,12 +208,15 @@ class Client: | |||||||
|         """ |         """ | ||||||
|         # this method does not raise NotImplementedError because it is actively used as dummy client for http log |         # this method does not raise NotImplementedError because it is actively used as dummy client for http log | ||||||
|  |  | ||||||
|     def package_logs_get(self, package_base: str, limit: int = -1, offset: int = 0) -> list[LogRecord]: |     def package_logs_get(self, package_base: str, version: str | None = None, process_id: str | None = None, | ||||||
|  |                          limit: int = -1, offset: int = 0) -> list[LogRecord]: | ||||||
|         """ |         """ | ||||||
|         get package logs |         get package logs | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             package_base(str): package base |             package_base(str): package base | ||||||
|  |             version(str | None, optional): package version to search (Default value = None) | ||||||
|  |             process_id(str | None, optional): process identifier to search (Default value = None) | ||||||
|             limit(int, optional): limit records to the specified count, -1 means unlimited (Default value = -1) |             limit(int, optional): limit records to the specified count, -1 means unlimited (Default value = -1) | ||||||
|             offset(int, optional): records offset (Default value = 0) |             offset(int, optional): records offset (Default value = 0) | ||||||
|  |  | ||||||
|  | |||||||
| @ -152,19 +152,22 @@ class LocalClient(Client): | |||||||
|         """ |         """ | ||||||
|         self.database.logs_insert(log_record, self.repository_id) |         self.database.logs_insert(log_record, self.repository_id) | ||||||
|  |  | ||||||
|     def package_logs_get(self, package_base: str, limit: int = -1, offset: int = 0) -> list[LogRecord]: |     def package_logs_get(self, package_base: str, version: str | None = None, process_id: str | None = None, | ||||||
|  |                          limit: int = -1, offset: int = 0) -> list[LogRecord]: | ||||||
|         """ |         """ | ||||||
|         get package logs |         get package logs | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             package_base(str): package base |             package_base(str): package base | ||||||
|  |             version(str | None, optional): package version to search (Default value = None) | ||||||
|  |             process_id(str | None, optional): process identifier to search (Default value = None) | ||||||
|             limit(int, optional): limit records to the specified count, -1 means unlimited (Default value = -1) |             limit(int, optional): limit records to the specified count, -1 means unlimited (Default value = -1) | ||||||
|             offset(int, optional): records offset (Default value = 0) |             offset(int, optional): records offset (Default value = 0) | ||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             list[LogRecord]: package logs |             list[LogRecord]: package logs | ||||||
|         """ |         """ | ||||||
|         return self.database.logs_get(package_base, limit, offset, self.repository_id) |         return self.database.logs_get(package_base, version, process_id, limit, offset, self.repository_id) | ||||||
|  |  | ||||||
|     def package_logs_remove(self, package_base: str, version: str | None) -> None: |     def package_logs_remove(self, package_base: str, version: str | None) -> None: | ||||||
|         """ |         """ | ||||||
|  | |||||||
| @ -109,7 +109,7 @@ class Watcher(LazyLogging): | |||||||
|  |  | ||||||
|     package_logs_add: Callable[[LogRecord], None] |     package_logs_add: Callable[[LogRecord], None] | ||||||
|  |  | ||||||
|     package_logs_get: Callable[[str, int, int], list[LogRecord]] |     package_logs_get: Callable[[str, str | None, str | None, int, int], list[LogRecord]] | ||||||
|  |  | ||||||
|     package_logs_remove: Callable[[str, str | None], None] |     package_logs_remove: Callable[[str, str | None], None] | ||||||
|  |  | ||||||
|  | |||||||
| @ -17,6 +17,7 @@ | |||||||
| # 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/>. | ||||||
| # | # | ||||||
|  | # pylint: disable=too-many-public-methods | ||||||
| import contextlib | import contextlib | ||||||
|  |  | ||||||
| from urllib.parse import quote_plus as url_encode | from urllib.parse import quote_plus as url_encode | ||||||
| @ -165,6 +166,13 @@ class WebClient(Client, SyncAhrimanClient): | |||||||
|         """ |         """ | ||||||
|         return f"{self.address}/api/v1/status" |         return f"{self.address}/api/v1/status" | ||||||
|  |  | ||||||
|  |     def configuration_reload(self) -> None: | ||||||
|  |         """ | ||||||
|  |         reload configuration | ||||||
|  |         """ | ||||||
|  |         with contextlib.suppress(Exception): | ||||||
|  |             self.make_request("POST", f"{self.address}/api/v1/service/config") | ||||||
|  |  | ||||||
|     def event_add(self, event: Event) -> None: |     def event_add(self, event: Event) -> None: | ||||||
|         """ |         """ | ||||||
|         create new event |         create new event | ||||||
| @ -326,12 +334,15 @@ class WebClient(Client, SyncAhrimanClient): | |||||||
|         self.make_request("POST", self._logs_url(log_record.log_record_id.package_base), |         self.make_request("POST", self._logs_url(log_record.log_record_id.package_base), | ||||||
|                           params=self.repository_id.query(), json=payload, suppress_errors=True) |                           params=self.repository_id.query(), json=payload, suppress_errors=True) | ||||||
|  |  | ||||||
|     def package_logs_get(self, package_base: str, limit: int = -1, offset: int = 0) -> list[LogRecord]: |     def package_logs_get(self, package_base: str, version: str | None = None, process_id: str | None = None, | ||||||
|  |                          limit: int = -1, offset: int = 0) -> list[LogRecord]: | ||||||
|         """ |         """ | ||||||
|         get package logs |         get package logs | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             package_base(str): package base |             package_base(str): package base | ||||||
|  |             version(str | None, optional): package version to search (Default value = None) | ||||||
|  |             process_id(str | None, optional): process identifier to search (Default value = None) | ||||||
|             limit(int, optional): limit records to the specified count, -1 means unlimited (Default value = -1) |             limit(int, optional): limit records to the specified count, -1 means unlimited (Default value = -1) | ||||||
|             offset(int, optional): records offset (Default value = 0) |             offset(int, optional): records offset (Default value = 0) | ||||||
|  |  | ||||||
| @ -339,6 +350,10 @@ class WebClient(Client, SyncAhrimanClient): | |||||||
|             list[LogRecord]: package logs |             list[LogRecord]: package logs | ||||||
|         """ |         """ | ||||||
|         query = self.repository_id.query() + [("limit", str(limit)), ("offset", str(offset))] |         query = self.repository_id.query() + [("limit", str(limit)), ("offset", str(offset))] | ||||||
|  |         if version is not None: | ||||||
|  |             query.append(("version", version)) | ||||||
|  |         if process_id is not None: | ||||||
|  |             query.append(("process_id", process_id)) | ||||||
|  |  | ||||||
|         with contextlib.suppress(Exception): |         with contextlib.suppress(Exception): | ||||||
|             response = self.make_request("GET", self._logs_url(package_base), params=query) |             response = self.make_request("GET", self._logs_url(package_base), params=query) | ||||||
|  | |||||||
| @ -33,6 +33,7 @@ class Leaf: | |||||||
|  |  | ||||||
|     Attributes: |     Attributes: | ||||||
|         dependencies(set[str]): list of package dependencies |         dependencies(set[str]): list of package dependencies | ||||||
|  |         items(list[str]): list of packages in this leaf including provides | ||||||
|         package(Package): leaf package properties |         package(Package): leaf package properties | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
| @ -42,17 +43,9 @@ class Leaf: | |||||||
|             package(Package): package properties |             package(Package): package properties | ||||||
|         """ |         """ | ||||||
|         self.package = package |         self.package = package | ||||||
|  |         # store frequently used properties | ||||||
|         self.dependencies = package.depends_build |         self.dependencies = package.depends_build | ||||||
|  |         self.items = self.package.packages_full | ||||||
|     @property |  | ||||||
|     def items(self) -> Iterable[str]: |  | ||||||
|         """ |  | ||||||
|         extract all packages from the leaf |  | ||||||
|  |  | ||||||
|         Returns: |  | ||||||
|             Iterable[str]: packages containing in this leaf |  | ||||||
|         """ |  | ||||||
|         return self.package.packages.keys() |  | ||||||
|  |  | ||||||
|     def is_dependency(self, packages: Iterable[Leaf]) -> bool: |     def is_dependency(self, packages: Iterable[Leaf]) -> bool: | ||||||
|         """ |         """ | ||||||
|  | |||||||
| @ -36,6 +36,7 @@ class Trigger(LazyLogging): | |||||||
|         CONFIGURATION_SCHEMA(ConfigurationSchema): (class attribute) configuration schema template |         CONFIGURATION_SCHEMA(ConfigurationSchema): (class attribute) configuration schema template | ||||||
|         CONFIGURATION_SCHEMA_FALLBACK(str | None): (class attribute) optional fallback option for defining |         CONFIGURATION_SCHEMA_FALLBACK(str | None): (class attribute) optional fallback option for defining | ||||||
|             configuration schema type used |             configuration schema type used | ||||||
|  |         REQUIRES_REPOSITORY(bool): (class attribute) either trigger requires loaded repository or not | ||||||
|         configuration(Configuration): configuration instance |         configuration(Configuration): configuration instance | ||||||
|         repository_id(RepositoryId): repository unique identifier |         repository_id(RepositoryId): repository unique identifier | ||||||
|  |  | ||||||
| @ -59,6 +60,7 @@ class Trigger(LazyLogging): | |||||||
|  |  | ||||||
|     CONFIGURATION_SCHEMA: ClassVar[ConfigurationSchema] = {} |     CONFIGURATION_SCHEMA: ClassVar[ConfigurationSchema] = {} | ||||||
|     CONFIGURATION_SCHEMA_FALLBACK: ClassVar[str | None] = None |     CONFIGURATION_SCHEMA_FALLBACK: ClassVar[str | None] = None | ||||||
|  |     REQUIRES_REPOSITORY: ClassVar[bool] = True | ||||||
|  |  | ||||||
|     def __init__(self, repository_id: RepositoryId, configuration: Configuration) -> None: |     def __init__(self, repository_id: RepositoryId, configuration: Configuration) -> None: | ||||||
|         """ |         """ | ||||||
| @ -79,9 +81,18 @@ class Trigger(LazyLogging): | |||||||
|         """ |         """ | ||||||
|         return self.repository_id.architecture |         return self.repository_id.architecture | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def is_allowed_to_run(self) -> bool: | ||||||
|  |         """ | ||||||
|  |         whether trigger allowed to run or not | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             bool: ``True`` in case if trigger allowed to run and ``False`` otherwise | ||||||
|  |         """ | ||||||
|  |         return not (self.REQUIRES_REPOSITORY and self.repository_id.is_empty) | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def configuration_schema(cls, repository_id: RepositoryId, |     def configuration_schema(cls, configuration: Configuration | None) -> ConfigurationSchema: | ||||||
|                              configuration: Configuration | None) -> ConfigurationSchema: |  | ||||||
|         """ |         """ | ||||||
|         configuration schema based on supplied service configuration |         configuration schema based on supplied service configuration | ||||||
|  |  | ||||||
| @ -89,7 +100,6 @@ class Trigger(LazyLogging): | |||||||
|             Schema must be in cerberus format, for details and examples you can check built-in triggers. |             Schema must be in cerberus format, for details and examples you can check built-in triggers. | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             repository_id(str): repository unique identifier |  | ||||||
|             configuration(Configuration | None): configuration instance. If set to None, the default schema |             configuration(Configuration | None): configuration instance. If set to None, the default schema | ||||||
|                 should be returned |                 should be returned | ||||||
|  |  | ||||||
| @ -101,10 +111,12 @@ class Trigger(LazyLogging): | |||||||
|  |  | ||||||
|         result: ConfigurationSchema = {} |         result: ConfigurationSchema = {} | ||||||
|         for target in cls.configuration_sections(configuration): |         for target in cls.configuration_sections(configuration): | ||||||
|             if not configuration.has_section(target): |             for section in configuration.sections(): | ||||||
|  |                 if not (section == target or section.startswith(f"{target}:")): | ||||||
|  |                     # either repository specific or exact name | ||||||
|                     continue |                     continue | ||||||
|             section, schema_name = configuration.gettype( |                 schema_name = configuration.get(section, "type", fallback=section) | ||||||
|                 target, repository_id, fallback=cls.CONFIGURATION_SCHEMA_FALLBACK) |  | ||||||
|                 if schema_name not in cls.CONFIGURATION_SCHEMA: |                 if schema_name not in cls.CONFIGURATION_SCHEMA: | ||||||
|                     continue |                     continue | ||||||
|                 result[section] = cls.CONFIGURATION_SCHEMA[schema_name] |                 result[section] = cls.CONFIGURATION_SCHEMA[schema_name] | ||||||
|  | |||||||
| @ -17,6 +17,7 @@ | |||||||
| # 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 atexit | ||||||
| import contextlib | import contextlib | ||||||
| import os | import os | ||||||
|  |  | ||||||
| @ -60,17 +61,8 @@ class TriggerLoader(LazyLogging): | |||||||
|  |  | ||||||
|     def __init__(self) -> None: |     def __init__(self) -> None: | ||||||
|         """""" |         """""" | ||||||
|         self._on_stop_requested = False |  | ||||||
|         self.triggers: list[Trigger] = [] |         self.triggers: list[Trigger] = [] | ||||||
|  |  | ||||||
|     def __del__(self) -> None: |  | ||||||
|         """ |  | ||||||
|         custom destructor object which calls on_stop in case if it was requested |  | ||||||
|         """ |  | ||||||
|         if not self._on_stop_requested: |  | ||||||
|             return |  | ||||||
|         self.on_stop() |  | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def load(cls, repository_id: RepositoryId, configuration: Configuration) -> Self: |     def load(cls, repository_id: RepositoryId, configuration: Configuration) -> Self: | ||||||
|         """ |         """ | ||||||
| @ -85,8 +77,9 @@ class TriggerLoader(LazyLogging): | |||||||
|         """ |         """ | ||||||
|         instance = cls() |         instance = cls() | ||||||
|         instance.triggers = [ |         instance.triggers = [ | ||||||
|             instance.load_trigger(trigger, repository_id, configuration) |             trigger | ||||||
|             for trigger in instance.selected_triggers(configuration) |             for trigger_name in instance.selected_triggers(configuration) | ||||||
|  |             if (trigger := instance.load_trigger(trigger_name, repository_id, configuration)).is_allowed_to_run | ||||||
|         ] |         ] | ||||||
|  |  | ||||||
|         return instance |         return instance | ||||||
| @ -250,10 +243,11 @@ class TriggerLoader(LazyLogging): | |||||||
|         run triggers on load |         run triggers on load | ||||||
|         """ |         """ | ||||||
|         self.logger.debug("executing triggers on start") |         self.logger.debug("executing triggers on start") | ||||||
|         self._on_stop_requested = True |  | ||||||
|         for trigger in self.triggers: |         for trigger in self.triggers: | ||||||
|             with self.__execute_trigger(trigger): |             with self.__execute_trigger(trigger): | ||||||
|                 trigger.on_start() |                 trigger.on_start() | ||||||
|  |         # register on_stop call | ||||||
|  |         atexit.register(self.on_stop) | ||||||
|  |  | ||||||
|     def on_stop(self) -> None: |     def on_stop(self) -> None: | ||||||
|         """ |         """ | ||||||
|  | |||||||
| @ -17,7 +17,15 @@ | |||||||
| # 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 typing import Protocol | from typing import Any, Protocol | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Comparable(Protocol): | ||||||
|  |     """ | ||||||
|  |     class which supports :func:`__lt__` operation` | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def __lt__(self, other: Any) -> bool: ... | ||||||
|  |  | ||||||
|  |  | ||||||
| class HasBool(Protocol): | class HasBool(Protocol): | ||||||
|  | |||||||
| @ -18,13 +18,16 @@ | |||||||
| # along with this program. If not, see <http://www.gnu.org/licenses/>. | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
| # | # | ||||||
| # pylint: disable=too-many-lines | # pylint: disable=too-many-lines | ||||||
|  | import contextlib | ||||||
| import datetime | import datetime | ||||||
|  | import fcntl | ||||||
| import io | import io | ||||||
| import itertools | import itertools | ||||||
| import logging | import logging | ||||||
| import os | import os | ||||||
| import re | import re | ||||||
| import selectors | import selectors | ||||||
|  | import shutil | ||||||
| import subprocess | import subprocess | ||||||
|  |  | ||||||
| from collections.abc import Callable, Generator, Iterable, Mapping | from collections.abc import Callable, Generator, Iterable, Mapping | ||||||
| @ -39,11 +42,13 @@ from ahriman.models.repository_paths import RepositoryPaths | |||||||
|  |  | ||||||
|  |  | ||||||
| __all__ = [ | __all__ = [ | ||||||
|  |     "atomic_move", | ||||||
|     "check_output", |     "check_output", | ||||||
|     "check_user", |     "check_user", | ||||||
|     "dataclass_view", |     "dataclass_view", | ||||||
|     "enum_values", |     "enum_values", | ||||||
|     "extract_user", |     "extract_user", | ||||||
|  |     "filelock", | ||||||
|     "filter_json", |     "filter_json", | ||||||
|     "full_version", |     "full_version", | ||||||
|     "minmax", |     "minmax", | ||||||
| @ -51,6 +56,7 @@ __all__ = [ | |||||||
|     "parse_version", |     "parse_version", | ||||||
|     "partition", |     "partition", | ||||||
|     "pretty_datetime", |     "pretty_datetime", | ||||||
|  |     "pretty_interval", | ||||||
|     "pretty_size", |     "pretty_size", | ||||||
|     "safe_filename", |     "safe_filename", | ||||||
|     "srcinfo_property", |     "srcinfo_property", | ||||||
| @ -64,6 +70,25 @@ __all__ = [ | |||||||
| T = TypeVar("T") | T = TypeVar("T") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def atomic_move(src: Path, dst: Path) -> None: | ||||||
|  |     """ | ||||||
|  |     move file from ``source`` location to ``destination``. This method uses lock and :func:`shutil.move` to ensure that | ||||||
|  |     file will be copied (if not rename) atomically. This method blocks execution until lock is available | ||||||
|  |  | ||||||
|  |     Args: | ||||||
|  |         src(Path): path to the source file | ||||||
|  |         dst(Path): path to the destination | ||||||
|  |  | ||||||
|  |     Examples: | ||||||
|  |         This method is a drop-in replacement for :func:`shutil.move` (except it doesn't allow to override copy method) | ||||||
|  |         which first locking destination file. To use it simply call method with arguments:: | ||||||
|  |  | ||||||
|  |             >>> atomic_move(src, dst) | ||||||
|  |     """ | ||||||
|  |     with filelock(dst): | ||||||
|  |         shutil.move(src, dst) | ||||||
|  |  | ||||||
|  |  | ||||||
| # pylint: disable=too-many-locals | # pylint: disable=too-many-locals | ||||||
| def check_output(*args: str, exception: Exception | Callable[[int, list[str], str, str], Exception] | None = None, | def check_output(*args: str, exception: Exception | Callable[[int, list[str], str, str], Exception] | None = None, | ||||||
|                  cwd: Path | None = None, input_data: str | None = None, |                  cwd: Path | None = None, input_data: str | None = None, | ||||||
| @ -136,7 +161,8 @@ def check_output(*args: str, exception: Exception | Callable[[int, list[str], st | |||||||
|     } | environment |     } | environment | ||||||
|  |  | ||||||
|     with subprocess.Popen(args, cwd=cwd, stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE, |     with subprocess.Popen(args, cwd=cwd, stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE, | ||||||
|                           user=user, env=full_environment, text=True, encoding="utf8", bufsize=1) as process: |                           user=user, env=full_environment, text=True, encoding="utf8", errors="backslashreplace", | ||||||
|  |                           bufsize=1) as process: | ||||||
|         if input_data is not None: |         if input_data is not None: | ||||||
|             input_channel = get_io(process, "stdin") |             input_channel = get_io(process, "stdin") | ||||||
|             input_channel.write(input_data) |             input_channel.write(input_data) | ||||||
| @ -231,6 +257,27 @@ def extract_user() -> str | None: | |||||||
|     return os.getenv("SUDO_USER") or os.getenv("DOAS_USER") or os.getenv("USER") |     return os.getenv("SUDO_USER") or os.getenv("DOAS_USER") or os.getenv("USER") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @contextlib.contextmanager | ||||||
|  | def filelock(path: Path) -> Generator[None, None, None]: | ||||||
|  |     """ | ||||||
|  |     lock on file passed as argument | ||||||
|  |  | ||||||
|  |     Args: | ||||||
|  |         path(Path): path object on which lock must be performed | ||||||
|  |     """ | ||||||
|  |     lock_path = path.with_name(f".{path.name}") | ||||||
|  |     try: | ||||||
|  |         with lock_path.open("ab") as lock_file: | ||||||
|  |             fd = lock_file.fileno() | ||||||
|  |             try: | ||||||
|  |                 fcntl.flock(fd, fcntl.LOCK_EX)  # lock file and wait lock is until available | ||||||
|  |                 yield | ||||||
|  |             finally: | ||||||
|  |                 fcntl.flock(fd, fcntl.LOCK_UN)  # unlock file first | ||||||
|  |     finally: | ||||||
|  |         lock_path.unlink(missing_ok=True)  # remove lock file at the end | ||||||
|  |  | ||||||
|  |  | ||||||
| def filter_json(source: dict[str, Any], known_fields: Iterable[str]) -> dict[str, Any]: | def filter_json(source: dict[str, Any], known_fields: Iterable[str]) -> dict[str, Any]: | ||||||
|     """ |     """ | ||||||
|     filter json object by fields used for json-to-object conversion |     filter json object by fields used for json-to-object conversion | ||||||
| @ -352,6 +399,28 @@ def pretty_datetime(timestamp: datetime.datetime | float | int | None) -> str: | |||||||
|     return timestamp.strftime("%Y-%m-%d %H:%M:%S") |     return timestamp.strftime("%Y-%m-%d %H:%M:%S") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def pretty_interval(interval: int) -> str: | ||||||
|  |     """ | ||||||
|  |     convert time interval to string | ||||||
|  |  | ||||||
|  |     Args: | ||||||
|  |         interval(int): time interval in seconds | ||||||
|  |  | ||||||
|  |     Returns: | ||||||
|  |         str: pretty printable interval as string | ||||||
|  |     """ | ||||||
|  |     minutes, seconds = divmod(interval, 60) | ||||||
|  |     hours, minutes = divmod(minutes, 60) | ||||||
|  |     return " ".join([ | ||||||
|  |         f"{value} {description}{"s" if value > 1 else ""}" | ||||||
|  |         for value, description in [ | ||||||
|  |             (hours, "hour"), | ||||||
|  |             (minutes, "minute"), | ||||||
|  |             (seconds, "second"), | ||||||
|  |         ] if value > 0 | ||||||
|  |     ]) | ||||||
|  |  | ||||||
|  |  | ||||||
| def pretty_size(size: float | None, level: int = 0) -> str: | def pretty_size(size: float | None, level: int = 0) -> str: | ||||||
|     """ |     """ | ||||||
|     convert size to string |     convert size to string | ||||||
|  | |||||||
| @ -25,7 +25,7 @@ from dataclasses import dataclass, field, fields | |||||||
| from pyalpm import Package  # type: ignore[import-not-found] | from pyalpm import Package  # type: ignore[import-not-found] | ||||||
| from typing import Any, Self | from typing import Any, Self | ||||||
|  |  | ||||||
| from ahriman.core.utils import filter_json, full_version | from ahriman.core.utils import filter_json, full_version, trim_package | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(frozen=True, kw_only=True) | @dataclass(frozen=True, kw_only=True) | ||||||
| @ -103,6 +103,17 @@ class AURPackage: | |||||||
|     keywords: list[str] = field(default_factory=list) |     keywords: list[str] = field(default_factory=list) | ||||||
|     groups: list[str] = field(default_factory=list) |     groups: list[str] = field(default_factory=list) | ||||||
|  |  | ||||||
|  |     def __post_init__(self) -> None: | ||||||
|  |         """ | ||||||
|  |         update packages lists accordingly | ||||||
|  |         """ | ||||||
|  |         object.__setattr__(self, "depends", [trim_package(package) for package in self.depends]) | ||||||
|  |         object.__setattr__(self, "make_depends", [trim_package(package) for package in self.make_depends]) | ||||||
|  |         object.__setattr__(self, "opt_depends", [trim_package(package) for package in self.opt_depends]) | ||||||
|  |         object.__setattr__(self, "check_depends", [trim_package(package) for package in self.check_depends]) | ||||||
|  |         object.__setattr__(self, "conflicts", [trim_package(package) for package in self.conflicts]) | ||||||
|  |         object.__setattr__(self, "provides", [trim_package(package) for package in self.provides]) | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def from_json(cls, dump: dict[str, Any]) -> Self: |     def from_json(cls, dump: dict[str, Any]) -> Self: | ||||||
|         """ |         """ | ||||||
|  | |||||||
| @ -213,18 +213,19 @@ class Package(LazyLogging): | |||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def from_aur(cls, name: str, packager: str | None = None) -> Self: |     def from_aur(cls, name: str, packager: str | None = None, *, include_provides: bool = False) -> Self: | ||||||
|         """ |         """ | ||||||
|         construct package properties from AUR page |         construct package properties from AUR page | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             name(str): package name (either base or normal name) |             name(str): package name (either base or normal name) | ||||||
|             packager(str | None, optional): packager to be used for this build (Default value = None) |             packager(str | None, optional): packager to be used for this build (Default value = None) | ||||||
|  |             include_provides(bool, optional): search by provides if no exact match found (Default value = False) | ||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             Self: package properties |             Self: package properties | ||||||
|         """ |         """ | ||||||
|         package = AUR.info(name) |         package = AUR.info(name, include_provides=include_provides) | ||||||
|  |  | ||||||
|         remote = RemoteSource( |         remote = RemoteSource( | ||||||
|             source=PackageSource.AUR, |             source=PackageSource.AUR, | ||||||
| @ -310,7 +311,8 @@ class Package(LazyLogging): | |||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def from_official(cls, name: str, pacman: Pacman, packager: str | None = None, *, use_syncdb: bool = True) -> Self: |     def from_official(cls, name: str, pacman: Pacman, packager: str | None = None, *, use_syncdb: bool = True, | ||||||
|  |                       include_provides: bool = False) -> Self: | ||||||
|         """ |         """ | ||||||
|         construct package properties from official repository page |         construct package properties from official repository page | ||||||
|  |  | ||||||
| @ -319,11 +321,13 @@ class Package(LazyLogging): | |||||||
|             pacman(Pacman): alpm wrapper instance |             pacman(Pacman): alpm wrapper instance | ||||||
|             packager(str | None, optional): packager to be used for this build (Default value = None) |             packager(str | None, optional): packager to be used for this build (Default value = None) | ||||||
|             use_syncdb(bool, optional): use pacman databases instead of official repositories RPC (Default value = True) |             use_syncdb(bool, optional): use pacman databases instead of official repositories RPC (Default value = True) | ||||||
|  |             include_provides(bool, optional): search by provides if no exact match found (Default value = False) | ||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             Self: package properties |             Self: package properties | ||||||
|         """ |         """ | ||||||
|         package = OfficialSyncdb.info(name, pacman=pacman) if use_syncdb else Official.info(name) |         impl = OfficialSyncdb if use_syncdb else Official | ||||||
|  |         package = impl.info(name, pacman=pacman, include_provides=include_provides) | ||||||
|  |  | ||||||
|         remote = RemoteSource( |         remote = RemoteSource( | ||||||
|             source=PackageSource.Repository, |             source=PackageSource.Repository, | ||||||
| @ -516,8 +520,7 @@ class Package(LazyLogging): | |||||||
|         else: |         else: | ||||||
|             remote_version = remote.version |             remote_version = remote.version | ||||||
|  |  | ||||||
|         result: int = vercmp(self.version, remote_version) |         return self.vercmp(remote_version) < 0 | ||||||
|         return result < 0 |  | ||||||
|  |  | ||||||
|     def next_pkgrel(self, local_version: str | None) -> str | None: |     def next_pkgrel(self, local_version: str | None) -> str | None: | ||||||
|         """ |         """ | ||||||
| @ -536,7 +539,7 @@ class Package(LazyLogging): | |||||||
|         if local_version is None: |         if local_version is None: | ||||||
|             return None  # local version not found, keep upstream pkgrel |             return None  # local version not found, keep upstream pkgrel | ||||||
|  |  | ||||||
|         if vercmp(self.version, local_version) > 0: |         if self.vercmp(local_version) > 0: | ||||||
|             return None  # upstream version is newer than local one, keep upstream pkgrel |             return None  # upstream version is newer than local one, keep upstream pkgrel | ||||||
|  |  | ||||||
|         *_, local_pkgrel = parse_version(local_version) |         *_, local_pkgrel = parse_version(local_version) | ||||||
| @ -557,6 +560,19 @@ class Package(LazyLogging): | |||||||
|         details = "" if self.is_single_package else f""" ({" ".join(sorted(self.packages.keys()))})""" |         details = "" if self.is_single_package else f""" ({" ".join(sorted(self.packages.keys()))})""" | ||||||
|         return f"{self.base}{details}" |         return f"{self.base}{details}" | ||||||
|  |  | ||||||
|  |     def vercmp(self, version: str) -> int: | ||||||
|  |         """ | ||||||
|  |         typed wrapper around :func:`pyalpm.vercmp()` | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             version(str): version to compare | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             int: negative if current version is less than provided, positive if greater than and zero if equals | ||||||
|  |         """ | ||||||
|  |         result: int = vercmp(self.version, version) | ||||||
|  |         return result | ||||||
|  |  | ||||||
|     def view(self) -> dict[str, Any]: |     def view(self) -> dict[str, Any]: | ||||||
|         """ |         """ | ||||||
|         generate json package view |         generate json package view | ||||||
|  | |||||||
| @ -83,12 +83,13 @@ class PackageDescription: | |||||||
|  |  | ||||||
|     def __post_init__(self) -> None: |     def __post_init__(self) -> None: | ||||||
|         """ |         """ | ||||||
|         update dependencies list accordingly |         update packages lists accordingly | ||||||
|         """ |         """ | ||||||
|         self.depends = [trim_package(package) for package in self.depends] |         self.depends = [trim_package(package) for package in self.depends] | ||||||
|         self.opt_depends = [trim_package(package) for package in self.opt_depends] |  | ||||||
|         self.make_depends = [trim_package(package) for package in self.make_depends] |         self.make_depends = [trim_package(package) for package in self.make_depends] | ||||||
|  |         self.opt_depends = [trim_package(package) for package in self.opt_depends] | ||||||
|         self.check_depends = [trim_package(package) for package in self.check_depends] |         self.check_depends = [trim_package(package) for package in self.check_depends] | ||||||
|  |         self.provides = [trim_package(package) for package in self.provides] | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def filepath(self) -> Path | None: |     def filepath(self) -> Path | None: | ||||||
|  | |||||||
| @ -17,6 +17,7 @@ | |||||||
| # 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 contextlib | ||||||
| import os | import os | ||||||
| import shutil | import shutil | ||||||
|  |  | ||||||
| @ -84,6 +85,16 @@ class RepositoryPaths(LazyLogging): | |||||||
|                 return Path(self.repository_id.architecture)  # legacy tree suffix |                 return Path(self.repository_id.architecture)  # legacy tree suffix | ||||||
|         return Path(self.repository_id.name) / self.repository_id.architecture |         return Path(self.repository_id.name) / self.repository_id.architecture | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def archive(self) -> Path: | ||||||
|  |         """ | ||||||
|  |         archive directory root | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             Path: archive directory root | ||||||
|  |         """ | ||||||
|  |         return self.root / "archive" | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def build_root(self) -> Path: |     def build_root(self) -> Path: | ||||||
|         """ |         """ | ||||||
| @ -221,22 +232,14 @@ class RepositoryPaths(LazyLogging): | |||||||
|         stat = path.stat() |         stat = path.stat() | ||||||
|         return stat.st_uid, stat.st_gid |         return stat.st_uid, stat.st_gid | ||||||
|  |  | ||||||
|     def cache_for(self, package_base: str) -> Path: |     def _chown(self, path: Path) -> None: | ||||||
|         """ |  | ||||||
|         get path to cached PKGBUILD and package sources for the package base |  | ||||||
|  |  | ||||||
|         Args: |  | ||||||
|             package_base(str): package base name |  | ||||||
|  |  | ||||||
|         Returns: |  | ||||||
|             Path: full path to directory for specified package base cache |  | ||||||
|         """ |  | ||||||
|         return self.cache / package_base |  | ||||||
|  |  | ||||||
|     def chown(self, path: Path) -> None: |  | ||||||
|         """ |         """ | ||||||
|         set owner of path recursively (from root) to root owner |         set owner of path recursively (from root) to root owner | ||||||
|  |  | ||||||
|  |         Notes: | ||||||
|  |             More likely you don't want to call this method explicitly, consider using :func:`preserve_owner()` | ||||||
|  |             as context manager instead | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             path(Path): path to be chown |             path(Path): path to be chown | ||||||
|  |  | ||||||
| @ -256,6 +259,77 @@ class RepositoryPaths(LazyLogging): | |||||||
|             set_owner(path) |             set_owner(path) | ||||||
|             path = path.parent |             path = path.parent | ||||||
|  |  | ||||||
|  |     def archive_for(self, package_base: str) -> Path: | ||||||
|  |         """ | ||||||
|  |         get path to archive specified search criteria | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             package_base(str): package base name | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             Path: path to archive directory for package base | ||||||
|  |         """ | ||||||
|  |         directory = self.archive / "packages" / package_base[0] / package_base | ||||||
|  |         if not directory.is_dir():  # create if not exists | ||||||
|  |             with self.preserve_owner(self.archive): | ||||||
|  |                 directory.mkdir(mode=0o755, parents=True) | ||||||
|  |  | ||||||
|  |         return directory | ||||||
|  |  | ||||||
|  |     def cache_for(self, package_base: str) -> Path: | ||||||
|  |         """ | ||||||
|  |         get path to cached PKGBUILD and package sources for the package base | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             package_base(str): package base name | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             Path: full path to directory for specified package base cache | ||||||
|  |         """ | ||||||
|  |         return self.cache / package_base | ||||||
|  |  | ||||||
|  |     @contextlib.contextmanager | ||||||
|  |     def preserve_owner(self, path: Path | None = None) -> Generator[None, None, None]: | ||||||
|  |         """ | ||||||
|  |         perform any action preserving owner for any newly created file or directory | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             path(Path | None, optional): use this path as root instead of repository root (Default value = None) | ||||||
|  |  | ||||||
|  |         Examples: | ||||||
|  |             This method is designed to use as context manager when you are going to perform operations which might | ||||||
|  |             change filesystem, especially if you are doing it under unsafe flag, e.g.:: | ||||||
|  |  | ||||||
|  |                 >>> with paths.preserve_owner(): | ||||||
|  |                 >>>     paths.tree_create() | ||||||
|  |  | ||||||
|  |             Note, however, that this method doesn't handle any exceptions and will eventually interrupt | ||||||
|  |             if there will be any. | ||||||
|  |         """ | ||||||
|  |         path = path or self.root | ||||||
|  |  | ||||||
|  |         def walk(root: Path) -> Generator[Path, None, None]: | ||||||
|  |             yield root | ||||||
|  |             if not root.exists(): | ||||||
|  |                 return | ||||||
|  |  | ||||||
|  |             # basically walk, but skipping some content | ||||||
|  |             for child in root.iterdir(): | ||||||
|  |                 yield child | ||||||
|  |                 if child in (self.chroot.parent,): | ||||||
|  |                     yield from child.iterdir()  # we only yield top-level in chroot directory | ||||||
|  |                 elif child.is_dir(): | ||||||
|  |                     yield from walk(child) | ||||||
|  |  | ||||||
|  |         # get current filesystem and run action | ||||||
|  |         previous_snapshot = set(walk(path)) | ||||||
|  |         yield | ||||||
|  |  | ||||||
|  |         # get newly created files and directories and chown them | ||||||
|  |         new_entries = set(walk(path)).difference(previous_snapshot) | ||||||
|  |         for entry in new_entries: | ||||||
|  |             self._chown(entry) | ||||||
|  |  | ||||||
|     def tree_clear(self, package_base: str) -> None: |     def tree_clear(self, package_base: str) -> None: | ||||||
|         """ |         """ | ||||||
|         clear package specific files |         clear package specific files | ||||||
| @ -274,7 +348,10 @@ class RepositoryPaths(LazyLogging): | |||||||
|         """ |         """ | ||||||
|         if self.repository_id.is_empty: |         if self.repository_id.is_empty: | ||||||
|             return  # do not even try to create tree in case if no repository id set |             return  # do not even try to create tree in case if no repository id set | ||||||
|  |  | ||||||
|  |         with self.preserve_owner(): | ||||||
|             for directory in ( |             for directory in ( | ||||||
|  |                     self.archive, | ||||||
|                     self.cache, |                     self.cache, | ||||||
|                     self.chroot, |                     self.chroot, | ||||||
|                     self.packages, |                     self.packages, | ||||||
| @ -282,4 +359,3 @@ class RepositoryPaths(LazyLogging): | |||||||
|                     self.repository, |                     self.repository, | ||||||
|             ): |             ): | ||||||
|                 directory.mkdir(mode=0o755, parents=True, exist_ok=True) |                 directory.mkdir(mode=0o755, parents=True, exist_ok=True) | ||||||
|             self.chown(directory) |  | ||||||
|  | |||||||
| @ -21,8 +21,18 @@ import aiohttp_jinja2 | |||||||
| import logging | import logging | ||||||
|  |  | ||||||
| from aiohttp.typedefs import Middleware | from aiohttp.typedefs import Middleware | ||||||
| from aiohttp.web import HTTPClientError, HTTPException, HTTPMethodNotAllowed, HTTPNoContent, HTTPServerError, \ | from aiohttp.web import ( | ||||||
|     HTTPUnauthorized, Request, StreamResponse, json_response, middleware |     HTTPClientError, | ||||||
|  |     HTTPException, | ||||||
|  |     HTTPMethodNotAllowed, | ||||||
|  |     HTTPNoContent, | ||||||
|  |     HTTPServerError, | ||||||
|  |     HTTPUnauthorized, | ||||||
|  |     Request, | ||||||
|  |     StreamResponse, | ||||||
|  |     json_response, | ||||||
|  |     middleware, | ||||||
|  | ) | ||||||
|  |  | ||||||
| from ahriman.web.middlewares import HandlerType | from ahriman.web.middlewares import HandlerType | ||||||
|  |  | ||||||
|  | |||||||
| @ -22,6 +22,7 @@ from ahriman.web.schemas.aur_package_schema import AURPackageSchema | |||||||
| from ahriman.web.schemas.auth_schema import AuthSchema | from ahriman.web.schemas.auth_schema import AuthSchema | ||||||
| from ahriman.web.schemas.build_options_schema import BuildOptionsSchema | from ahriman.web.schemas.build_options_schema import BuildOptionsSchema | ||||||
| from ahriman.web.schemas.changes_schema import ChangesSchema | from ahriman.web.schemas.changes_schema import ChangesSchema | ||||||
|  | from ahriman.web.schemas.configuration_schema import ConfigurationSchema | ||||||
| from ahriman.web.schemas.counters_schema import CountersSchema | from ahriman.web.schemas.counters_schema import CountersSchema | ||||||
| from ahriman.web.schemas.dependencies_schema import DependenciesSchema | from ahriman.web.schemas.dependencies_schema import DependenciesSchema | ||||||
| from ahriman.web.schemas.error_schema import ErrorSchema | from ahriman.web.schemas.error_schema import ErrorSchema | ||||||
| @ -34,6 +35,7 @@ from ahriman.web.schemas.log_schema import LogSchema | |||||||
| from ahriman.web.schemas.login_schema import LoginSchema | from ahriman.web.schemas.login_schema import LoginSchema | ||||||
| from ahriman.web.schemas.logs_rotate_schema import LogsRotateSchema | from ahriman.web.schemas.logs_rotate_schema import LogsRotateSchema | ||||||
| from ahriman.web.schemas.logs_schema import LogsSchema | from ahriman.web.schemas.logs_schema import LogsSchema | ||||||
|  | from ahriman.web.schemas.logs_search_schema import LogsSearchSchema | ||||||
| from ahriman.web.schemas.oauth2_schema import OAuth2Schema | from ahriman.web.schemas.oauth2_schema import OAuth2Schema | ||||||
| from ahriman.web.schemas.package_name_schema import PackageNameSchema | from ahriman.web.schemas.package_name_schema import PackageNameSchema | ||||||
| from ahriman.web.schemas.package_names_schema import PackageNamesSchema | from ahriman.web.schemas.package_names_schema import PackageNamesSchema | ||||||
|  | |||||||
| @ -25,11 +25,11 @@ class AURPackageSchema(Schema): | |||||||
|     response AUR package schema |     response AUR package schema | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     package = fields.String(required=True, metadata={ |  | ||||||
|         "description": "Package base", |  | ||||||
|         "example": "ahriman", |  | ||||||
|     }) |  | ||||||
|     description = fields.String(required=True, metadata={ |     description = fields.String(required=True, metadata={ | ||||||
|         "description": "Package description", |         "description": "Package description", | ||||||
|         "example": "ArcH linux ReposItory MANager", |         "example": "ArcH linux ReposItory MANager", | ||||||
|     }) |     }) | ||||||
|  |     package = fields.String(required=True, metadata={ | ||||||
|  |         "description": "Package base", | ||||||
|  |         "example": "ahriman", | ||||||
|  |     }) | ||||||
|  | |||||||
| @ -25,10 +25,10 @@ class ChangesSchema(Schema): | |||||||
|     response package changes schema |     response package changes schema | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|  |     changes = fields.String(metadata={ | ||||||
|  |         "description": "Package changes in patch format", | ||||||
|  |     }) | ||||||
|     last_commit_sha = fields.String(metadata={ |     last_commit_sha = fields.String(metadata={ | ||||||
|         "description": "Last recorded commit hash", |         "description": "Last recorded commit hash", | ||||||
|         "example": "f1875edca1eb8fc0e55c41d1cae5fa05b6b7c6", |         "example": "f1875edca1eb8fc0e55c41d1cae5fa05b6b7c6", | ||||||
|     }) |     }) | ||||||
|     changes = fields.String(metadata={ |  | ||||||
|         "description": "Package changes in patch format", |  | ||||||
|     }) |  | ||||||
|  | |||||||
							
								
								
									
										39
									
								
								src/ahriman/web/schemas/configuration_schema.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								src/ahriman/web/schemas/configuration_schema.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,39 @@ | |||||||
|  | # | ||||||
|  | # Copyright (c) 2021-2025 ahriman team. | ||||||
|  | # | ||||||
|  | # This file is part of ahriman | ||||||
|  | # (see https://github.com/arcan1s/ahriman). | ||||||
|  | # | ||||||
|  | # This program is free software: you can redistribute it and/or modify | ||||||
|  | # it under the terms of the GNU General Public License as published by | ||||||
|  | # the Free Software Foundation, either version 3 of the License, or | ||||||
|  | # (at your option) any later version. | ||||||
|  | # | ||||||
|  | # This program is distributed in the hope that it will be useful, | ||||||
|  | # but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  | # GNU General Public License for more details. | ||||||
|  | # | ||||||
|  | # You should have received a copy of the GNU General Public License | ||||||
|  | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
|  | # | ||||||
|  | from ahriman.web.apispec import Schema, fields | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ConfigurationSchema(Schema): | ||||||
|  |     """ | ||||||
|  |     response configuration schema | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     key = fields.String(required=True, metadata={ | ||||||
|  |         "description": "Configuration key", | ||||||
|  |         "example": "host", | ||||||
|  |     }) | ||||||
|  |     section = fields.String(required=True, metadata={ | ||||||
|  |         "description": "Configuration section", | ||||||
|  |         "example": "web", | ||||||
|  |     }) | ||||||
|  |     value = fields.String(required=True, metadata={ | ||||||
|  |         "description": "Configuration value", | ||||||
|  |         "example": "127.0.0.1", | ||||||
|  |     }) | ||||||
| @ -25,18 +25,6 @@ class CountersSchema(Schema): | |||||||
|     response package counters schema |     response package counters schema | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     total = fields.Integer(required=True, metadata={ |  | ||||||
|         "description": "Total amount of packages", |  | ||||||
|         "example": 6, |  | ||||||
|     }) |  | ||||||
|     _unknown = fields.Integer(data_key="unknown", required=True, metadata={ |  | ||||||
|         "description": "Amount of packages in unknown state", |  | ||||||
|         "example": 0, |  | ||||||
|     }) |  | ||||||
|     pending = fields.Integer(required=True, metadata={ |  | ||||||
|         "description": "Amount of packages in pending state", |  | ||||||
|         "example": 2, |  | ||||||
|     }) |  | ||||||
|     building = fields.Integer(required=True, metadata={ |     building = fields.Integer(required=True, metadata={ | ||||||
|         "description": "Amount of packages in building state", |         "description": "Amount of packages in building state", | ||||||
|         "example": 1, |         "example": 1, | ||||||
| @ -45,7 +33,19 @@ class CountersSchema(Schema): | |||||||
|         "description": "Amount of packages in failed state", |         "description": "Amount of packages in failed state", | ||||||
|         "example": 1, |         "example": 1, | ||||||
|     }) |     }) | ||||||
|  |     pending = fields.Integer(required=True, metadata={ | ||||||
|  |         "description": "Amount of packages in pending state", | ||||||
|  |         "example": 2, | ||||||
|  |     }) | ||||||
|     success = fields.Integer(required=True, metadata={ |     success = fields.Integer(required=True, metadata={ | ||||||
|         "description": "Amount of packages in success state", |         "description": "Amount of packages in success state", | ||||||
|         "example": 3, |         "example": 3, | ||||||
|     }) |     }) | ||||||
|  |     total = fields.Integer(required=True, metadata={ | ||||||
|  |         "description": "Total amount of packages", | ||||||
|  |         "example": 6, | ||||||
|  |     }) | ||||||
|  |     unknown_ = fields.Integer(data_key="unknown", required=True, metadata={ | ||||||
|  |         "description": "Amount of packages in unknown state", | ||||||
|  |         "example": 0, | ||||||
|  |     }) | ||||||
|  | |||||||
| @ -30,17 +30,17 @@ class EventSchema(Schema): | |||||||
|         "description": "Event creation timestamp", |         "description": "Event creation timestamp", | ||||||
|         "example": 1680537091, |         "example": 1680537091, | ||||||
|     }) |     }) | ||||||
|  |     data = fields.Dict(keys=fields.String(), metadata={ | ||||||
|  |         "description": "Event metadata if available", | ||||||
|  |     }) | ||||||
|     event = fields.String(required=True, metadata={ |     event = fields.String(required=True, metadata={ | ||||||
|         "description": "Event type", |         "description": "Event type", | ||||||
|         "example": EventType.PackageUpdated, |         "example": EventType.PackageUpdated, | ||||||
|     }) |     }) | ||||||
|  |     message = fields.String(metadata={ | ||||||
|  |         "description": "Event message if available", | ||||||
|  |     }) | ||||||
|     object_id = fields.String(required=True, metadata={ |     object_id = fields.String(required=True, metadata={ | ||||||
|         "description": "Event object identifier", |         "description": "Event object identifier", | ||||||
|         "example": "ahriman", |         "example": "ahriman", | ||||||
|     }) |     }) | ||||||
|     message = fields.String(metadata={ |  | ||||||
|         "description": "Event message if available", |  | ||||||
|     }) |  | ||||||
|     data = fields.Dict(keys=fields.String(), metadata={ |  | ||||||
|         "description": "Event metadata if available", |  | ||||||
|     }) |  | ||||||
|  | |||||||
| @ -31,14 +31,14 @@ class EventSearchSchema(PaginationSchema): | |||||||
|         "description": "Event type", |         "description": "Event type", | ||||||
|         "example": EventType.PackageUpdated, |         "example": EventType.PackageUpdated, | ||||||
|     }) |     }) | ||||||
|     object_id = fields.String(metadata={ |  | ||||||
|         "description": "Event object identifier", |  | ||||||
|         "example": "ahriman", |  | ||||||
|     }) |  | ||||||
|     from_date = fields.Integer(metadata={ |     from_date = fields.Integer(metadata={ | ||||||
|         "description": "Minimal creation timestamp, inclusive", |         "description": "Minimal creation timestamp, inclusive", | ||||||
|         "example": 1680537091, |         "example": 1680537091, | ||||||
|     }) |     }) | ||||||
|  |     object_id = fields.String(metadata={ | ||||||
|  |         "description": "Event object identifier", | ||||||
|  |         "example": "ahriman", | ||||||
|  |     }) | ||||||
|     to_date = fields.Integer(metadata={ |     to_date = fields.Integer(metadata={ | ||||||
|         "description": "Maximal creation timestamp, exclusive", |         "description": "Maximal creation timestamp, exclusive", | ||||||
|         "example": 1680537091, |         "example": 1680537091, | ||||||
|  | |||||||
| @ -33,10 +33,10 @@ class LogSchema(Schema): | |||||||
|     message = fields.String(required=True, metadata={ |     message = fields.String(required=True, metadata={ | ||||||
|         "description": "Log message", |         "description": "Log message", | ||||||
|     }) |     }) | ||||||
|  |     process_id = fields.String(metadata={ | ||||||
|  |         "description": "Process unique identifier", | ||||||
|  |     }) | ||||||
|     version = fields.String(required=True, metadata={ |     version = fields.String(required=True, metadata={ | ||||||
|         "description": "Package version to tag", |         "description": "Package version to tag", | ||||||
|         "example": __version__, |         "example": __version__, | ||||||
|     }) |     }) | ||||||
|     process_id = fields.String(metadata={ |  | ||||||
|         "description": "Process unique identifier", |  | ||||||
|     }) |  | ||||||
|  | |||||||
| @ -25,11 +25,11 @@ class LoginSchema(Schema): | |||||||
|     request login schema |     request login schema | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     username = fields.String(required=True, metadata={ |  | ||||||
|         "description": "Login username", |  | ||||||
|         "example": "user", |  | ||||||
|     }) |  | ||||||
|     password = fields.String(required=True, metadata={ |     password = fields.String(required=True, metadata={ | ||||||
|         "description": "Login password", |         "description": "Login password", | ||||||
|         "example": "pa55w0rd", |         "example": "pa55w0rd", | ||||||
|     }) |     }) | ||||||
|  |     username = fields.String(required=True, metadata={ | ||||||
|  |         "description": "Login username", | ||||||
|  |         "example": "user", | ||||||
|  |     }) | ||||||
|  | |||||||
							
								
								
									
										39
									
								
								src/ahriman/web/schemas/logs_search_schema.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								src/ahriman/web/schemas/logs_search_schema.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,39 @@ | |||||||
|  | # | ||||||
|  | # Copyright (c) 2021-2025 ahriman team. | ||||||
|  | # | ||||||
|  | # This file is part of ahriman | ||||||
|  | # (see https://github.com/arcan1s/ahriman). | ||||||
|  | # | ||||||
|  | # This program is free software: you can redistribute it and/or modify | ||||||
|  | # it under the terms of the GNU General Public License as published by | ||||||
|  | # the Free Software Foundation, either version 3 of the License, or | ||||||
|  | # (at your option) any later version. | ||||||
|  | # | ||||||
|  | # This program is distributed in the hope that it will be useful, | ||||||
|  | # but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  | # GNU General Public License for more details. | ||||||
|  | # | ||||||
|  | # You should have received a copy of the GNU General Public License | ||||||
|  | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
|  | # | ||||||
|  | from ahriman import __version__ | ||||||
|  | from ahriman.web.apispec import fields | ||||||
|  | from ahriman.web.schemas.pagination_schema import PaginationSchema | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class LogsSearchSchema(PaginationSchema): | ||||||
|  |     """ | ||||||
|  |     request log search schema | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     head = fields.Boolean(metadata={ | ||||||
|  |         "description": "Return versions only without fetching logs themselves", | ||||||
|  |     }) | ||||||
|  |     process_id = fields.String(metadata={ | ||||||
|  |         "description": "Process unique identifier to search", | ||||||
|  |     }) | ||||||
|  |     version = fields.String(metadata={ | ||||||
|  |         "description": "Package version to search", | ||||||
|  |         "example": __version__, | ||||||
|  |     }) | ||||||
| @ -37,22 +37,14 @@ class PackagePropertiesSchema(Schema): | |||||||
|         "description": "Package build timestamp", |         "description": "Package build timestamp", | ||||||
|         "example": 1680537091, |         "example": 1680537091, | ||||||
|     }) |     }) | ||||||
|     depends = fields.List(fields.String(), metadata={ |  | ||||||
|         "description": "Package dependencies list", |  | ||||||
|         "example": ["devtools"], |  | ||||||
|     }) |  | ||||||
|     make_depends = fields.List(fields.String(), metadata={ |  | ||||||
|         "description": "Package make dependencies list", |  | ||||||
|         "example": ["python-build"], |  | ||||||
|     }) |  | ||||||
|     opt_depends = fields.List(fields.String(), metadata={ |  | ||||||
|         "description": "Package optional dependencies list", |  | ||||||
|         "example": ["python-aiohttp"], |  | ||||||
|     }) |  | ||||||
|     check_depends = fields.List(fields.String(), metadata={ |     check_depends = fields.List(fields.String(), metadata={ | ||||||
|         "description": "Package test dependencies list", |         "description": "Package test dependencies list", | ||||||
|         "example": ["python-pytest"], |         "example": ["python-pytest"], | ||||||
|     }) |     }) | ||||||
|  |     depends = fields.List(fields.String(), metadata={ | ||||||
|  |         "description": "Package dependencies list", | ||||||
|  |         "example": ["devtools"], | ||||||
|  |     }) | ||||||
|     description = fields.String(metadata={ |     description = fields.String(metadata={ | ||||||
|         "description": "Package description", |         "description": "Package description", | ||||||
|         "example": "ArcH linux ReposItory MANager", |         "example": "ArcH linux ReposItory MANager", | ||||||
| @ -73,6 +65,14 @@ class PackagePropertiesSchema(Schema): | |||||||
|         "description": "Package licenses", |         "description": "Package licenses", | ||||||
|         "example": ["GPL3"], |         "example": ["GPL3"], | ||||||
|     }) |     }) | ||||||
|  |     make_depends = fields.List(fields.String(), metadata={ | ||||||
|  |         "description": "Package make dependencies list", | ||||||
|  |         "example": ["python-build"], | ||||||
|  |     }) | ||||||
|  |     opt_depends = fields.List(fields.String(), metadata={ | ||||||
|  |         "description": "Package optional dependencies list", | ||||||
|  |         "example": ["python-aiohttp"], | ||||||
|  |     }) | ||||||
|     provides = fields.List(fields.String(), metadata={ |     provides = fields.List(fields.String(), metadata={ | ||||||
|         "description": "Package provides list", |         "description": "Package provides list", | ||||||
|         "example": ["ahriman-git"], |         "example": ["ahriman-git"], | ||||||
|  | |||||||
| @ -32,18 +32,18 @@ class PackageSchema(Schema): | |||||||
|         "description": "Package base", |         "description": "Package base", | ||||||
|         "example": "ahriman", |         "example": "ahriman", | ||||||
|     }) |     }) | ||||||
|     version = fields.String(required=True, metadata={ |     packager = fields.String(metadata={ | ||||||
|         "description": "Package version", |         "description": "packager for the last success package build", | ||||||
|         "example": __version__, |         "example": "ahriman bot <ahriman@example.com>", | ||||||
|     }) |  | ||||||
|     remote = fields.Nested(RemoteSchema(), required=True, metadata={ |  | ||||||
|         "description": "Package remote properties", |  | ||||||
|     }) |     }) | ||||||
|     packages = fields.Dict( |     packages = fields.Dict( | ||||||
|         keys=fields.String(), values=fields.Nested(PackagePropertiesSchema()), required=True, metadata={ |         keys=fields.String(), values=fields.Nested(PackagePropertiesSchema()), required=True, metadata={ | ||||||
|             "description": "Packages which belong to this base", |             "description": "Packages which belong to this base", | ||||||
|         }) |         }) | ||||||
|     packager = fields.String(metadata={ |     remote = fields.Nested(RemoteSchema(), required=True, metadata={ | ||||||
|         "description": "packager for the last success package build", |         "description": "Package remote properties", | ||||||
|         "example": "ahriman bot <ahriman@example.com>", |     }) | ||||||
|  |     version = fields.String(required=True, metadata={ | ||||||
|  |         "description": "Package version", | ||||||
|  |         "example": __version__, | ||||||
|     }) |     }) | ||||||
|  | |||||||
| @ -25,19 +25,19 @@ class RepositoryStatsSchema(Schema): | |||||||
|     response repository stats schema |     response repository stats schema | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     bases = fields.Int(metadata={ |  | ||||||
|         "description": "Amount of unique packages bases", |  | ||||||
|         "example": 2, |  | ||||||
|     }) |  | ||||||
|     packages = fields.Int(metadata={ |  | ||||||
|         "description": "Amount of unique packages", |  | ||||||
|         "example": 4, |  | ||||||
|     }) |  | ||||||
|     archive_size = fields.Int(metadata={ |     archive_size = fields.Int(metadata={ | ||||||
|         "description": "Total archive size of the packages in bytes", |         "description": "Total archive size of the packages in bytes", | ||||||
|         "example": 42000, |         "example": 42000, | ||||||
|     }) |     }) | ||||||
|  |     bases = fields.Int(metadata={ | ||||||
|  |         "description": "Amount of unique packages bases", | ||||||
|  |         "example": 2, | ||||||
|  |     }) | ||||||
|     installed_size = fields.Int(metadata={ |     installed_size = fields.Int(metadata={ | ||||||
|         "description": "Total installed size of the packages in bytes", |         "description": "Total installed size of the packages in bytes", | ||||||
|         "example": 42000000, |         "example": 42000000, | ||||||
|     }) |     }) | ||||||
|  |     packages = fields.Int(metadata={ | ||||||
|  |         "description": "Amount of unique packages", | ||||||
|  |         "example": 4, | ||||||
|  |     }) | ||||||
|  | |||||||
| @ -25,7 +25,7 @@ class SearchSchema(Schema): | |||||||
|     request package search schema |     request package search schema | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     _for = fields.List(fields.String(), data_key="for", required=True, metadata={ |     for_ = fields.List(fields.String(), data_key="for", required=True, metadata={ | ||||||
|         "description": "Keyword for search", |         "description": "Keyword for search", | ||||||
|         "example": ["ahriman"], |         "example": ["ahriman"], | ||||||
|     }) |     }) | ||||||
|  | |||||||
| @ -22,6 +22,7 @@ import aiohttp_jinja2 | |||||||
| from typing import Any, ClassVar | from typing import Any, ClassVar | ||||||
|  |  | ||||||
| from ahriman.core.auth.helpers import authorized_userid | from ahriman.core.auth.helpers import authorized_userid | ||||||
|  | from ahriman.core.utils import pretty_interval | ||||||
| from ahriman.models.user_access import UserAccess | from ahriman.models.user_access import UserAccess | ||||||
| from ahriman.web.apispec import aiohttp_apispec | from ahriman.web.apispec import aiohttp_apispec | ||||||
| from ahriman.web.views.base import BaseView | from ahriman.web.views.base import BaseView | ||||||
| @ -37,6 +38,10 @@ class IndexView(BaseView): | |||||||
|             * control - HTML to insert for login control, HTML string, required |             * control - HTML to insert for login control, HTML string, required | ||||||
|             * enabled - whether authorization is enabled by configuration or not, boolean, required |             * enabled - whether authorization is enabled by configuration or not, boolean, required | ||||||
|             * username - authenticated username if any, string, null means not authenticated |             * username - authenticated username if any, string, null means not authenticated | ||||||
|  |         * autorefresh_intervals - auto refresh intervals, optional | ||||||
|  |             * interval - auto refresh interval in milliseconds, integer, required | ||||||
|  |             * is_active - is current interval active or not, boolean, required | ||||||
|  |             * text - text representation of the interval (e.g. "30 seconds"), string, required | ||||||
|         * docs_enabled - indicates if api docs is enabled, boolean, required |         * docs_enabled - indicates if api docs is enabled, boolean, required | ||||||
|         * index_url - url to the repository index, string, optional |         * index_url - url to the repository index, string, optional | ||||||
|         * repositories - list of repositories unique identifiers, required |         * repositories - list of repositories unique identifiers, required | ||||||
| @ -66,8 +71,19 @@ class IndexView(BaseView): | |||||||
|             "username": auth_username, |             "username": auth_username, | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         autorefresh_intervals = [ | ||||||
|  |             { | ||||||
|  |                 "interval": interval * 1000,  # milliseconds | ||||||
|  |                 "is_active": index == 0,  # first element is always default | ||||||
|  |                 "text": pretty_interval(interval), | ||||||
|  |             } | ||||||
|  |             for index, interval in enumerate(self.configuration.getintlist("web", "autorefresh_intervals", fallback=[])) | ||||||
|  |             if interval > 0  # special case if 0 exists and first, refresh will not be turned on by default | ||||||
|  |         ] | ||||||
|  |  | ||||||
|         return { |         return { | ||||||
|             "auth": auth, |             "auth": auth, | ||||||
|  |             "autorefresh_intervals": sorted(autorefresh_intervals, key=lambda interval: interval["interval"]), | ||||||
|             "docs_enabled": aiohttp_apispec is not None, |             "docs_enabled": aiohttp_apispec is not None, | ||||||
|             "index_url": self.configuration.get("web", "index_url", fallback=None), |             "index_url": self.configuration.get("web", "index_url", fallback=None), | ||||||
|             "repositories": [ |             "repositories": [ | ||||||
|  | |||||||
| @ -21,6 +21,7 @@ from aiohttp.web import HTTPBadRequest, HTTPNoContent, Response, json_response | |||||||
| from collections.abc import Callable | from collections.abc import Callable | ||||||
| from typing import ClassVar | from typing import ClassVar | ||||||
|  |  | ||||||
|  | from ahriman.core.types import Comparable | ||||||
| from ahriman.models.user_access import UserAccess | from ahriman.models.user_access import UserAccess | ||||||
| from ahriman.models.worker import Worker | from ahriman.models.worker import Worker | ||||||
| from ahriman.web.apispec.decorators import apidocs | from ahriman.web.apispec.decorators import apidocs | ||||||
| @ -74,7 +75,7 @@ class WorkersView(BaseView): | |||||||
|         """ |         """ | ||||||
|         workers = self.workers.workers |         workers = self.workers.workers | ||||||
|  |  | ||||||
|         comparator: Callable[[Worker], str] = lambda item: item.identifier |         comparator: Callable[[Worker], Comparable] = lambda item: item.identifier | ||||||
|         response = [worker.view() for worker in sorted(workers, key=comparator)] |         response = [worker.view() for worker in sorted(workers, key=comparator)] | ||||||
|  |  | ||||||
|         return json_response(response) |         return json_response(response) | ||||||
|  | |||||||
| @ -90,7 +90,7 @@ class LogsView(StatusViewGuard, BaseView): | |||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             _, status = self.service().package_get(package_base) |             _, status = self.service().package_get(package_base) | ||||||
|             logs = self.service(package_base=package_base).package_logs_get(package_base, -1, 0) |             logs = self.service(package_base=package_base).package_logs_get(package_base, None, None, -1, 0) | ||||||
|         except UnknownPackageError: |         except UnknownPackageError: | ||||||
|             raise HTTPNotFound(reason=f"Package {package_base} is unknown") |             raise HTTPNotFound(reason=f"Package {package_base} is unknown") | ||||||
|  |  | ||||||
|  | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user