mirror of
				https://github.com/arcan1s/ahriman.git
				synced 2025-10-25 02:43:45 +00:00 
			
		
		
		
	Compare commits
	
		
			36 Commits
		
	
	
		
			2.19.1
			...
			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 | 
| @ -165,6 +165,11 @@ Again, the most checks can be performed by `tox` command, though some additional | ||||
|  | ||||
|     # Blank line again and package imports | ||||
|     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. | ||||
|  | ||||
| @ -100,6 +100,14 @@ ahriman.application.handlers.rebuild module | ||||
|    :no-undoc-members: | ||||
|    :show-inheritance: | ||||
|  | ||||
| ahriman.application.handlers.reload module | ||||
| ------------------------------------------ | ||||
|  | ||||
| .. automodule:: ahriman.application.handlers.reload | ||||
|    :members: | ||||
|    :no-undoc-members: | ||||
|    :show-inheritance: | ||||
|  | ||||
| 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: | ||||
|    :show-inheritance: | ||||
|  | ||||
| ahriman.core.database.migrations.m016\_archive module | ||||
| ----------------------------------------------------- | ||||
|  | ||||
| .. automodule:: ahriman.core.database.migrations.m016_archive | ||||
|    :members: | ||||
|    :no-undoc-members: | ||||
|    :show-inheritance: | ||||
|  | ||||
| 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 | ||||
|  | ||||
|    ahriman.core.alpm | ||||
|    ahriman.core.archive | ||||
|    ahriman.core.auth | ||||
|    ahriman.core.build_tools | ||||
|    ahriman.core.configuration | ||||
| @ -15,6 +16,7 @@ Subpackages | ||||
|    ahriman.core.distributed | ||||
|    ahriman.core.formatters | ||||
|    ahriman.core.gitremote | ||||
|    ahriman.core.housekeeping | ||||
|    ahriman.core.http | ||||
|    ahriman.core.log | ||||
|    ahriman.core.report | ||||
|  | ||||
| @ -44,6 +44,14 @@ ahriman.web.schemas.changes\_schema module | ||||
|    :no-undoc-members: | ||||
|    :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 | ||||
| ------------------------------------------- | ||||
|  | ||||
| @ -140,6 +148,14 @@ ahriman.web.schemas.logs\_schema module | ||||
|    :no-undoc-members: | ||||
|    :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 | ||||
| ----------------------------------------- | ||||
|  | ||||
|  | ||||
| @ -12,6 +12,14 @@ ahriman.web.views.v1.service.add module | ||||
|    :no-undoc-members: | ||||
|    :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 | ||||
| ---------------------------------------- | ||||
|  | ||||
|  | ||||
| @ -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.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.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.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. | ||||
|  | ||||
| @ -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). | ||||
|  | ||||
| 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.: | ||||
|  | ||||
| .. 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. | ||||
| * ``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. | ||||
| * ``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. | ||||
|  | ||||
| ``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. | ||||
| * ``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 | ||||
| -------------- | ||||
|  | ||||
| @ -138,6 +146,8 @@ Build related configuration. Group name can refer to architecture, e.g. ``build: | ||||
|  | ||||
| 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. | ||||
|  | ||||
| ``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. | ||||
|  | ||||
| * ``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``. | ||||
| * ``host`` - host to bind, 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. | ||||
|  | ||||
| ``keyring`` group | ||||
| -------------------- | ||||
| ----------------- | ||||
|  | ||||
| Keyring package generator plugin. | ||||
|  | ||||
| @ -197,6 +208,13 @@ Keyring generator plugin | ||||
| * ``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. | ||||
|  | ||||
| ``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 | ||||
| -------------------- | ||||
|  | ||||
|  | ||||
| @ -2,7 +2,7 @@ | ||||
|  | ||||
| pkgbase='ahriman' | ||||
| pkgname=('ahriman' 'ahriman-core' 'ahriman-triggers' 'ahriman-web') | ||||
| pkgver=2.19.1 | ||||
| pkgver=2.19.0 | ||||
| pkgrel=1 | ||||
| pkgdesc="ArcH linux ReposItory MANager" | ||||
| arch=('any') | ||||
| @ -40,6 +40,7 @@ package_ahriman-core() { | ||||
|                 'rsync: sync by using rsync') | ||||
|     install="$pkgbase.install" | ||||
|     backup=('etc/ahriman.ini' | ||||
|             'etc/ahriman.ini.d/00-housekeeping.ini' | ||||
|             'etc/ahriman.ini.d/logging.ini') | ||||
|  | ||||
|     cd "$pkgbase-$pkgver" | ||||
| @ -49,6 +50,7 @@ package_ahriman-core() { | ||||
|  | ||||
|     # 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.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 "$srcdir/$pkgbase.sysusers" "$pkgdir/usr/lib/sysusers.d/$pkgbase.conf" | ||||
|  | ||||
| @ -5,6 +5,7 @@ After=network.target | ||||
| [Service] | ||||
| Type=simple | ||||
| ExecStart=/usr/bin/ahriman web | ||||
| ExecReload=/usr/bin/ahriman web-reload | ||||
| User=ahriman | ||||
| Group=ahriman | ||||
|  | ||||
|  | ||||
| @ -7,8 +7,6 @@ logging = ahriman.ini.d/logging.ini | ||||
| ;apply_migrations = yes | ||||
| ; Path to the application SQLite database. | ||||
| database = ${repository:root}/ahriman.db | ||||
| ; Keep last build logs for each package | ||||
| keep_last_logs = 5 | ||||
|  | ||||
| [alpm] | ||||
| ; Path to pacman system database cache. | ||||
| @ -45,9 +43,13 @@ triggers[] = ahriman.core.gitremote.RemotePullTrigger | ||||
| triggers[] = ahriman.core.report.ReportTrigger | ||||
| triggers[] = ahriman.core.upload.UploadTrigger | ||||
| 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. | ||||
| triggers_known[] = ahriman.core.gitremote.RemotePullTrigger | ||||
| 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.upload.UploadTrigger | ||||
| ; 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] | ||||
| ; 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.WorkerTrigger | ||||
| 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 | ||||
| ;     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_archive_upload = no | ||||
| ; Address to bind the server. | ||||
|  | ||||
| @ -80,10 +80,28 @@ | ||||
|                 <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> | ||||
|                 </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> | ||||
|  | ||||
|             <table id="packages" | ||||
|                    data-classes="table table-hover" | ||||
|                    data-cookie="true" | ||||
|                    data-cookie-id-table="ahriman-packages" | ||||
|                    data-cookie-storage="localStorage" | ||||
|                    data-export-options='{"fileName": "packages"}' | ||||
|                    data-filter-control="true" | ||||
|                    data-filter-control-visible="false" | ||||
| @ -102,8 +120,8 @@ | ||||
|                    data-sortable="true" | ||||
|                    data-sort-name="base" | ||||
|                    data-sort-order="asc" | ||||
|                    data-toggle="table" | ||||
|                    data-toolbar="#toolbar"> | ||||
|                    data-toolbar="#toolbar" | ||||
|                    data-unique-id="id"> | ||||
|                 <thead class="table-primary"> | ||||
|                     <tr> | ||||
|                         <th data-checkbox="true"></th> | ||||
|  | ||||
| @ -3,7 +3,9 @@ | ||||
|  | ||||
|     function createAlert(title, message, clz, action, id) { | ||||
|         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"); | ||||
|         wrapper.id = `alert-${id}`; | ||||
|  | ||||
| @ -51,6 +51,87 @@ | ||||
|     const dashboardPackagesCountChartCanvas = document.getElementById("dashboard-packages-count-chart"); | ||||
|     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(_ => { | ||||
|         dashboardPackagesStatusesChart = new Chart(dashboardPackagesStatusesChartCanvas, { | ||||
|             type: "pie", | ||||
|  | ||||
| @ -148,8 +148,19 @@ | ||||
|  | ||||
|         packageAddInput.addEventListener("keyup", _ => { | ||||
|             clearTimeout(packageAddInput.requestTimeout); | ||||
|  | ||||
|             // do not update datalist if search string didn't change yet | ||||
|             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(_ => { | ||||
|                 const value = packageAddInput.value; | ||||
|  | ||||
|                 if (value.length >= 3) { | ||||
|                     makeRequest( | ||||
|  | ||||
| @ -80,8 +80,7 @@ | ||||
|                                data-classes="table table-hover" | ||||
|                                data-sortable="true" | ||||
|                                data-sort-name="timestamp" | ||||
|                                data-sort-order="desc" | ||||
|                                data-toggle="table"> | ||||
|                                data-sort-order="desc"> | ||||
|                             <thead class="table-primary"> | ||||
|                                 <tr> | ||||
|                                     <th data-align="right" data-field="timestamp">date</th> | ||||
| @ -98,10 +97,24 @@ | ||||
|                     <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()" data-bs-dismiss="modal"><i class="bi bi-play"></i><span class="d-none d-sm-inline"> update</span></button> | ||||
|                     <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> | ||||
|                 {% endif %} | ||||
|                 <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> | ||||
|                 {% 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> | ||||
|                     <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> | ||||
|             </div> | ||||
|         </div> | ||||
| @ -140,6 +153,10 @@ | ||||
|  | ||||
|     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() { | ||||
|         packageInfoEventsUpdateChartCanvas.hidden = true; | ||||
|         if (packageInfoEventsUpdateChart) { | ||||
| @ -148,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() { | ||||
|         const changes = packageInfoChangesInput.textContent; | ||||
|         await copyToClipboard(changes, packageInfoChangesCopyButton); | ||||
| @ -291,6 +315,69 @@ | ||||
|     } | ||||
|  | ||||
|     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( | ||||
|             `/api/v2/packages/${packageBase}/logs`, | ||||
|             { | ||||
| @ -319,15 +406,19 @@ | ||||
|                         const link = document.createElement("a"); | ||||
|                         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.href = "#"; | ||||
|                         link.onclick = _ => { | ||||
|                             const logs = data | ||||
|                                 .filter(log_record => log_record.version === version.version && log_record.process_id === version.process_id) | ||||
|                                 .map(log_record => `[${new Date(1000 * log_record.created).toISOString()}] ${log_record.message}`); | ||||
|  | ||||
|                             packageInfoLogsInput.textContent = logs.join("\n"); | ||||
|                             // check if we are at the bottom of the code block | ||||
|                             const isScrolledToBottom = packageInfoLogsInput.scrollTop + packageInfoLogsInput.clientHeight >= packageInfoLogsInput.scrollHeight; | ||||
|                             packageInfoLogsInput.textContent = link.dataset.logs; | ||||
|                             highlight(packageInfoLogsInput); | ||||
|                             if (isScrolledToBottom) | ||||
|                                 packageInfoLogsInput.scrollTop = packageInfoLogsInput.scrollHeight; // scroll to the new end | ||||
|  | ||||
|                             Array.from(packageInfoLogsVersions.children).forEach(el => el.classList.remove("active")); | ||||
|                             link.classList.add("active"); | ||||
| @ -403,12 +494,12 @@ | ||||
|     } | ||||
|  | ||||
|     function packageInfoRemove() { | ||||
|         const packageBase = packageInfoModal.package; | ||||
|         const packageBase = packageInfoModal.dataset.package; | ||||
|         packagesRemove([packageBase]); | ||||
|     } | ||||
|  | ||||
|     function packageInfoUpdate() { | ||||
|         const packageBase = packageInfoModal.package; | ||||
|         const packageBase = packageInfoModal.dataset.package; | ||||
|         packagesAdd(packageBase, [], repository, {refresh: packageInfoRefreshInput.checked}); | ||||
|     } | ||||
|  | ||||
| @ -416,10 +507,10 @@ | ||||
|         const isPackageBaseSet = packageBase !== undefined; | ||||
|         if (isPackageBaseSet) { | ||||
|             // set package base as currently used | ||||
|             packageInfoModal.package = packageBase; | ||||
|             packageInfoModal.dataset.package = packageBase; | ||||
|         } else { | ||||
|             // read package base from the current window attribute | ||||
|             packageBase = packageInfoModal.package; | ||||
|             packageBase = packageInfoModal.dataset.package; | ||||
|         } | ||||
|  | ||||
|         const onFailure = error => { | ||||
| @ -438,10 +529,27 @@ | ||||
|  | ||||
|         if (isPackageBaseSet) { | ||||
|             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(_ => { | ||||
|         packageInfoEventsTable.bootstrapTable({}); | ||||
|  | ||||
|         packageInfoEventsUpdateChart = new Chart(packageInfoEventsUpdateChartCanvas, { | ||||
|             type: "line", | ||||
|             data: {}, | ||||
| @ -468,6 +576,11 @@ | ||||
|             packageInfoChangesInput.textContent = ""; | ||||
|             packageInfoEventsTable.bootstrapTable("load", []); | ||||
|             clearChart(); | ||||
|  | ||||
|             clearInterval(packageInfoAutoReloadTask); | ||||
|             packageInfoAutoReloadTask = null; // not really required (?) but lets clear everything | ||||
|         }); | ||||
|  | ||||
|         restoreAutoReloadSettings(packageInfoAutoReloadButton, packageInfoAutoReloadInput); | ||||
|     }); | ||||
| </script> | ||||
|  | ||||
| @ -10,6 +10,10 @@ | ||||
|     const dashboardButton = document.getElementById("dashboard-button"); | ||||
|     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) { | ||||
|         makeRequest( | ||||
|             uri, | ||||
| @ -55,6 +59,41 @@ | ||||
|         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) { | ||||
|         packages = packages ?? getSelection(); | ||||
|         const onSuccess = update => `Packages ${update} have been removed`; | ||||
| @ -88,130 +127,22 @@ | ||||
|  | ||||
|     function reload() { | ||||
|         table.bootstrapTable("showLoading"); | ||||
|  | ||||
|         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"; | ||||
|         const onFailure = error => { | ||||
|             if ((error.status === 401) || (error.status === 403)) { | ||||
|                 // authorization error | ||||
|                 const text = "In order to see statuses you must login first."; | ||||
|                 table.find("tr.unauthorized").remove(); | ||||
|                 table.find("tbody").append(`<tr class="unauthorized"><td colspan="100%">${safe(text)}</td></tr>`); | ||||
|                 table.bootstrapTable("hideLoading"); | ||||
|             } else { | ||||
|                 // other errors | ||||
|                 const message = details => `Could not load list of packages: ${details}`; | ||||
|                 showFailure("Load failure", message, error); | ||||
|             } | ||||
|         }; | ||||
|  | ||||
|         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)) { | ||||
|                     // authorization error | ||||
|                     const text = "In order to see statuses you must login first."; | ||||
|                     table.find("tr.unauthorized").remove(); | ||||
|                     table.find("tbody").append(`<tr class="unauthorized"><td colspan="100%">${safe(text)}</td></tr>`); | ||||
|                     table.bootstrapTable("hideLoading"); | ||||
|                 } else { | ||||
|                     // other errors | ||||
|                     const message = details => `Could not load list of packages: ${details}`; | ||||
|                     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) { | ||||
|                     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; | ||||
|             }, | ||||
|         ); | ||||
|         packagesLoad(onFailure); | ||||
|         statusLoad(); | ||||
|     } | ||||
|  | ||||
|     function selectRepository() { | ||||
| @ -230,7 +161,24 @@ | ||||
|         return {classes: cellClass(value)}; | ||||
|     } | ||||
|  | ||||
|     function toggleTableAutoReload(interval) { | ||||
|         clearInterval(tableAutoReloadTask); | ||||
|         tableAutoReloadTask = toggleAutoReload(tableAutoReloadButton, interval, tableAutoReloadInput, _ => { | ||||
|             if (!hasActiveModal() && | ||||
|                 !hasActiveDropdown()) { | ||||
|                 packagesLoad(); | ||||
|                 statusLoad(); | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     ready(_ => { | ||||
|         const onCheckFunction = function () { | ||||
|             if (packageRemoveButton) { | ||||
|                 packageRemoveButton.disabled = !getSelection().length; | ||||
|             } | ||||
|         }; | ||||
|  | ||||
|         document.querySelectorAll("#repositories a").forEach(element => { | ||||
|             element.onclick = _ => { | ||||
|                 repository = { | ||||
| @ -245,49 +193,55 @@ | ||||
|             }; | ||||
|         }); | ||||
|  | ||||
|         table.on("check.bs.table uncheck.bs.table check-all.bs.table uncheck-all.bs.table", _ => { | ||||
|             if (packageRemoveButton) { | ||||
|                 packageRemoveButton.disabled = !table.bootstrapTable("getSelections").length; | ||||
|             } | ||||
|         }); | ||||
|         table.on("click-row.bs.table", (self, data, row, cell) => { | ||||
|             if (0 === cell || "base" === cell) { | ||||
|                 const method = data[0] === true ? "uncheckBy" : "checkBy"; // fck javascript | ||||
|                 table.bootstrapTable(method, {field: "id", values: [data.id]}); | ||||
|             } else showPackageInfo(data.id); | ||||
|         }); | ||||
|         table.on("created-controls.bs.table", _ => { | ||||
|             new easepick.create({ | ||||
|                 element: document.querySelector(".bootstrap-table-filter-control-timestamp"), | ||||
|                 css: [ | ||||
|                     "https://cdn.jsdelivr.net/npm/@easepick/bundle@1.2.1/dist/index.css", | ||||
|                 ], | ||||
|                 grid: 2, | ||||
|                 calendars: 2, | ||||
|                 autoApply: false, | ||||
|                 locale: { | ||||
|                     cancel: "Clear", | ||||
|                 }, | ||||
|                 RangePlugin: { | ||||
|                     tooltip: false, | ||||
|                 }, | ||||
|                 plugins: [ | ||||
|                     "RangePlugin", | ||||
|                 ], | ||||
|                 setup: picker => { | ||||
|                     picker.on("select", _ => { table.bootstrapTable("triggerSearch"); }); | ||||
|                     // replace "Cancel" behaviour to "Clear" | ||||
|                     picker.onClickCancelButton = element => { | ||||
|                         if (picker.isCancelButton(element)) { | ||||
|                             picker.clear(); | ||||
|                             picker.hide(); | ||||
|                             table.bootstrapTable("triggerSearch"); | ||||
|                         } | ||||
|                     }; | ||||
|                 }, | ||||
|             }); | ||||
|         table.bootstrapTable({ | ||||
|             onCheck: onCheckFunction, | ||||
|             onCheckAll: onCheckFunction, | ||||
|             onClickRow: (data, row, cell) => { | ||||
|                 if (0 === cell || "base" === cell) { | ||||
|                     const method = data[0] === true ? "uncheckBy" : "checkBy"; // fck javascript | ||||
|                     table.bootstrapTable(method, {field: "id", values: [data.id]}); | ||||
|                 } else showPackageInfo(data.id); | ||||
|             }, | ||||
|             onCreatedControls: _ => { | ||||
|                 new easepick.create({ | ||||
|                     element: document.querySelector(".bootstrap-table-filter-control-timestamp"), | ||||
|                     css: [ | ||||
|                         "https://cdn.jsdelivr.net/npm/@easepick/bundle@1.2.1/dist/index.css", | ||||
|                     ], | ||||
|                     grid: 2, | ||||
|                     calendars: 2, | ||||
|                     autoApply: false, | ||||
|                     locale: { | ||||
|                         cancel: "Clear", | ||||
|                     }, | ||||
|                     RangePlugin: { | ||||
|                         tooltip: false, | ||||
|                     }, | ||||
|                     plugins: [ | ||||
|                         "RangePlugin", | ||||
|                     ], | ||||
|                     setup: picker => { | ||||
|                         picker.on("select", _ => { table.bootstrapTable("triggerSearch"); }); | ||||
|                         // replace "Cancel" behaviour to "Clear" | ||||
|                         picker.onClickCancelButton = element => { | ||||
|                             if (picker.isCancelButton(element)) { | ||||
|                                 picker.clear(); | ||||
|                                 picker.hide(); | ||||
|                                 table.bootstrapTable("triggerSearch"); | ||||
|                             } | ||||
|                         }; | ||||
|                     }, | ||||
|                 }); | ||||
|             }, | ||||
|             onUncheck: onCheckFunction, | ||||
|             onUncheckAll: onCheckFunction, | ||||
|         }); | ||||
|  | ||||
|         restoreAutoReloadSettings(tableAutoReloadButton, tableAutoReloadInput); | ||||
|  | ||||
|         selectRepository(); | ||||
|         {% if autorefresh_intervals %} | ||||
|             toggleTableAutoReload(); | ||||
|         {% endif %} | ||||
|     }); | ||||
| </script> | ||||
|  | ||||
| @ -53,8 +53,7 @@ SigLevel = Database{% if has_repo_signed %}Required{% else %}Never{% endif %} Pa | ||||
|                    data-show-search-clear-button="true" | ||||
|                    data-sortable="true" | ||||
|                    data-sort-name="base" | ||||
|                    data-sort-order="asc" | ||||
|                    data-toggle="table"> | ||||
|                    data-sort-order="asc"> | ||||
|                 <thead class="table-primary"> | ||||
|                     <tr> | ||||
|                         <th data-sortable="true" data-switchable="false" data-field="name" data-filter-control="input" data-filter-control-placeholder="(any package)">package</th> | ||||
| @ -128,36 +127,38 @@ SigLevel = Database{% if has_repo_signed %}Required{% else %}Never{% endif %} Pa | ||||
|             } | ||||
|  | ||||
|             ready(_ => { | ||||
|                 table.on("created-controls.bs.table", _ => { | ||||
|                     new easepick.create({ | ||||
|                         element: document.querySelector(".bootstrap-table-filter-control-timestamp"), | ||||
|                         css: [ | ||||
|                             "https://cdn.jsdelivr.net/npm/@easepick/bundle@1.2.1/dist/index.css", | ||||
|                         ], | ||||
|                         grid: 2, | ||||
|                         calendars: 2, | ||||
|                         autoApply: false, | ||||
|                         locale: { | ||||
|                             cancel: "Clear", | ||||
|                         }, | ||||
|                         RangePlugin: { | ||||
|                             tooltip: false, | ||||
|                         }, | ||||
|                         plugins: [ | ||||
|                             "RangePlugin", | ||||
|                         ], | ||||
|                         setup: picker => { | ||||
|                             picker.on("select", _ => { table.bootstrapTable("triggerSearch"); }); | ||||
|                             // replace "Cancel" behaviour to "Clear" | ||||
|                             picker.onClickCancelButton = element => { | ||||
|                                 if (picker.isCancelButton(element)) { | ||||
|                                     picker.clear(); | ||||
|                                     picker.hide(); | ||||
|                                     table.bootstrapTable("triggerSearch"); | ||||
|                                 } | ||||
|                             }; | ||||
|                         }, | ||||
|                     }); | ||||
|                 table.bootstrapTable({ | ||||
|                     onCreatedControls: _ => { | ||||
|                         new easepick.create({ | ||||
|                             element: document.querySelector(".bootstrap-table-filter-control-timestamp"), | ||||
|                             css: [ | ||||
|                                 "https://cdn.jsdelivr.net/npm/@easepick/bundle@1.2.1/dist/index.css", | ||||
|                             ], | ||||
|                             grid: 2, | ||||
|                             calendars: 2, | ||||
|                             autoApply: false, | ||||
|                             locale: { | ||||
|                                 cancel: "Clear", | ||||
|                             }, | ||||
|                             RangePlugin: { | ||||
|                                 tooltip: false, | ||||
|                             }, | ||||
|                             plugins: [ | ||||
|                                 "RangePlugin", | ||||
|                             ], | ||||
|                             setup: picker => { | ||||
|                                 picker.on("select", _ => { table.bootstrapTable("triggerSearch"); }); | ||||
|                                 // replace "Cancel" behaviour to "Clear" | ||||
|                                 picker.onClickCancelButton = element => { | ||||
|                                     if (picker.isCancelButton(element)) { | ||||
|                                         picker.clear(); | ||||
|                                         picker.hide(); | ||||
|                                         table.bootstrapTable("triggerSearch"); | ||||
|                                     } | ||||
|                                 }; | ||||
|                             }, | ||||
|                         }); | ||||
|                     }, | ||||
|                 }); | ||||
|             }); | ||||
|         </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/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/@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-table@1.23.2/dist/bootstrap-table.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.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.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.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/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/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/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/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> | ||||
|     async function copyToClipboard(text, button) { | ||||
| @ -58,6 +59,20 @@ | ||||
|         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) { | ||||
|         if (status === "pending") return ["bg-warning"]; | ||||
|         if (status === "building") return ["bg-warning"]; | ||||
| @ -106,6 +121,12 @@ | ||||
|             .catch(error => onFailure && onFailure(error)); | ||||
|     } | ||||
|  | ||||
|     function readOptional(extractor, callback) { | ||||
|         for (let value = extractor(); !!value; value = null) { | ||||
|             callback(value); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     function ready(fn) { | ||||
|         if (document.readyState === "complete" || document.readyState === "interactive") { | ||||
|             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) { | ||||
|         return String(string) | ||||
|             .replace(/&/g, "&") | ||||
| @ -133,7 +159,86 @@ | ||||
|         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"); | ||||
|         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-icons@1.11.1/font/bootstrap-icons.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.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/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> | ||||
|     .pre-scrollable { | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| .TH AHRIMAN "1" "2025\-07\-14" "ahriman 2.19.1" "ArcH linux ReposItory MANager" | ||||
| .TH AHRIMAN "1" "2025\-06\-29" "ahriman 2.19.0" "ArcH linux ReposItory MANager" | ||||
| .SH NAME | ||||
| ahriman \- ArcH linux ReposItory MANager | ||||
| .SH SYNOPSIS | ||||
|  | ||||
| @ -8,7 +8,7 @@ services: | ||||
|       AHRIMAN_OUTPUT: console | ||||
|       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} | ||||
|       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_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock | ||||
|  | ||||
|  | ||||
| @ -8,7 +8,7 @@ services: | ||||
|       AHRIMAN_OUTPUT: console | ||||
|       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} | ||||
|       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_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock | ||||
|  | ||||
|  | ||||
| @ -8,7 +8,7 @@ services: | ||||
|       AHRIMAN_OUTPUT: console | ||||
|       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} | ||||
|       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_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock | ||||
|  | ||||
| @ -62,7 +62,7 @@ services: | ||||
|       AHRIMAN_OUTPUT: console | ||||
|       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} | ||||
|       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_SERVER: http://frontend/repo/$$repo/$$arch | ||||
|  | ||||
|  | ||||
| @ -12,7 +12,7 @@ services: | ||||
|       AHRIMAN_PACMAN_MIRROR: https://de.mirror.archlinux32.org/$$arch/$$repo | ||||
|       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} | ||||
|       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_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock | ||||
|  | ||||
|  | ||||
| @ -8,8 +8,8 @@ services: | ||||
|       AHRIMAN_OUTPUT: console | ||||
|       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} | ||||
|       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_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_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_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock | ||||
|  | ||||
|  | ||||
| @ -9,7 +9,7 @@ services: | ||||
|       AHRIMAN_OAUTH_CLIENT_SECRET: ${AHRIMAN_OAUTH_CLIENT_SECRET} | ||||
|       AHRIMAN_OUTPUT: console | ||||
|       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_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock | ||||
|  | ||||
|  | ||||
| @ -6,7 +6,7 @@ services: | ||||
|     environment: | ||||
|       AHRIMAN_DEBUG: yes | ||||
|       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 | ||||
|  | ||||
|     configs: | ||||
|  | ||||
| @ -8,7 +8,7 @@ services: | ||||
|       AHRIMAN_OUTPUT: console | ||||
|       AHRIMAN_PASSWORD: ${AHRIMAN_PASSWORD} | ||||
|       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_UNIX_SOCKET: /var/lib/ahriman/ahriman/ahriman.sock | ||||
|  | ||||
|  | ||||
| @ -17,4 +17,4 @@ | ||||
| # You should have received a copy of the GNU General Public License | ||||
| # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||
| # | ||||
| __version__ = "2.19.1" | ||||
| __version__ = "2.19.0" | ||||
|  | ||||
							
								
								
									
										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.exceptions import OptionError | ||||
| from ahriman.core.formatters import AurPrinter | ||||
| from ahriman.core.types import Comparable | ||||
| from ahriman.models.aur_package import AURPackage | ||||
| from ahriman.models.repository_id import RepositoryId | ||||
|  | ||||
| @ -115,7 +116,7 @@ class Search(Handler): | ||||
|             raise OptionError(sort_by) | ||||
|         # always sort by package name at the last | ||||
|         # 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) | ||||
|         return sorted(packages, key=comparator) | ||||
|  | ||||
|  | ||||
| @ -72,16 +72,17 @@ class Setup(Handler): | ||||
|  | ||||
|         application = Application(repository_id, configuration, report=report) | ||||
|  | ||||
|         Setup.configuration_create_makepkg(args.packager, args.makeflags_jobs, application.repository.paths) | ||||
|         Setup.executable_create(application.repository.paths, repository_id) | ||||
|         repository_server = f"file://{application.repository.paths.repository}" if args.server is None else args.server | ||||
|         Setup.configuration_create_devtools( | ||||
|             repository_id, args.from_configuration, args.mirror, args.multilib, repository_server) | ||||
|         Setup.configuration_create_sudo(application.repository.paths, repository_id) | ||||
|         with application.repository.paths.preserve_owner(): | ||||
|             Setup.configuration_create_makepkg(args.packager, args.makeflags_jobs, application.repository.paths) | ||||
|             Setup.executable_create(application.repository.paths, repository_id) | ||||
|             repository_server = f"file://{application.repository.paths.repository}" if args.server is None else args.server | ||||
|             Setup.configuration_create_devtools( | ||||
|                 repository_id, args.from_configuration, args.mirror, args.multilib, repository_server) | ||||
|             Setup.configuration_create_sudo(application.repository.paths, repository_id) | ||||
|  | ||||
|         application.repository.repo.init() | ||||
|         # lazy database sync | ||||
|         application.repository.pacman.handle  # pylint: disable=pointless-statement | ||||
|             application.repository.repo.init() | ||||
|             # lazy database sync | ||||
|             application.repository.pacman.handle  # pylint: disable=pointless-statement | ||||
|  | ||||
|     @staticmethod | ||||
|     def _set_service_setup_parser(root: SubParserAction) -> argparse.ArgumentParser: | ||||
| @ -280,6 +281,5 @@ class Setup(Handler): | ||||
|         command = Setup.build_command(paths.root, repository_id) | ||||
|         command.unlink(missing_ok=True) | ||||
|         command.symlink_to(Setup.ARCHBUILD_COMMAND_PATH) | ||||
|         paths.chown(command)  # we would like to keep owner inside ahriman's home | ||||
|  | ||||
|     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.core.configuration import Configuration | ||||
| from ahriman.core.formatters import PackagePrinter, StatusPrinter | ||||
| from ahriman.core.types import Comparable | ||||
| from ahriman.core.utils import enum_values | ||||
| from ahriman.models.build_status import BuildStatus, BuildStatusEnum | ||||
| from ahriman.models.package import Package | ||||
| @ -64,8 +65,8 @@ class Status(Handler): | ||||
|  | ||||
|         Status.check_status(args.exit_code, packages) | ||||
|  | ||||
|         comparator: Callable[[tuple[Package, BuildStatus]], str] = lambda item: item[0].base | ||||
|         filter_fn: Callable[[tuple[Package, BuildStatus]], bool] =\ | ||||
|         comparator: Callable[[tuple[Package, BuildStatus]], Comparable] = lambda item: item[0].base | ||||
|         filter_fn: Callable[[tuple[Package, BuildStatus]], bool] = \ | ||||
|             lambda item: args.status is None or item[1].status == args.status | ||||
|         for package, package_status in sorted(filter(filter_fn, packages), key=comparator): | ||||
|             PackagePrinter(package, package_status)(verbose=args.info) | ||||
|  | ||||
| @ -21,6 +21,7 @@ import argparse | ||||
|  | ||||
| from ahriman.application.handlers.handler import Handler, SubParserAction | ||||
| from ahriman.core.configuration import Configuration | ||||
| from ahriman.core.utils import walk | ||||
| from ahriman.models.repository_id import RepositoryId | ||||
| from ahriman.models.repository_paths import RepositoryPaths | ||||
|  | ||||
| @ -49,6 +50,7 @@ class TreeMigrate(Handler): | ||||
|         target_tree.tree_create() | ||||
|         # perform migration | ||||
|         TreeMigrate.tree_move(current_tree, target_tree) | ||||
|         TreeMigrate.fix_symlinks(target_tree) | ||||
|  | ||||
|     @staticmethod | ||||
|     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) | ||||
|         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 | ||||
|     def tree_move(from_tree: RepositoryPaths, to_tree: RepositoryPaths) -> None: | ||||
|         """ | ||||
|  | ||||
| @ -52,7 +52,7 @@ class Validate(Handler): | ||||
|         """ | ||||
|         from ahriman.core.configuration.validator import Validator | ||||
|  | ||||
|         schema = Validate.schema(repository_id, configuration) | ||||
|         schema = Validate.schema(configuration) | ||||
|         validator = Validator(configuration=configuration, schema=schema) | ||||
|  | ||||
|         if validator.validate(configuration.dump()): | ||||
| @ -83,12 +83,11 @@ class Validate(Handler): | ||||
|         return parser | ||||
|  | ||||
|     @staticmethod | ||||
|     def schema(repository_id: RepositoryId, configuration: Configuration) -> ConfigurationSchema: | ||||
|     def schema(configuration: Configuration) -> ConfigurationSchema: | ||||
|         """ | ||||
|         get schema with triggers | ||||
|  | ||||
|         Args: | ||||
|             repository_id(RepositoryId): repository unique identifier | ||||
|             configuration(Configuration): configuration instance | ||||
|  | ||||
|         Returns: | ||||
| @ -107,12 +106,12 @@ class Validate(Handler): | ||||
|                 continue | ||||
|  | ||||
|             # 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)) | ||||
|                 root[schema_name] = Validate.schema_merge(root.get(schema_name, {}), erased) | ||||
|  | ||||
|             # 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)) | ||||
|  | ||||
|         return root | ||||
|  | ||||
| @ -130,8 +130,8 @@ class Pacman(LazyLogging): | ||||
|             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) | ||||
|         shutil.copy(src, dst) | ||||
|         self.repository_paths.chown(dst) | ||||
|         with self.repository_paths.preserve_owner(dst.parent): | ||||
|             shutil.copy(src, dst) | ||||
|  | ||||
|     def database_init(self, handle: Handle, repository: str, architecture: str) -> DB: | ||||
|         """ | ||||
|  | ||||
| @ -94,6 +94,15 @@ class Remote(SyncHttpClient): | ||||
|                 for package in portion | ||||
|                 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()) | ||||
|  | ||||
|     @classmethod | ||||
|  | ||||
| @ -31,20 +31,21 @@ class Repo(LazyLogging): | ||||
|  | ||||
|     Attributes: | ||||
|         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 | ||||
|         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: | ||||
|             name(str): repository name | ||||
|             paths(RepositoryPaths): repository paths instance | ||||
|             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.paths = paths | ||||
|         self.root = root or paths.repository | ||||
|         self.uid, _ = paths.root_owner | ||||
|         self.sign_args = sign_args | ||||
|  | ||||
| @ -56,45 +57,56 @@ class Repo(LazyLogging): | ||||
|         Returns: | ||||
|             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 | ||||
|  | ||||
|         Args: | ||||
|             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( | ||||
|             "repo-add", *self.sign_args, "-R", str(self.repo_path), str(path), | ||||
|             *command, | ||||
|             exception=BuildError.from_process(path.name), | ||||
|             cwd=self.paths.repository, | ||||
|             cwd=self.root, | ||||
|             logger=self.logger, | ||||
|             user=self.uid) | ||||
|             user=self.uid, | ||||
|         ) | ||||
|  | ||||
|     def init(self) -> None: | ||||
|         """ | ||||
|         create empty repository database. It just calls add with empty arguments | ||||
|         """ | ||||
|         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 | ||||
|  | ||||
|         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 | ||||
|         """ | ||||
|         package_name = package_name or filename.name.rsplit("-", maxsplit=3)[0] | ||||
|  | ||||
|         # 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() | ||||
|  | ||||
|         # remove package from registry | ||||
|         check_output( | ||||
|             "repo-remove", *self.sign_args, str(self.repo_path), package, | ||||
|             exception=BuildError.from_process(package), | ||||
|             cwd=self.paths.repository, | ||||
|             "repo-remove", *self.sign_args, str(self.repo_path), package_name, | ||||
|             exception=BuildError.from_process(package_name), | ||||
|             cwd=self.root, | ||||
|             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 | ||||
| # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||
| # | ||||
| # pylint: disable=too-many-public-methods | ||||
| import configparser | ||||
| import os | ||||
| import shlex | ||||
| import sys | ||||
|  | ||||
| @ -41,7 +43,6 @@ class Configuration(configparser.RawConfigParser): | ||||
|         SYSTEM_CONFIGURATION_PATH(Path): (class attribute) default system configuration path distributed by package | ||||
|         includes(list[Path]): list of includes which were read | ||||
|         path(Path | None): path to root configuration file | ||||
|         repository_id(RepositoryId | None): repository unique identifier | ||||
|  | ||||
|     Examples: | ||||
|         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, | ||||
|             interpolation=ShellInterpolator(), | ||||
|             converters={ | ||||
|                 "intlist": lambda value: list(map(int, shlex.split(value))), | ||||
|                 "list": shlex.split, | ||||
|                 "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.includes: list[Path] = [] | ||||
|  | ||||
| @ -126,6 +128,32 @@ class Configuration(configparser.RawConfigParser): | ||||
|         """ | ||||
|         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 | ||||
|     def repository_name(self) -> str: | ||||
|         """ | ||||
| @ -162,6 +190,7 @@ class Configuration(configparser.RawConfigParser): | ||||
|         """ | ||||
|         configuration = cls() | ||||
|         configuration.load(path) | ||||
|         configuration.load_environment() | ||||
|         configuration.merge_sections(repository_id) | ||||
|         return configuration | ||||
|  | ||||
| @ -236,6 +265,8 @@ class Configuration(configparser.RawConfigParser): | ||||
|  | ||||
|     # pylint and mypy are too stupid to find these methods | ||||
|     # 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 getpath(self, *args: Any, **kwargs: Any) -> Path: ...  # type: ignore[empty-body] | ||||
| @ -284,6 +315,16 @@ class Configuration(configparser.RawConfigParser): | ||||
|         self.read(self.path) | ||||
|         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: | ||||
|         """ | ||||
|         load configuration includes from specified path | ||||
| @ -352,11 +393,16 @@ class Configuration(configparser.RawConfigParser): | ||||
|         """ | ||||
|         reload configuration if possible or raise exception otherwise | ||||
|         """ | ||||
|         # get current properties and validate input | ||||
|         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.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: | ||||
|         """ | ||||
|  | ||||
| @ -45,11 +45,6 @@ CONFIGURATION_SCHEMA: ConfigurationSchema = { | ||||
|                 "path_exists": True, | ||||
|                 "path_type": "dir", | ||||
|             }, | ||||
|             "keep_last_logs": { | ||||
|                 "type": "integer", | ||||
|                 "coerce": "integer", | ||||
|                 "min": 0, | ||||
|             }, | ||||
|             "logging": { | ||||
|                 "type": "path", | ||||
|                 "coerce": "absolute_path", | ||||
| @ -254,6 +249,10 @@ CONFIGURATION_SCHEMA: ConfigurationSchema = { | ||||
|     "repository": { | ||||
|         "type": "dict", | ||||
|         "schema": { | ||||
|             "architecture": { | ||||
|                 "type": "string", | ||||
|                 "empty": False, | ||||
|             }, | ||||
|             "name": { | ||||
|                 "type": "string", | ||||
|                 "empty": False, | ||||
| @ -324,6 +323,15 @@ CONFIGURATION_SCHEMA: ConfigurationSchema = { | ||||
|                 "empty": False, | ||||
|                 "is_url": ["http", "https"], | ||||
|             }, | ||||
|             "autorefresh_intervals": { | ||||
|                 "type": "list", | ||||
|                 "coerce": "list", | ||||
|                 "schema": { | ||||
|                     "type": "integer", | ||||
|                     "coerce": "integer", | ||||
|                     "min": 0, | ||||
|                 }, | ||||
|             }, | ||||
|             "enable_archive_upload": { | ||||
|                 "type": "boolean", | ||||
|                 "coerce": "boolean", | ||||
|  | ||||
| @ -203,8 +203,6 @@ def migrate_package_repository(connection: Connection, configuration: Configurat | ||||
|         configuration(Configuration): configuration instance | ||||
|     """ | ||||
|     _, repository_id = configuration.check_loaded() | ||||
|     if repository_id.is_empty: | ||||
|         return  # no repository available yet | ||||
|  | ||||
|     connection.execute("""update build_queue set repository = :repository""", {"repository": repository_id.id}) | ||||
|     connection.execute("""update package_bases set repository = :repository""", {"repository": repository_id.id}) | ||||
|  | ||||
							
								
								
									
										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 | ||||
|     """ | ||||
|  | ||||
|     def logs_get(self, package_base: str, limit: int = -1, offset: int = 0, | ||||
|                  repository_id: RepositoryId | None = None) -> list[LogRecord]: | ||||
|     def logs_get(self, package_base: str, version: str | None = None, process_id: str | None = None, | ||||
|                  limit: int = -1, offset: int = 0, repository_id: RepositoryId | None = None) -> list[LogRecord]: | ||||
|         """ | ||||
|         extract logs for specified package base | ||||
|  | ||||
|         Args: | ||||
|             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) | ||||
|             offset(int, optional): records offset (Default value = 0) | ||||
|             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 * 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 asc | ||||
|                     """, | ||||
|                     { | ||||
|                         "package_base": package_base, | ||||
|                         "version": version, | ||||
|                         "process_id": process_id, | ||||
|                         "repository": repository_id.id, | ||||
|                         "limit": limit, | ||||
|                         "offset": offset, | ||||
|  | ||||
| @ -25,8 +25,16 @@ from typing import Self | ||||
|  | ||||
| from ahriman.core.configuration import Configuration | ||||
| from ahriman.core.database.migrations import Migrations | ||||
| from ahriman.core.database.operations import AuthOperations, BuildOperations, ChangesOperations, \ | ||||
|     DependenciesOperations, EventOperations, LogsOperations, PackageOperations, PatchOperations | ||||
| from ahriman.core.database.operations import ( | ||||
|     AuthOperations, | ||||
|     BuildOperations, | ||||
|     ChangesOperations, | ||||
|     DependenciesOperations, | ||||
|     EventOperations, | ||||
|     LogsOperations, | ||||
|     PackageOperations, | ||||
|     PatchOperations, | ||||
| ) | ||||
| from ahriman.models.repository_id import RepositoryId | ||||
|  | ||||
|  | ||||
| @ -94,9 +102,13 @@ class SQLite( | ||||
|         sqlite3.register_adapter(list, json.dumps) | ||||
|         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._repository_paths.chown(self.path) | ||||
|  | ||||
|     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) | ||||
| @ -17,7 +17,6 @@ | ||||
| # 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 atexit | ||||
| import logging | ||||
| import uuid | ||||
|  | ||||
| @ -37,7 +36,6 @@ class HttpLogHandler(logging.Handler): | ||||
|     method | ||||
|  | ||||
|     Attributes: | ||||
|         keep_last_records(int): number of last records to keep | ||||
|         reporter(Client): build status reporter instance | ||||
|         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.suppress_errors = suppress_errors | ||||
|         self.keep_last_records = configuration.getint("settings", "keep_last_logs", fallback=0) | ||||
|  | ||||
|     @classmethod | ||||
|     def load(cls, repository_id: RepositoryId, configuration: Configuration, *, report: bool) -> Self: | ||||
| @ -83,7 +80,6 @@ class HttpLogHandler(logging.Handler): | ||||
|         root.addHandler(handler) | ||||
|  | ||||
|         LogRecordId.DEFAULT_PROCESS_ID = str(uuid.uuid4())  # assign default process identifier for log records | ||||
|         atexit.register(handler.rotate) | ||||
|  | ||||
|         return handler | ||||
|  | ||||
| @ -104,9 +100,3 @@ class HttpLogHandler(logging.Handler): | ||||
|             if self.suppress_errors: | ||||
|                 return | ||||
|             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.sign.gpg import GPG | ||||
| from ahriman.core.types import Comparable | ||||
| from ahriman.core.utils import pretty_datetime, pretty_size, utcnow | ||||
| from ahriman.models.repository_id import RepositoryId | ||||
| from ahriman.models.result import Result | ||||
| @ -111,7 +112,7 @@ class JinjaTemplate: | ||||
|         Returns: | ||||
|             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) | ||||
|  | ||||
|     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.report import Report | ||||
| from ahriman.core.status import Client | ||||
| from ahriman.core.types import Comparable | ||||
| from ahriman.models.event import EventType | ||||
| from ahriman.models.package import Package | ||||
| from ahriman.models.repository_id import RepositoryId | ||||
| @ -86,7 +87,7 @@ class RSS(Report, JinjaTemplate): | ||||
|         Returns: | ||||
|             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"]) | ||||
|         return sorted(content, key=comparator, reverse=True) | ||||
|  | ||||
|  | ||||
| @ -19,7 +19,7 @@ | ||||
| # | ||||
| import shutil | ||||
|  | ||||
| from collections.abc import Iterable | ||||
| from collections.abc import Generator, Iterable | ||||
| from pathlib import Path | ||||
| 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.repository.cleaner import Cleaner | ||||
| 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.event import EventType | ||||
| from ahriman.models.package import Package | ||||
| @ -41,6 +41,141 @@ class Executor(PackageInfo, Cleaner): | ||||
|     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, *, | ||||
|                       bump_pkgrel: bool = False) -> Result: | ||||
|         """ | ||||
| @ -55,21 +190,6 @@ class Executor(PackageInfo, Cleaner): | ||||
|         Returns: | ||||
|             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() | ||||
|         local_versions = {package.base: package.version for package in self.packages()} | ||||
|  | ||||
| @ -80,16 +200,21 @@ class Executor(PackageInfo, Cleaner): | ||||
|                 try: | ||||
|                     with self.in_event(single.base, EventType.PackageUpdated, failure=EventType.PackageUpdateFailed): | ||||
|                         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 | ||||
|                         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 | ||||
|                         package_archive = PackageArchive(self.paths.build_root, single, self.pacman, self.scan_paths) | ||||
|                         dependencies = package_archive.depends_on() | ||||
|                         self.reporter.package_dependencies_update(single.base, dependencies) | ||||
|  | ||||
|                         # update result set | ||||
|                         result.add_updated(single) | ||||
|  | ||||
|                 except Exception: | ||||
|                     self.reporter.set_failed(single.base) | ||||
|                     result.add_failed(single) | ||||
| @ -107,19 +232,6 @@ class Executor(PackageInfo, Cleaner): | ||||
|         Returns: | ||||
|             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] = {} | ||||
|         bases_to_remove: list[str] = [] | ||||
|  | ||||
| @ -136,6 +248,7 @@ class Executor(PackageInfo, Cleaner): | ||||
|                 }) | ||||
|                 bases_to_remove.append(local.base) | ||||
|                 result.add_removed(local) | ||||
|  | ||||
|             elif requested.intersection(local.packages.keys()): | ||||
|                 packages_to_remove.update({ | ||||
|                     package: properties.filepath | ||||
| @ -152,11 +265,11 @@ class Executor(PackageInfo, Cleaner): | ||||
|  | ||||
|         # remove packages from repository files | ||||
|         for package, filename in packages_to_remove.items(): | ||||
|             remove_package(package, filename) | ||||
|             self._package_remove(package, filename) | ||||
|  | ||||
|         # remove bases from registered | ||||
|         for package in bases_to_remove: | ||||
|             remove_base(package) | ||||
|             self._package_remove_base(package) | ||||
|  | ||||
|         return result | ||||
|  | ||||
| @ -172,27 +285,6 @@ class Executor(PackageInfo, Cleaner): | ||||
|         Returns: | ||||
|             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()} | ||||
|         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) | ||||
|  | ||||
|                     for description in local.packages.values(): | ||||
|                         rename(description, local.base) | ||||
|                         update_single(description.filename, local.base, packager.key) | ||||
|                         self._archive_rename(description, local.base) | ||||
|                         self._package_update(description.filename, local.base, packager.key) | ||||
|                     self.reporter.set_success(local) | ||||
|                     result.add_updated(local) | ||||
|  | ||||
| @ -216,12 +308,13 @@ class Executor(PackageInfo, Cleaner): | ||||
|                     if local.base in current_packages: | ||||
|                         current_package_archives = set(current_packages[local.base].packages.keys()) | ||||
|                     removed_packages.extend(current_package_archives.difference(local.packages)) | ||||
|  | ||||
|                 except Exception: | ||||
|                     self.reporter.set_failed(local.base) | ||||
|                     result.add_failed(local) | ||||
|                     self.logger.exception("could not process %s", local.base) | ||||
|         self.clear_packages() | ||||
|  | ||||
|         self.clear_packages() | ||||
|         self.process_remove(removed_packages) | ||||
|  | ||||
|         return result | ||||
|  | ||||
| @ -81,6 +81,11 @@ class Client: | ||||
|  | ||||
|         return make_local_client() | ||||
|  | ||||
|     def configuration_reload(self) -> None: | ||||
|         """ | ||||
|         reload configuration | ||||
|         """ | ||||
|  | ||||
|     def event_add(self, event: Event) -> None: | ||||
|         """ | ||||
|         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 | ||||
|  | ||||
|     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 | ||||
|  | ||||
|         Args: | ||||
|             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) | ||||
|             offset(int, optional): records offset (Default value = 0) | ||||
|  | ||||
|  | ||||
| @ -152,19 +152,22 @@ class LocalClient(Client): | ||||
|         """ | ||||
|         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 | ||||
|  | ||||
|         Args: | ||||
|             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) | ||||
|             offset(int, optional): records offset (Default value = 0) | ||||
|  | ||||
|         Returns: | ||||
|             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: | ||||
|         """ | ||||
|  | ||||
| @ -109,7 +109,7 @@ class Watcher(LazyLogging): | ||||
|  | ||||
|     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] | ||||
|  | ||||
|  | ||||
| @ -17,6 +17,7 @@ | ||||
| # You should have received a copy of the GNU General Public License | ||||
| # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||
| # | ||||
| # pylint: disable=too-many-public-methods | ||||
| import contextlib | ||||
|  | ||||
| from urllib.parse import quote_plus as url_encode | ||||
| @ -165,6 +166,13 @@ class WebClient(Client, SyncAhrimanClient): | ||||
|         """ | ||||
|         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: | ||||
|         """ | ||||
|         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), | ||||
|                           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 | ||||
|  | ||||
|         Args: | ||||
|             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) | ||||
|             offset(int, optional): records offset (Default value = 0) | ||||
|  | ||||
| @ -339,6 +350,10 @@ class WebClient(Client, SyncAhrimanClient): | ||||
|             list[LogRecord]: package logs | ||||
|         """ | ||||
|         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): | ||||
|             response = self.make_request("GET", self._logs_url(package_base), params=query) | ||||
|  | ||||
| @ -36,6 +36,7 @@ class Trigger(LazyLogging): | ||||
|         CONFIGURATION_SCHEMA(ConfigurationSchema): (class attribute) configuration schema template | ||||
|         CONFIGURATION_SCHEMA_FALLBACK(str | None): (class attribute) optional fallback option for defining | ||||
|             configuration schema type used | ||||
|         REQUIRES_REPOSITORY(bool): (class attribute) either trigger requires loaded repository or not | ||||
|         configuration(Configuration): configuration instance | ||||
|         repository_id(RepositoryId): repository unique identifier | ||||
|  | ||||
| @ -59,6 +60,7 @@ class Trigger(LazyLogging): | ||||
|  | ||||
|     CONFIGURATION_SCHEMA: ClassVar[ConfigurationSchema] = {} | ||||
|     CONFIGURATION_SCHEMA_FALLBACK: ClassVar[str | None] = None | ||||
|     REQUIRES_REPOSITORY: ClassVar[bool] = True | ||||
|  | ||||
|     def __init__(self, repository_id: RepositoryId, configuration: Configuration) -> None: | ||||
|         """ | ||||
| @ -79,9 +81,18 @@ class Trigger(LazyLogging): | ||||
|         """ | ||||
|         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 | ||||
|     def configuration_schema(cls, repository_id: RepositoryId, | ||||
|                              configuration: Configuration | None) -> ConfigurationSchema: | ||||
|     def configuration_schema(cls, configuration: Configuration | None) -> ConfigurationSchema: | ||||
|         """ | ||||
|         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. | ||||
|  | ||||
|         Args: | ||||
|             repository_id(str): repository unique identifier | ||||
|             configuration(Configuration | None): configuration instance. If set to None, the default schema | ||||
|                 should be returned | ||||
|  | ||||
| @ -101,13 +111,15 @@ class Trigger(LazyLogging): | ||||
|  | ||||
|         result: ConfigurationSchema = {} | ||||
|         for target in cls.configuration_sections(configuration): | ||||
|             if not configuration.has_section(target): | ||||
|                 continue | ||||
|             section, schema_name = configuration.gettype( | ||||
|                 target, repository_id, fallback=cls.CONFIGURATION_SCHEMA_FALLBACK) | ||||
|             if schema_name not in cls.CONFIGURATION_SCHEMA: | ||||
|                 continue | ||||
|             result[section] = cls.CONFIGURATION_SCHEMA[schema_name] | ||||
|             for section in configuration.sections(): | ||||
|                 if not (section == target or section.startswith(f"{target}:")): | ||||
|                     # either repository specific or exact name | ||||
|                     continue | ||||
|                 schema_name = configuration.get(section, "type", fallback=section) | ||||
|  | ||||
|                 if schema_name not in cls.CONFIGURATION_SCHEMA: | ||||
|                     continue | ||||
|                 result[section] = cls.CONFIGURATION_SCHEMA[schema_name] | ||||
|  | ||||
|         return result | ||||
|  | ||||
|  | ||||
| @ -17,6 +17,7 @@ | ||||
| # 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 atexit | ||||
| import contextlib | ||||
| import os | ||||
|  | ||||
| @ -60,17 +61,8 @@ class TriggerLoader(LazyLogging): | ||||
|  | ||||
|     def __init__(self) -> None: | ||||
|         """""" | ||||
|         self._on_stop_requested = False | ||||
|         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 | ||||
|     def load(cls, repository_id: RepositoryId, configuration: Configuration) -> Self: | ||||
|         """ | ||||
| @ -85,8 +77,9 @@ class TriggerLoader(LazyLogging): | ||||
|         """ | ||||
|         instance = cls() | ||||
|         instance.triggers = [ | ||||
|             instance.load_trigger(trigger, repository_id, configuration) | ||||
|             for trigger in instance.selected_triggers(configuration) | ||||
|             trigger | ||||
|             for trigger_name in instance.selected_triggers(configuration) | ||||
|             if (trigger := instance.load_trigger(trigger_name, repository_id, configuration)).is_allowed_to_run | ||||
|         ] | ||||
|  | ||||
|         return instance | ||||
| @ -250,10 +243,11 @@ class TriggerLoader(LazyLogging): | ||||
|         run triggers on load | ||||
|         """ | ||||
|         self.logger.debug("executing triggers on start") | ||||
|         self._on_stop_requested = True | ||||
|         for trigger in self.triggers: | ||||
|             with self.__execute_trigger(trigger): | ||||
|                 trigger.on_start() | ||||
|         # register on_stop call | ||||
|         atexit.register(self.on_stop) | ||||
|  | ||||
|     def on_stop(self) -> None: | ||||
|         """ | ||||
|  | ||||
| @ -17,7 +17,15 @@ | ||||
| # 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 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): | ||||
|  | ||||
| @ -18,13 +18,16 @@ | ||||
| # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||
| # | ||||
| # pylint: disable=too-many-lines | ||||
| import contextlib | ||||
| import datetime | ||||
| import fcntl | ||||
| import io | ||||
| import itertools | ||||
| import logging | ||||
| import os | ||||
| import re | ||||
| import selectors | ||||
| import shutil | ||||
| import subprocess | ||||
|  | ||||
| from collections.abc import Callable, Generator, Iterable, Mapping | ||||
| @ -39,11 +42,13 @@ from ahriman.models.repository_paths import RepositoryPaths | ||||
|  | ||||
|  | ||||
| __all__ = [ | ||||
|     "atomic_move", | ||||
|     "check_output", | ||||
|     "check_user", | ||||
|     "dataclass_view", | ||||
|     "enum_values", | ||||
|     "extract_user", | ||||
|     "filelock", | ||||
|     "filter_json", | ||||
|     "full_version", | ||||
|     "minmax", | ||||
| @ -51,6 +56,7 @@ __all__ = [ | ||||
|     "parse_version", | ||||
|     "partition", | ||||
|     "pretty_datetime", | ||||
|     "pretty_interval", | ||||
|     "pretty_size", | ||||
|     "safe_filename", | ||||
|     "srcinfo_property", | ||||
| @ -64,6 +70,25 @@ __all__ = [ | ||||
| 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 | ||||
| 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, | ||||
| @ -232,6 +257,27 @@ def extract_user() -> str | None: | ||||
|     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]: | ||||
|     """ | ||||
|     filter json object by fields used for json-to-object conversion | ||||
| @ -353,6 +399,28 @@ def pretty_datetime(timestamp: datetime.datetime | float | int | None) -> str: | ||||
|     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: | ||||
|     """ | ||||
|     convert size to string | ||||
|  | ||||
| @ -520,8 +520,7 @@ class Package(LazyLogging): | ||||
|         else: | ||||
|             remote_version = remote.version | ||||
|  | ||||
|         result: int = vercmp(self.version, remote_version) | ||||
|         return result < 0 | ||||
|         return self.vercmp(remote_version) < 0 | ||||
|  | ||||
|     def next_pkgrel(self, local_version: str | None) -> str | None: | ||||
|         """ | ||||
| @ -540,7 +539,7 @@ class Package(LazyLogging): | ||||
|         if local_version is None: | ||||
|             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 | ||||
|  | ||||
|         *_, local_pkgrel = parse_version(local_version) | ||||
| @ -561,6 +560,19 @@ class Package(LazyLogging): | ||||
|         details = "" if self.is_single_package else f""" ({" ".join(sorted(self.packages.keys()))})""" | ||||
|         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]: | ||||
|         """ | ||||
|         generate json package view | ||||
|  | ||||
| @ -17,6 +17,7 @@ | ||||
| # 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 contextlib | ||||
| import os | ||||
| import shutil | ||||
|  | ||||
| @ -84,6 +85,16 @@ class RepositoryPaths(LazyLogging): | ||||
|                 return Path(self.repository_id.architecture)  # legacy tree suffix | ||||
|         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 | ||||
|     def build_root(self) -> Path: | ||||
|         """ | ||||
| @ -221,22 +232,14 @@ class RepositoryPaths(LazyLogging): | ||||
|         stat = path.stat() | ||||
|         return stat.st_uid, stat.st_gid | ||||
|  | ||||
|     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 | ||||
|  | ||||
|     def chown(self, path: Path) -> None: | ||||
|     def _chown(self, path: Path) -> None: | ||||
|         """ | ||||
|         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: | ||||
|             path(Path): path to be chown | ||||
|  | ||||
| @ -256,6 +259,77 @@ class RepositoryPaths(LazyLogging): | ||||
|             set_owner(path) | ||||
|             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: | ||||
|         """ | ||||
|         clear package specific files | ||||
| @ -274,12 +348,14 @@ class RepositoryPaths(LazyLogging): | ||||
|         """ | ||||
|         if self.repository_id.is_empty: | ||||
|             return  # do not even try to create tree in case if no repository id set | ||||
|         for directory in ( | ||||
|                 self.cache, | ||||
|                 self.chroot, | ||||
|                 self.packages, | ||||
|                 self.pacman, | ||||
|                 self.repository, | ||||
|         ): | ||||
|             directory.mkdir(mode=0o755, parents=True, exist_ok=True) | ||||
|             self.chown(directory) | ||||
|  | ||||
|         with self.preserve_owner(): | ||||
|             for directory in ( | ||||
|                     self.archive, | ||||
|                     self.cache, | ||||
|                     self.chroot, | ||||
|                     self.packages, | ||||
|                     self.pacman, | ||||
|                     self.repository, | ||||
|             ): | ||||
|                 directory.mkdir(mode=0o755, parents=True, exist_ok=True) | ||||
|  | ||||
| @ -21,8 +21,18 @@ import aiohttp_jinja2 | ||||
| import logging | ||||
|  | ||||
| from aiohttp.typedefs import Middleware | ||||
| from aiohttp.web import HTTPClientError, HTTPException, HTTPMethodNotAllowed, HTTPNoContent, HTTPServerError, \ | ||||
|     HTTPUnauthorized, Request, StreamResponse, json_response, middleware | ||||
| from aiohttp.web import ( | ||||
|     HTTPClientError, | ||||
|     HTTPException, | ||||
|     HTTPMethodNotAllowed, | ||||
|     HTTPNoContent, | ||||
|     HTTPServerError, | ||||
|     HTTPUnauthorized, | ||||
|     Request, | ||||
|     StreamResponse, | ||||
|     json_response, | ||||
|     middleware, | ||||
| ) | ||||
|  | ||||
| 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.build_options_schema import BuildOptionsSchema | ||||
| 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.dependencies_schema import DependenciesSchema | ||||
| 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.logs_rotate_schema import LogsRotateSchema | ||||
| 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.package_name_schema import PackageNameSchema | ||||
| from ahriman.web.schemas.package_names_schema import PackageNamesSchema | ||||
|  | ||||
| @ -25,11 +25,11 @@ class AURPackageSchema(Schema): | ||||
|     response AUR package schema | ||||
|     """ | ||||
|  | ||||
|     package = fields.String(required=True, metadata={ | ||||
|         "description": "Package base", | ||||
|         "example": "ahriman", | ||||
|     }) | ||||
|     description = fields.String(required=True, metadata={ | ||||
|         "description": "Package description", | ||||
|         "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 | ||||
|     """ | ||||
|  | ||||
|     changes = fields.String(metadata={ | ||||
|         "description": "Package changes in patch format", | ||||
|     }) | ||||
|     last_commit_sha = fields.String(metadata={ | ||||
|         "description": "Last recorded commit hash", | ||||
|         "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 | ||||
|     """ | ||||
|  | ||||
|     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={ | ||||
|         "description": "Amount of packages in building state", | ||||
|         "example": 1, | ||||
| @ -45,7 +33,19 @@ class CountersSchema(Schema): | ||||
|         "description": "Amount of packages in failed state", | ||||
|         "example": 1, | ||||
|     }) | ||||
|     pending = fields.Integer(required=True, metadata={ | ||||
|         "description": "Amount of packages in pending state", | ||||
|         "example": 2, | ||||
|     }) | ||||
|     success = fields.Integer(required=True, metadata={ | ||||
|         "description": "Amount of packages in success state", | ||||
|         "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", | ||||
|         "example": 1680537091, | ||||
|     }) | ||||
|     data = fields.Dict(keys=fields.String(), metadata={ | ||||
|         "description": "Event metadata if available", | ||||
|     }) | ||||
|     event = fields.String(required=True, metadata={ | ||||
|         "description": "Event type", | ||||
|         "example": EventType.PackageUpdated, | ||||
|     }) | ||||
|     message = fields.String(metadata={ | ||||
|         "description": "Event message if available", | ||||
|     }) | ||||
|     object_id = fields.String(required=True, metadata={ | ||||
|         "description": "Event object identifier", | ||||
|         "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", | ||||
|         "example": EventType.PackageUpdated, | ||||
|     }) | ||||
|     object_id = fields.String(metadata={ | ||||
|         "description": "Event object identifier", | ||||
|         "example": "ahriman", | ||||
|     }) | ||||
|     from_date = fields.Integer(metadata={ | ||||
|         "description": "Minimal creation timestamp, inclusive", | ||||
|         "example": 1680537091, | ||||
|     }) | ||||
|     object_id = fields.String(metadata={ | ||||
|         "description": "Event object identifier", | ||||
|         "example": "ahriman", | ||||
|     }) | ||||
|     to_date = fields.Integer(metadata={ | ||||
|         "description": "Maximal creation timestamp, exclusive", | ||||
|         "example": 1680537091, | ||||
|  | ||||
| @ -33,10 +33,10 @@ class LogSchema(Schema): | ||||
|     message = fields.String(required=True, metadata={ | ||||
|         "description": "Log message", | ||||
|     }) | ||||
|     process_id = fields.String(metadata={ | ||||
|         "description": "Process unique identifier", | ||||
|     }) | ||||
|     version = fields.String(required=True, metadata={ | ||||
|         "description": "Package version to tag", | ||||
|         "example": __version__, | ||||
|     }) | ||||
|     process_id = fields.String(metadata={ | ||||
|         "description": "Process unique identifier", | ||||
|     }) | ||||
|  | ||||
| @ -25,11 +25,11 @@ class LoginSchema(Schema): | ||||
|     request login schema | ||||
|     """ | ||||
|  | ||||
|     username = fields.String(required=True, metadata={ | ||||
|         "description": "Login username", | ||||
|         "example": "user", | ||||
|     }) | ||||
|     password = fields.String(required=True, metadata={ | ||||
|         "description": "Login password", | ||||
|         "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", | ||||
|         "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={ | ||||
|         "description": "Package test dependencies list", | ||||
|         "example": ["python-pytest"], | ||||
|     }) | ||||
|     depends = fields.List(fields.String(), metadata={ | ||||
|         "description": "Package dependencies list", | ||||
|         "example": ["devtools"], | ||||
|     }) | ||||
|     description = fields.String(metadata={ | ||||
|         "description": "Package description", | ||||
|         "example": "ArcH linux ReposItory MANager", | ||||
| @ -73,6 +65,14 @@ class PackagePropertiesSchema(Schema): | ||||
|         "description": "Package licenses", | ||||
|         "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={ | ||||
|         "description": "Package provides list", | ||||
|         "example": ["ahriman-git"], | ||||
|  | ||||
| @ -32,18 +32,18 @@ class PackageSchema(Schema): | ||||
|         "description": "Package base", | ||||
|         "example": "ahriman", | ||||
|     }) | ||||
|     version = fields.String(required=True, metadata={ | ||||
|         "description": "Package version", | ||||
|         "example": __version__, | ||||
|     }) | ||||
|     remote = fields.Nested(RemoteSchema(), required=True, metadata={ | ||||
|         "description": "Package remote properties", | ||||
|     packager = fields.String(metadata={ | ||||
|         "description": "packager for the last success package build", | ||||
|         "example": "ahriman bot <ahriman@example.com>", | ||||
|     }) | ||||
|     packages = fields.Dict( | ||||
|         keys=fields.String(), values=fields.Nested(PackagePropertiesSchema()), required=True, metadata={ | ||||
|             "description": "Packages which belong to this base", | ||||
|         }) | ||||
|     packager = fields.String(metadata={ | ||||
|         "description": "packager for the last success package build", | ||||
|         "example": "ahriman bot <ahriman@example.com>", | ||||
|     remote = fields.Nested(RemoteSchema(), required=True, metadata={ | ||||
|         "description": "Package remote properties", | ||||
|     }) | ||||
|     version = fields.String(required=True, metadata={ | ||||
|         "description": "Package version", | ||||
|         "example": __version__, | ||||
|     }) | ||||
|  | ||||
| @ -25,19 +25,19 @@ class RepositoryStatsSchema(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={ | ||||
|         "description": "Total archive size of the packages in bytes", | ||||
|         "example": 42000, | ||||
|     }) | ||||
|     bases = fields.Int(metadata={ | ||||
|         "description": "Amount of unique packages bases", | ||||
|         "example": 2, | ||||
|     }) | ||||
|     installed_size = fields.Int(metadata={ | ||||
|         "description": "Total installed size of the packages in bytes", | ||||
|         "example": 42000000, | ||||
|     }) | ||||
|     packages = fields.Int(metadata={ | ||||
|         "description": "Amount of unique packages", | ||||
|         "example": 4, | ||||
|     }) | ||||
|  | ||||
| @ -25,7 +25,7 @@ class SearchSchema(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", | ||||
|         "example": ["ahriman"], | ||||
|     }) | ||||
|  | ||||
| @ -22,6 +22,7 @@ import aiohttp_jinja2 | ||||
| from typing import Any, ClassVar | ||||
|  | ||||
| from ahriman.core.auth.helpers import authorized_userid | ||||
| from ahriman.core.utils import pretty_interval | ||||
| from ahriman.models.user_access import UserAccess | ||||
| from ahriman.web.apispec import aiohttp_apispec | ||||
| from ahriman.web.views.base import BaseView | ||||
| @ -37,6 +38,10 @@ class IndexView(BaseView): | ||||
|             * control - HTML to insert for login control, HTML string, required | ||||
|             * enabled - whether authorization is enabled by configuration or not, boolean, required | ||||
|             * 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 | ||||
|         * index_url - url to the repository index, string, optional | ||||
|         * repositories - list of repositories unique identifiers, required | ||||
| @ -66,8 +71,19 @@ class IndexView(BaseView): | ||||
|             "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 { | ||||
|             "auth": auth, | ||||
|             "autorefresh_intervals": sorted(autorefresh_intervals, key=lambda interval: interval["interval"]), | ||||
|             "docs_enabled": aiohttp_apispec is not None, | ||||
|             "index_url": self.configuration.get("web", "index_url", fallback=None), | ||||
|             "repositories": [ | ||||
|  | ||||
| @ -21,6 +21,7 @@ from aiohttp.web import HTTPBadRequest, HTTPNoContent, Response, json_response | ||||
| from collections.abc import Callable | ||||
| from typing import ClassVar | ||||
|  | ||||
| from ahriman.core.types import Comparable | ||||
| from ahriman.models.user_access import UserAccess | ||||
| from ahriman.models.worker import Worker | ||||
| from ahriman.web.apispec.decorators import apidocs | ||||
| @ -74,7 +75,7 @@ class WorkersView(BaseView): | ||||
|         """ | ||||
|         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)] | ||||
|  | ||||
|         return json_response(response) | ||||
|  | ||||
| @ -90,7 +90,7 @@ class LogsView(StatusViewGuard, BaseView): | ||||
|  | ||||
|         try: | ||||
|             _, 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: | ||||
|             raise HTTPNotFound(reason=f"Package {package_base} is unknown") | ||||
|  | ||||
|  | ||||
| @ -25,8 +25,12 @@ from ahriman.models.build_status import BuildStatusEnum | ||||
| from ahriman.models.package import Package | ||||
| from ahriman.models.user_access import UserAccess | ||||
| from ahriman.web.apispec.decorators import apidocs | ||||
| from ahriman.web.schemas import PackageNameSchema, PackageStatusSchema, PackageStatusSimplifiedSchema, \ | ||||
|     RepositoryIdSchema | ||||
| from ahriman.web.schemas import ( | ||||
|     PackageNameSchema, | ||||
|     PackageStatusSchema, | ||||
|     PackageStatusSimplifiedSchema, | ||||
|     RepositoryIdSchema, | ||||
| ) | ||||
| from ahriman.web.views.base import BaseView | ||||
| from ahriman.web.views.status_view_guard import StatusViewGuard | ||||
|  | ||||
|  | ||||
| @ -23,6 +23,7 @@ from aiohttp.web import HTTPNoContent, Response, json_response | ||||
| from collections.abc import Callable | ||||
| from typing import ClassVar | ||||
|  | ||||
| from ahriman.core.types import Comparable | ||||
| from ahriman.models.build_status import BuildStatus | ||||
| from ahriman.models.package import Package | ||||
| from ahriman.models.user_access import UserAccess | ||||
| @ -68,7 +69,7 @@ class PackagesView(StatusViewGuard, BaseView): | ||||
|         repository_id = self.repository_id() | ||||
|         packages = self.service(repository_id).packages | ||||
|  | ||||
|         comparator: Callable[[tuple[Package, BuildStatus]], str] = lambda items: items[0].base | ||||
|         comparator: Callable[[tuple[Package, BuildStatus]], Comparable] = lambda items: items[0].base | ||||
|         response = [ | ||||
|             { | ||||
|                 "package": package.view(), | ||||
|  | ||||
							
								
								
									
										84
									
								
								src/ahriman/web/views/v1/service/config.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								src/ahriman/web/views/v1/service/config.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/>. | ||||
| # | ||||
| from aiohttp.web import HTTPNoContent, Response, json_response | ||||
| from typing import ClassVar | ||||
|  | ||||
| from ahriman.core.formatters import ConfigurationPrinter | ||||
| from ahriman.models.user_access import UserAccess | ||||
| from ahriman.web.apispec.decorators import apidocs | ||||
| from ahriman.web.schemas import ConfigurationSchema | ||||
| from ahriman.web.views.base import BaseView | ||||
|  | ||||
|  | ||||
| class ConfigView(BaseView): | ||||
|     """ | ||||
|     configuration control view | ||||
|  | ||||
|     Attributes: | ||||
|         GET_PERMISSION(UserAccess): (class attribute) get permissions of self | ||||
|         POST_PERMISSION(UserAccess): (class attribute) post permissions of self | ||||
|     """ | ||||
|  | ||||
|     GET_PERMISSION = POST_PERMISSION = UserAccess.Full  # type: ClassVar[UserAccess] | ||||
|     ROUTES = ["/api/v1/service/config"] | ||||
|  | ||||
|     @apidocs( | ||||
|         tags=["Actions"], | ||||
|         summary="Get configuration", | ||||
|         description="Get current web service configuration as nested dictionary", | ||||
|         permission=GET_PERMISSION, | ||||
|         schema=ConfigurationSchema(many=True), | ||||
|     ) | ||||
|     async def get(self) -> Response: | ||||
|         """ | ||||
|         get current web service configuration | ||||
|  | ||||
|         Returns: | ||||
|             Response: current web service configuration as nested dictionary | ||||
|         """ | ||||
|         dump = self.configuration.dump() | ||||
|  | ||||
|         response = [ | ||||
|             { | ||||
|                 "section": section, | ||||
|                 "key": key, | ||||
|                 "value": value, | ||||
|             } for section, values in dump.items() | ||||
|             for key, value in values.items() | ||||
|             if key not in ConfigurationPrinter.HIDE_KEYS | ||||
|         ] | ||||
|         return json_response(response) | ||||
|  | ||||
|     @apidocs( | ||||
|         tags=["Actions"], | ||||
|         summary="Reload configuration", | ||||
|         description="Reload configuration from current files", | ||||
|         permission=POST_PERMISSION, | ||||
|     ) | ||||
|     async def post(self) -> None: | ||||
|         """ | ||||
|         reload web service configuration | ||||
|  | ||||
|         Raises: | ||||
|             HTTPNoContent: on success response | ||||
|         """ | ||||
|         self.configuration.reload() | ||||
|  | ||||
|         raise HTTPNoContent | ||||
| @ -22,6 +22,7 @@ from collections.abc import Callable | ||||
| from typing import ClassVar | ||||
|  | ||||
| from ahriman.core.alpm.remote import AUR | ||||
| from ahriman.core.types import Comparable | ||||
| from ahriman.models.aur_package import AURPackage | ||||
| from ahriman.models.user_access import UserAccess | ||||
| from ahriman.web.apispec.decorators import apidocs | ||||
| @ -70,7 +71,12 @@ class SearchView(BaseView): | ||||
|         if not packages: | ||||
|             raise HTTPNotFound(reason=f"No packages found for terms: {search}") | ||||
|  | ||||
|         comparator: Callable[[AURPackage], str] = lambda item: str(item.package_base) | ||||
|         comparator: Callable[[AURPackage], Comparable] = \ | ||||
|             lambda item: ( | ||||
|                 item.package_base not in search,  # inverted because False < True | ||||
|                 not any(item.package_base.startswith(term) for term in search),  # same as above | ||||
|                 item.package_base, | ||||
|         ) | ||||
|         response = [ | ||||
|             { | ||||
|                 "package": package.package_base, | ||||
|  | ||||
| @ -26,6 +26,7 @@ from tempfile import NamedTemporaryFile | ||||
| from typing import ClassVar | ||||
|  | ||||
| from ahriman.core.configuration import Configuration | ||||
| from ahriman.core.utils import atomic_move | ||||
| from ahriman.models.repository_paths import RepositoryPaths | ||||
| from ahriman.models.user_access import UserAccess | ||||
| from ahriman.web.apispec.decorators import apidocs | ||||
| @ -152,10 +153,8 @@ class UploadView(BaseView): | ||||
|  | ||||
|             files.append(await self.save_file(part, target, max_body_size=max_body_size)) | ||||
|  | ||||
|         # and now we can rename files, which is relatively fast operation | ||||
|         # it is probably good way to call lock here, however | ||||
|         for filename, current_location in files: | ||||
|             target_location = current_location.parent / filename | ||||
|             current_location.rename(target_location) | ||||
|             atomic_move(current_location, target_location) | ||||
|  | ||||
|         raise HTTPCreated | ||||
|  | ||||
| @ -17,18 +17,22 @@ | ||||
| # 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 itertools | ||||
|  | ||||
| from aiohttp.web import Response, json_response | ||||
| from dataclasses import replace | ||||
| from typing import ClassVar | ||||
|  | ||||
| from ahriman.models.user_access import UserAccess | ||||
| from ahriman.web.apispec.decorators import apidocs | ||||
| from ahriman.web.schemas import LogSchema, PackageNameSchema, PaginationSchema | ||||
| from ahriman.web.schemas import LogSchema, LogsSearchSchema, PackageNameSchema | ||||
| from ahriman.web.views.base import BaseView | ||||
| from ahriman.web.views.status_view_guard import StatusViewGuard | ||||
|  | ||||
|  | ||||
| class LogsView(StatusViewGuard, BaseView): | ||||
|     """ | ||||
|     """        else: | ||||
|  | ||||
|     package logs web view | ||||
|  | ||||
|     Attributes: | ||||
| @ -47,7 +51,7 @@ class LogsView(StatusViewGuard, BaseView): | ||||
|         error_404_description="Package base and/or repository are unknown", | ||||
|         schema=LogSchema(many=True), | ||||
|         match_schema=PackageNameSchema, | ||||
|         query_schema=PaginationSchema, | ||||
|         query_schema=LogsSearchSchema, | ||||
|     ) | ||||
|     async def get(self) -> Response: | ||||
|         """ | ||||
| @ -61,8 +65,19 @@ class LogsView(StatusViewGuard, BaseView): | ||||
|         """ | ||||
|         package_base = self.request.match_info["package"] | ||||
|         limit, offset = self.page() | ||||
|         version = self.request.query.get("version", None) | ||||
|         process = self.request.query.get("process_id", None) | ||||
|  | ||||
|         logs = self.service(package_base=package_base).package_logs_get(package_base, limit, offset) | ||||
|         logs = self.service(package_base=package_base).package_logs_get(package_base, version, process, limit, offset) | ||||
|  | ||||
|         head = self.request.query.get("head", "false") | ||||
|         # pylint: disable=protected-access | ||||
|         if self.configuration._convert_to_boolean(head):  # type: ignore[attr-defined] | ||||
|             # logs should be sorted already | ||||
|             logs = [ | ||||
|                 replace(next(log_records), message="")  # remove messages | ||||
|                 for _, log_records in itertools.groupby(logs, lambda log_record: log_record.log_record_id) | ||||
|             ] | ||||
|  | ||||
|         response = [log_record.view() for log_record in logs] | ||||
|         return json_response(response) | ||||
|  | ||||
| @ -166,11 +166,16 @@ def setup_server(configuration: Configuration, spawner: Spawn, repositories: lis | ||||
|     # package cache | ||||
|     if not repositories: | ||||
|         raise InitializeError("No repositories configured, exiting") | ||||
|     database = SQLite.load(configuration) | ||||
|     watchers: dict[RepositoryId, Watcher] = {} | ||||
|     configuration_path, _ = configuration.check_loaded() | ||||
|     for repository_id in repositories: | ||||
|         application.logger.info("load repository %s", repository_id) | ||||
|         client = Client.load(repository_id, configuration, database, report=False)  # explicitly load local client | ||||
|         # load settings explicitly for architecture if any | ||||
|         repository_configuration = Configuration.from_path(configuration_path, repository_id) | ||||
|         # load database instance, because it holds identifier | ||||
|         database = SQLite.load(repository_configuration) | ||||
|         # explicitly load local client | ||||
|         client = Client.load(repository_id, repository_configuration, database, report=False) | ||||
|         watchers[repository_id] = Watcher(client) | ||||
|     application[WatcherKey] = watchers | ||||
|     # workers cache | ||||
| @ -179,6 +184,7 @@ def setup_server(configuration: Configuration, spawner: Spawn, repositories: lis | ||||
|     application[SpawnKey] = spawner | ||||
|  | ||||
|     application.logger.info("setup authorization") | ||||
|     database = SQLite.load(configuration) | ||||
|     validator = application[AuthKey] = Auth.load(configuration, database) | ||||
|     if validator.enabled: | ||||
|         from ahriman.web.middlewares.auth_handler import setup_auth | ||||
|  | ||||
| @ -37,6 +37,7 @@ SUBPACKAGES = { | ||||
|     "ahriman-triggers": [ | ||||
|         prefix / "share" / "ahriman" / "settings" / "ahriman.ini.d" / "00-triggers.ini", | ||||
|         site_packages / "ahriman" / "application" / "handlers" / "triggers_support.py", | ||||
|         site_packages / "ahriman" / "core" / "archive", | ||||
|         site_packages / "ahriman" / "core" / "distributed", | ||||
|         site_packages / "ahriman" / "core" / "support", | ||||
|     ], | ||||
|  | ||||
| @ -141,7 +141,7 @@ def test_add_remote_missing(application_packages: ApplicationPackages, mocker: M | ||||
|     """ | ||||
|     must raise UnknownPackageError if remote package wasn't found | ||||
|     """ | ||||
|     mocker.patch("requests.get", side_effect=Exception()) | ||||
|     mocker.patch("requests.get", side_effect=Exception) | ||||
|     with pytest.raises(UnknownPackageError): | ||||
|         application_packages._add_remote("url") | ||||
|  | ||||
|  | ||||
| @ -135,7 +135,7 @@ def test_unknown_no_aur(application_repository: ApplicationRepository, package_a | ||||
|     must return empty list in case if there is locally stored PKGBUILD | ||||
|     """ | ||||
|     mocker.patch("ahriman.core.repository.repository.Repository.packages", return_value=[package_ahriman]) | ||||
|     mocker.patch("ahriman.models.package.Package.from_aur", side_effect=Exception()) | ||||
|     mocker.patch("ahriman.models.package.Package.from_aur", side_effect=Exception) | ||||
|     mocker.patch("ahriman.models.package.Package.from_build", return_value=package_ahriman) | ||||
|     mocker.patch("pathlib.Path.is_dir", return_value=True) | ||||
|     mocker.patch("ahriman.core.build_tools.sources.Sources.has_remotes", return_value=False) | ||||
| @ -149,7 +149,7 @@ def test_unknown_no_aur_no_local(application_repository: ApplicationRepository, | ||||
|     must return list of packages missing in aur and in local storage | ||||
|     """ | ||||
|     mocker.patch("ahriman.core.repository.repository.Repository.packages", return_value=[package_ahriman]) | ||||
|     mocker.patch("ahriman.models.package.Package.from_aur", side_effect=Exception()) | ||||
|     mocker.patch("ahriman.models.package.Package.from_aur", side_effect=Exception) | ||||
|     mocker.patch("pathlib.Path.is_dir", return_value=False) | ||||
|  | ||||
|     packages = application_repository.unknown() | ||||
|  | ||||
| @ -46,7 +46,7 @@ def test_call_exception(args: argparse.Namespace, configuration: Configuration, | ||||
|     """ | ||||
|     args.configuration = Path("") | ||||
|     args.quiet = False | ||||
|     mocker.patch("ahriman.core.configuration.Configuration.from_path", side_effect=Exception()) | ||||
|     mocker.patch("ahriman.core.configuration.Configuration.from_path", side_effect=Exception) | ||||
|     logging_mock = mocker.patch("logging.Logger.exception") | ||||
|  | ||||
|     _, repository_id = configuration.check_loaded() | ||||
| @ -60,7 +60,7 @@ def test_call_exit_code(args: argparse.Namespace, configuration: Configuration, | ||||
|     """ | ||||
|     args.configuration = Path("") | ||||
|     args.quiet = False | ||||
|     mocker.patch("ahriman.core.configuration.Configuration.from_path", side_effect=ExitCode()) | ||||
|     mocker.patch("ahriman.core.configuration.Configuration.from_path", side_effect=ExitCode) | ||||
|     logging_mock = mocker.patch("logging.Logger.exception") | ||||
|  | ||||
|     _, repository_id = configuration.check_loaded() | ||||
|  | ||||
							
								
								
									
										27
									
								
								tests/ahriman/application/handlers/test_handler_reload.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								tests/ahriman/application/handlers/test_handler_reload.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,27 @@ | ||||
| import argparse | ||||
|  | ||||
| from pytest_mock import MockerFixture | ||||
|  | ||||
| from ahriman.application.handlers.reload import Reload | ||||
| from ahriman.core.configuration import Configuration | ||||
| from ahriman.core.repository import Repository | ||||
|  | ||||
|  | ||||
| def test_run(args: argparse.Namespace, configuration: Configuration, repository: Repository, | ||||
|              mocker: MockerFixture) -> None: | ||||
|     """ | ||||
|     must run command | ||||
|     """ | ||||
|     mocker.patch("ahriman.core.repository.Repository.load", return_value=repository) | ||||
|     run_mock = mocker.patch("ahriman.core.status.local_client.LocalClient.configuration_reload") | ||||
|  | ||||
|     _, repository_id = configuration.check_loaded() | ||||
|     Reload.run(args, repository_id, configuration, report=False) | ||||
|     run_mock.assert_called_once_with() | ||||
|  | ||||
|  | ||||
| def test_disallow_multi_architecture_run() -> None: | ||||
|     """ | ||||
|     must not allow multi architecture run | ||||
|     """ | ||||
|     assert not Reload.ALLOW_MULTI_ARCHITECTURE_RUN | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user