| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329 | <?php/* * This file is part of the Symfony package. * * (c) Fabien Potencier <fabien@symfony.com> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */namespace Symfony\Component\Validator\Constraints;use Symfony\Component\HttpFoundation\File\File as FileObject;use Symfony\Component\HttpFoundation\File\UploadedFile;use Symfony\Component\Validator\Context\ExecutionContextInterface;use Symfony\Component\Validator\Constraint;use Symfony\Component\Validator\ConstraintValidator;use Symfony\Component\Validator\Exception\UnexpectedTypeException;/** * @author Bernhard Schussek <bschussek@gmail.com> */class FileValidator extends ConstraintValidator{    const KB_BYTES = 1000;    const MB_BYTES = 1000000;    const KIB_BYTES = 1024;    const MIB_BYTES = 1048576;    private static $suffices = array(        1 => 'bytes',        self::KB_BYTES => 'kB',        self::MB_BYTES => 'MB',        self::KIB_BYTES => 'KiB',        self::MIB_BYTES => 'MiB',    );    /**     * {@inheritdoc}     */    public function validate($value, Constraint $constraint)    {        if (!$constraint instanceof File) {            throw new UnexpectedTypeException($constraint, __NAMESPACE__.'\File');        }        if (null === $value || '' === $value) {            return;        }        if ($value instanceof UploadedFile && !$value->isValid()) {            switch ($value->getError()) {                case UPLOAD_ERR_INI_SIZE:                    $iniLimitSize = UploadedFile::getMaxFilesize();                    if ($constraint->maxSize && $constraint->maxSize < $iniLimitSize) {                        $limitInBytes = $constraint->maxSize;                        $binaryFormat = $constraint->binaryFormat;                    } else {                        $limitInBytes = $iniLimitSize;                        $binaryFormat = true;                    }                    list($sizeAsString, $limitAsString, $suffix) = $this->factorizeSizes(0, $limitInBytes, $binaryFormat);                    if ($this->context instanceof ExecutionContextInterface) {                        $this->context->buildViolation($constraint->uploadIniSizeErrorMessage)                            ->setParameter('{{ limit }}', $limitAsString)                            ->setParameter('{{ suffix }}', $suffix)                            ->setCode(UPLOAD_ERR_INI_SIZE)                            ->addViolation();                    } else {                        $this->buildViolation($constraint->uploadIniSizeErrorMessage)                            ->setParameter('{{ limit }}', $limitAsString)                            ->setParameter('{{ suffix }}', $suffix)                            ->setCode(UPLOAD_ERR_INI_SIZE)                            ->addViolation();                    }                    return;                case UPLOAD_ERR_FORM_SIZE:                    if ($this->context instanceof ExecutionContextInterface) {                        $this->context->buildViolation($constraint->uploadFormSizeErrorMessage)                            ->setCode(UPLOAD_ERR_FORM_SIZE)                            ->addViolation();                    } else {                        $this->buildViolation($constraint->uploadFormSizeErrorMessage)                            ->setCode(UPLOAD_ERR_FORM_SIZE)                            ->addViolation();                    }                    return;                case UPLOAD_ERR_PARTIAL:                    if ($this->context instanceof ExecutionContextInterface) {                        $this->context->buildViolation($constraint->uploadPartialErrorMessage)                            ->setCode(UPLOAD_ERR_PARTIAL)                            ->addViolation();                    } else {                        $this->buildViolation($constraint->uploadPartialErrorMessage)                            ->setCode(UPLOAD_ERR_PARTIAL)                            ->addViolation();                    }                    return;                case UPLOAD_ERR_NO_FILE:                    if ($this->context instanceof ExecutionContextInterface) {                        $this->context->buildViolation($constraint->uploadNoFileErrorMessage)                            ->setCode(UPLOAD_ERR_NO_FILE)                            ->addViolation();                    } else {                        $this->buildViolation($constraint->uploadNoFileErrorMessage)                            ->setCode(UPLOAD_ERR_NO_FILE)                            ->addViolation();                    }                    return;                case UPLOAD_ERR_NO_TMP_DIR:                    if ($this->context instanceof ExecutionContextInterface) {                        $this->context->buildViolation($constraint->uploadNoTmpDirErrorMessage)                            ->setCode(UPLOAD_ERR_NO_TMP_DIR)                            ->addViolation();                    } else {                        $this->buildViolation($constraint->uploadNoTmpDirErrorMessage)                            ->setCode(UPLOAD_ERR_NO_TMP_DIR)                            ->addViolation();                    }                    return;                case UPLOAD_ERR_CANT_WRITE:                    if ($this->context instanceof ExecutionContextInterface) {                        $this->context->buildViolation($constraint->uploadCantWriteErrorMessage)                            ->setCode(UPLOAD_ERR_CANT_WRITE)                            ->addViolation();                    } else {                        $this->buildViolation($constraint->uploadCantWriteErrorMessage)                            ->setCode(UPLOAD_ERR_CANT_WRITE)                            ->addViolation();                    }                    return;                case UPLOAD_ERR_EXTENSION:                    if ($this->context instanceof ExecutionContextInterface) {                        $this->context->buildViolation($constraint->uploadExtensionErrorMessage)                            ->setCode(UPLOAD_ERR_EXTENSION)                            ->addViolation();                    } else {                        $this->buildViolation($constraint->uploadExtensionErrorMessage)                            ->setCode(UPLOAD_ERR_EXTENSION)                            ->addViolation();                    }                    return;                default:                    if ($this->context instanceof ExecutionContextInterface) {                        $this->context->buildViolation($constraint->uploadErrorMessage)                            ->setCode($value->getError())                            ->addViolation();                    } else {                        $this->buildViolation($constraint->uploadErrorMessage)                            ->setCode($value->getError())                            ->addViolation();                    }                    return;            }        }        if (!is_scalar($value) && !$value instanceof FileObject && !(is_object($value) && method_exists($value, '__toString'))) {            throw new UnexpectedTypeException($value, 'string');        }        $path = $value instanceof FileObject ? $value->getPathname() : (string) $value;        if (!is_file($path)) {            if ($this->context instanceof ExecutionContextInterface) {                $this->context->buildViolation($constraint->notFoundMessage)                    ->setParameter('{{ file }}', $this->formatValue($path))                    ->setCode(File::NOT_FOUND_ERROR)                    ->addViolation();            } else {                $this->buildViolation($constraint->notFoundMessage)                    ->setParameter('{{ file }}', $this->formatValue($path))                    ->setCode(File::NOT_FOUND_ERROR)                    ->addViolation();            }            return;        }        if (!is_readable($path)) {            if ($this->context instanceof ExecutionContextInterface) {                $this->context->buildViolation($constraint->notReadableMessage)                    ->setParameter('{{ file }}', $this->formatValue($path))                    ->setCode(File::NOT_READABLE_ERROR)                    ->addViolation();            } else {                $this->buildViolation($constraint->notReadableMessage)                    ->setParameter('{{ file }}', $this->formatValue($path))                    ->setCode(File::NOT_READABLE_ERROR)                    ->addViolation();            }            return;        }        $sizeInBytes = filesize($path);        if (0 === $sizeInBytes) {            if ($this->context instanceof ExecutionContextInterface) {                $this->context->buildViolation($constraint->disallowEmptyMessage)                    ->setParameter('{{ file }}', $this->formatValue($path))                    ->setCode(File::EMPTY_ERROR)                    ->addViolation();            } else {                $this->buildViolation($constraint->disallowEmptyMessage)                    ->setParameter('{{ file }}', $this->formatValue($path))                    ->setCode(File::EMPTY_ERROR)                    ->addViolation();            }            return;        }        if ($constraint->maxSize) {            $limitInBytes = $constraint->maxSize;            if ($sizeInBytes > $limitInBytes) {                list($sizeAsString, $limitAsString, $suffix) = $this->factorizeSizes($sizeInBytes, $limitInBytes, $constraint->binaryFormat);                if ($this->context instanceof ExecutionContextInterface) {                    $this->context->buildViolation($constraint->maxSizeMessage)                        ->setParameter('{{ file }}', $this->formatValue($path))                        ->setParameter('{{ size }}', $sizeAsString)                        ->setParameter('{{ limit }}', $limitAsString)                        ->setParameter('{{ suffix }}', $suffix)                        ->setCode(File::TOO_LARGE_ERROR)                        ->addViolation();                } else {                    $this->buildViolation($constraint->maxSizeMessage)                        ->setParameter('{{ file }}', $this->formatValue($path))                        ->setParameter('{{ size }}', $sizeAsString)                        ->setParameter('{{ limit }}', $limitAsString)                        ->setParameter('{{ suffix }}', $suffix)                        ->setCode(File::TOO_LARGE_ERROR)                        ->addViolation();                }                return;            }        }        if ($constraint->mimeTypes) {            if (!$value instanceof FileObject) {                $value = new FileObject($value);            }            $mimeTypes = (array) $constraint->mimeTypes;            $mime = $value->getMimeType();            foreach ($mimeTypes as $mimeType) {                if ($mimeType === $mime) {                    return;                }                if ($discrete = strstr($mimeType, '/*', true)) {                    if (strstr($mime, '/', true) === $discrete) {                        return;                    }                }            }            if ($this->context instanceof ExecutionContextInterface) {                $this->context->buildViolation($constraint->mimeTypesMessage)                    ->setParameter('{{ file }}', $this->formatValue($path))                    ->setParameter('{{ type }}', $this->formatValue($mime))                    ->setParameter('{{ types }}', $this->formatValues($mimeTypes))                    ->setCode(File::INVALID_MIME_TYPE_ERROR)                    ->addViolation();            } else {                $this->buildViolation($constraint->mimeTypesMessage)                    ->setParameter('{{ file }}', $this->formatValue($path))                    ->setParameter('{{ type }}', $this->formatValue($mime))                    ->setParameter('{{ types }}', $this->formatValues($mimeTypes))                    ->setCode(File::INVALID_MIME_TYPE_ERROR)                    ->addViolation();            }        }    }    private static function moreDecimalsThan($double, $numberOfDecimals)    {        return strlen((string) $double) > strlen(round($double, $numberOfDecimals));    }    /**     * Convert the limit to the smallest possible number     * (i.e. try "MB", then "kB", then "bytes").     */    private function factorizeSizes($size, $limit, $binaryFormat)    {        if ($binaryFormat) {            $coef = self::MIB_BYTES;            $coefFactor = self::KIB_BYTES;        } else {            $coef = self::MB_BYTES;            $coefFactor = self::KB_BYTES;        }        $limitAsString = (string) ($limit / $coef);        // Restrict the limit to 2 decimals (without rounding! we        // need the precise value)        while (self::moreDecimalsThan($limitAsString, 2)) {            $coef /= $coefFactor;            $limitAsString = (string) ($limit / $coef);        }        // Convert size to the same measure, but round to 2 decimals        $sizeAsString = (string) round($size / $coef, 2);        // If the size and limit produce the same string output        // (due to rounding), reduce the coefficient        while ($sizeAsString === $limitAsString) {            $coef /= $coefFactor;            $limitAsString = (string) ($limit / $coef);            $sizeAsString = (string) round($size / $coef, 2);        }        return array($sizeAsString, $limitAsString, self::$suffices[$coef]);    }}
 |