Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add PropertyString, cleanHtml helper and escapeOrCleanHtml helper. #4004

Draft
wants to merge 14 commits into
base: dev
Choose a base branch
from
Draft
1 change: 1 addition & 0 deletions composer.json
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@
"composer/package-versions-deprecated": "1.11.99.5",
"composer/semver": "3.4.3",
"endroid/qr-code": "5.1.0",
"ezyang/htmlpurifier": "4.17.0",
"guzzlehttp/guzzle": "7.9.2",
"jaybizzle/crawler-detect": "^1.2",
"laminas/laminas-cache": "3.12.2",
Expand Down
63 changes: 62 additions & 1 deletion composer.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 4 additions & 0 deletions module/VuFind/src/VuFind/Content/Summaries/Demo.php
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,8 @@

namespace VuFind\Content\Summaries;

use VuFind\String\PropertyString;

/**
* Demo (fake data) summaries content loader.
*
Expand Down Expand Up @@ -56,6 +58,8 @@ public function loadByIsbn($key, \VuFindCode\ISBN $isbnObj)
return [
'Demo summary key: ' . $key,
'Demo summary ISBN: ' . $isbnObj->get13(),
(new PropertyString('Demo non-HTML summary'))
->setHtml('<strong>Demo HTML Summary:</strong><ul><li>Item 1</li><li>Item 2</li></ul>'),
maccabeelevine marked this conversation as resolved.
Show resolved Hide resolved
];
}
}
25 changes: 22 additions & 3 deletions module/VuFind/src/VuFind/RecordDriver/DefaultRecord.php
Original file line number Diff line number Diff line change
Expand Up @@ -1271,9 +1271,9 @@ public function getSystemDetails()
public function getSummary()
{
// We need to return an array, so if we have a description, turn it into an
// array (it should be a flat string according to the default schema, but we
// might as well support the array case just to be on the safe side:
return (array)($this->fields['description'] ?? []);
// array (it is a flat string in the default Solr schema, but we also
// support multivalued fields for other backends):
return $this->getFieldAsArray('description');
}

/**
Expand Down Expand Up @@ -1816,4 +1816,23 @@ public function getCoordinateLabels()
{
return (array)($this->fields['long_lat_label'] ?? []);
}

/**
* Get a field as an array
*
* @param string $field Field
*
* @return array
*/
protected function getFieldAsArray(string $field): array
{
// Make sure to return only non-empty values:
$value = $this->fields['description'] ?? '';
if ('' === $value) {
return [];
}
// Avoid casting since description can be a PropertyString too (and casting would return an array of object
// properties):
return is_array($value) ? $value : [$value];
demiankatz marked this conversation as resolved.
Show resolved Hide resolved
}
}
200 changes: 200 additions & 0 deletions module/VuFind/src/VuFind/String/PropertyString.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,200 @@
<?php

/**
* Class for a string with additional properties.
*
* PHP version 8
*
* Copyright (C) The National Library of Finland 2024.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2,
* as published by the Free Software Foundation.
*
* 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, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*
* @category VuFind
* @package String
* @author Ere Maijala <ere.maijala@helsinki.fi>
* @license http://opensource.org/licenses/gpl-2.0.php GNU General Public License
* @link https://vufind.org Main Site
*/

namespace VuFind\String;

use function array_key_exists;

/**
* Class for a string with additional properties.
*
* @category VuFind
* @package String
* @author Ere Maijala <ere.maijala@helsinki.fi>
* @license http://opensource.org/licenses/gpl-2.0.php GNU General Public License
* @link https://vufind.org Main Site
*/
class PropertyString implements PropertyStringInterface
{
/**
* Constructor
*
* @param string $string String value
* @param array $properties Associative array of any additional properties. Use a custom prefix for locally
* defined properties. Double underscore is a reserved prefix, and currently the following keys are defined:
* __ids Identifiers (e.g. subject URIs)
* __html HTML presentation
*/
public function __construct(protected string $string, protected array $properties = [])
{
}

/**
* Create a PropertyString from an HTML string
*
* @param string $html HTML
* @param array $properties Any additional properties (see __construct)
*
* @return PropertyString
*/
public static function fromHtml(string $html, array $properties = []): PropertyString
{
return (new PropertyString(strip_tags($html), $properties))->setHtml($html);
}

/**
* Set string value
*
* @param string $str String value
*
* @return static
*/
public function setString(string $str): static
{
$this->string = $str;
return $this;
}

/**
* Get string value
*
* @return string
*/
public function getString(): string
{
return $this->string;
}

/**
* Set HTML string
*
* @param string $html HTML
*
* @return static
*/
public function setHtml(string $html): static
{
$this['__html'] = $html;
return $this;
}

/**
* Get HTML string
*
* Note: This could contain anything and must be sanitized for display
*
* @return ?string
*/
public function getHtml(): ?string
{
return $this['__html'];
}

/**
* Set identifiers
*
* @param array $ids Identifiers
*
* @return static
*/
public function setIds(array $ids): static
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think this is meant for linked data identifiers (i.e. URIs) for the string, but not sure. Is it practical to include some example use case with the demo driver, or should that wait for future PRs?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Which Demo driver did you have in mind? I don't think linked data IDs make a lot of sense in the context of something like book summaries, so I'm not sure how we would demonstrate them there. A better use case would be to add support in the appropriate MARC-based record driver traits, but that feels like a big enough change that it would be better as a follow-up than as part of this PR. I'm open to other ideas, though, if I'm misunderstanding your proposal!

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No I agree, better to wait for another PR.

{
$this['__ids'] = $ids;
return $this;
}

/**
* Get identifiers
*
* @return ?array
*/
public function getIds(): ?array
{
return $this['__ids'];
}

/**
* Check if offset exists
*
* @param mixed $offset Offset
*
* @return bool
*/
public function offsetExists(mixed $offset): bool
{
return array_key_exists($offset, $this->properties);
}

/**
* Return value of offset
*
* @param mixed $offset Offset
*
* @return mixed
*/
public function offsetGet(mixed $offset): mixed
{
return $this->properties[$offset] ?? null;
}

/**
* Set value of offset
*
* @param mixed $offset Offset
* @param mixed $value Value
*
* @return void
*/
public function offsetSet(mixed $offset, mixed $value): void
{
$this->properties[$offset] = $value;
}

/**
* Unset value of offset
*
* @param mixed $offset Offset
*
* @return void
*/
public function offsetUnset(mixed $offset): void
{
unset($this->properties[$offset]);
}

/**
* Return string value
*
* @return string
*/
public function __toString(): string
{
return $this->string;
}
}
Loading
Loading