HEX
Server: LiteSpeed
System: Linux d8 4.18.0-553.121.1.lve.el8.x86_64 #1 SMP Thu Apr 30 16:40:41 UTC 2026 x86_64
User: wbwebdes (3015)
PHP: 8.1.31
Disabled: exec,system,passthru,shell_exec,proc_close,proc_open,dl,popen,show_source,posix_kill,posix_mkfifo,posix_getpwuid,posix_setpgid,posix_setsid,posix_setuid,posix_setgid,posix_seteuid,posix_setegid,posix_uname
Upload Files
File: /home/wbwebdes/domains/uren-registratie.blankevoort.net/public_html/src/Plugin/PackageManager.php
<?php

/*
 * This file is part of the Kimai time-tracking app.
 *
 * For the full copyright and license information, please view the LICENSE
 * file that was distributed with this source code.
 */

namespace App\Plugin;

/**
 * @internal
 */
final class PackageManager
{
    public const PACKAGE_DIR = 'var/packages';

    public function __construct(private readonly string $projectDirectory)
    {
    }

    /**
     * @return Package[]
     */
    public function getAvailablePackages(): array
    {
        return $this->findAvailablePackages($this->projectDirectory . '/' . self::PACKAGE_DIR);
    }

    /**
     * Copied from Composer\Repository\ArtifactRepository
     * @see https://github.com/composer/composer/blob/main/src/Composer/Repository/ArtifactRepository.php
     *
     * @return Package[]
     */
    private function findAvailablePackages(string $path): array
    {
        $packages = [];

        $directory = new \RecursiveDirectoryIterator($path, \RecursiveDirectoryIterator::FOLLOW_SYMLINKS);
        $iterator = new \RecursiveIteratorIterator($directory);
        $regex = new \RegexIterator($iterator, '/^.+\.zip$/i');
        /** @var \SplFileInfo $file */
        foreach ($regex as $file) {
            if (!$file->isFile()) {
                continue;
            }

            $package = $this->getComposerJson($file->getPathname());
            if ($package === null) {
                continue;
            }

            $content = json_decode($package, true);
            if (\JSON_ERROR_NONE !== json_last_error() || !\is_array($content)) {
                throw new \RuntimeException('Failed to parse composer.json file in: ' . $file->getPathname());
            }

            $packages[] = new Package($file, PluginMetadata::createFromArray($content));
        }

        return $packages;
    }

    /**
     * Copied from Composer\Util\Zip
     * @see https://github.com/composer/composer/blob/main/src/Composer/Util/Zip.php
     */
    private function getComposerJson(string $pathToZip): ?string
    {
        if (!\extension_loaded('zip')) {
            throw new \RuntimeException('The Zip Util requires PHP\'s zip extension');
        }

        $zip = new \ZipArchive();
        if ($zip->open($pathToZip) !== true) {
            return null;
        }

        if (0 === $zip->numFiles) {
            $zip->close();

            return null;
        }

        $foundFileIndex = self::locateFile($zip, 'composer.json');

        $content = null;
        $configurationFileName = $zip->getNameIndex($foundFileIndex);
        if ($configurationFileName !== false) {
            $stream = $zip->getStream($configurationFileName);

            if (false !== $stream) {
                $content = stream_get_contents($stream);
                if ($content === false) {
                    $content = null;
                }
            }
        }

        $zip->close();

        return $content;
    }

    /**
     * Copied from Composer\Util\Zip
     * @see https://github.com/composer/composer/blob/main/src/Composer/Util/Zip.php
     */
    private static function locateFile(\ZipArchive $zip, string $filename): int
    {
        // return root composer.json if it is there and is a file
        if (false !== ($index = $zip->locateName($filename)) && $zip->getFromIndex($index) !== false) {
            return $index;
        }

        $topLevelPaths = [];
        for ($i = 0; $i < $zip->numFiles; $i++) {
            $name = $zip->getNameIndex($i);
            if ($name === false) {
                continue;
            }
            $dirname = \dirname($name);

            // ignore OSX specific resource fork folder
            if (strpos($name, '__MACOSX') !== false) {
                continue;
            }

            // handle archives with proper TOC
            if ($dirname === '.') {
                $topLevelPaths[$name] = true;
                if (\count($topLevelPaths) > 1) {
                    throw new \RuntimeException('Archive has more than one top level directories, and no composer.json was found on the top level, so it\'s an invalid archive. Top level paths found were: ' . implode(',', array_keys($topLevelPaths)));
                }
                continue;
            }

            // handle archives which do not have a TOC record for the directory itself
            if (false === strpos($dirname, '\\') && false === strpos($dirname, '/')) {
                $topLevelPaths[$dirname . '/'] = true;
                if (\count($topLevelPaths) > 1) {
                    throw new \RuntimeException('Archive has more than one top level directories, and no composer.json was found on the top level, so it\'s an invalid archive. Top level paths found were: ' . implode(',', array_keys($topLevelPaths)));
                }
            }
        }

        if ($topLevelPaths && false !== ($index = $zip->locateName(key($topLevelPaths) . $filename)) && $zip->getFromIndex($index) !== false) {
            return $index;
        }

        throw new \RuntimeException('No composer.json found either at the top level or within the topmost directory');
    }
}