Merge remote-tracking branch 'upstream/nightly' into nightly

This commit is contained in:
www-data 2016-07-15 17:18:45 +02:00
commit 883ef2414f
10 changed files with 132 additions and 38 deletions

View File

@ -304,13 +304,12 @@ class File extends Managed_DataObject
$ext = common_supported_mime_to_ext($mimetype); $ext = common_supported_mime_to_ext($mimetype);
// we do, so use it! // we do, so use it!
return $ext; return $ext;
} catch (Exception $e) { // FIXME: Make this exception more specific to "unknown mime=>ext relation" } catch (UnknownMimeExtensionException $e) {
// We don't know the extension for this mimetype, but let's guess. // We don't know the extension for this mimetype, but let's guess.
// If we are very liberal with uploads ($config['attachments']['supported'] === true) // If we can't recognize the extension from the MIME, we try
// then we try to do some guessing based on the filename, if it was supplied. // to guess based on filename, if one was supplied.
if (!is_null($filename) && common_config('attachments', 'supported')===true if (!is_null($filename) && preg_match('/^.+\.([A-Za-z0-9]+)$/', $filename, $matches)) {
&& preg_match('/^.+\.([A-Za-z0-9]+)$/', $filename, $matches)) {
// we matched on a file extension, so let's see if it means something. // we matched on a file extension, so let's see if it means something.
$ext = mb_strtolower($matches[1]); $ext = mb_strtolower($matches[1]);
@ -330,6 +329,8 @@ class File extends Managed_DataObject
// the attachment extension based on its filename was not blacklisted so it's ok to use it // the attachment extension based on its filename was not blacklisted so it's ok to use it
return $ext; return $ext;
} }
} catch (Exception $e) {
common_log(LOG_INFO, 'Problem when figuring out extension for mimetype: '._ve($e));
} }
// If nothing else has given us a result, try to extract it from // If nothing else has given us a result, try to extract it from
@ -528,13 +529,23 @@ class File extends Managed_DataObject
return common_local_url('attachment', array('attachment'=>$this->getID())); return common_local_url('attachment', array('attachment'=>$this->getID()));
} }
public function getUrl($prefer_local=true) /**
* @param mixed $use_local true means require local, null means prefer local, false means use whatever is stored
*/
public function getUrl($use_local=null)
{ {
if ($prefer_local && !empty($this->filename)) { if ($use_local !== false) {
// A locally stored file, so let's generate a URL for our instance. if (is_string($this->filename) || !empty($this->filename)) {
return self::url($this->getFilename()); // A locally stored file, so let's generate a URL for our instance.
return self::url($this->getFilename());
}
if ($use_local) {
// if the file wasn't stored locally (has filename) and we require a local URL
throw new FileNotStoredLocallyException($this);
}
} }
// No local filename available, return the URL we have stored // No local filename available, return the URL we have stored
return $this->url; return $this->url;
} }

View File

@ -63,7 +63,7 @@ class AttachmentListItem extends Widget
} }
function title() { function title() {
return $this->attachment->getTitle(); return $this->attachment->getTitle() ?: _('Untitled attachment');
} }
function linkTitle() { function linkTitle() {
@ -141,7 +141,13 @@ class AttachmentListItem extends Widget
if ($thumb instanceof File_thumbnail) { if ($thumb instanceof File_thumbnail) {
$this->out->element('img', $thumb->getHtmlAttrs(['class'=>'u-photo', 'alt' => ''])); $this->out->element('img', $thumb->getHtmlAttrs(['class'=>'u-photo', 'alt' => '']));
} else { } else {
$this->out->element('img', array('class'=>'u-photo', 'src' => $this->attachment->getUrl(), 'alt' => $this->attachment->getTitle())); try {
// getUrl(true) because we don't want to hotlink, could be made configurable
$this->out->element('img', ['class'=>'u-photo', 'src'=>$this->attachment->getUrl(true), 'alt' => $this->attachment->getTitle()]);
} catch (FileNotStoredLocallyException $e) {
$url = $e->file->getUrl(false);
$this->out->element('a', ['href'=>$url, 'rel'=>'external'], $url);
}
} }
unset($thumb); // there's no need carrying this along after this unset($thumb); // there's no need carrying this along after this
break; break;

View File

@ -249,6 +249,7 @@ $default =
'application/zip' => 'zip', 'application/zip' => 'zip',
'application/x-go-sgf' => 'sgf', 'application/x-go-sgf' => 'sgf',
'application/xml' => 'xml', 'application/xml' => 'xml',
'application/gpx+xml' => 'gpx',
'image/png' => 'png', 'image/png' => 'png',
'image/jpeg' => 'jpg', 'image/jpeg' => 'jpg',
'image/gif' => 'gif', 'image/gif' => 'gif',
@ -273,7 +274,7 @@ $default =
'show_thumbs' => true, // show thumbnails in notice lists for uploaded images, and photos and videos linked remotely that provide oEmbed info 'show_thumbs' => true, // show thumbnails in notice lists for uploaded images, and photos and videos linked remotely that provide oEmbed info
'process_links' => true, // check linked resources for embeddable photos and videos; this will hit referenced external web sites when processing new messages. 'process_links' => true, // check linked resources for embeddable photos and videos; this will hit referenced external web sites when processing new messages.
'extblacklist' => [ 'extblacklist' => [
'php' => 'phps', 'php' => 'phps', // this turns .php into .phps
'exe' => false, // this would deny any uploads to keep the "exe" file extension 'exe' => false, // this would deny any uploads to keep the "exe" file extension
], ],
), ),

View File

@ -0,0 +1,15 @@
<?php
if (!defined('GNUSOCIAL')) { exit(1); }
class FileNotStoredLocallyException extends ServerException
{
public $file = null;
public function __construct(File $file)
{
$this->file = $file;
common_debug('Requested local URL for a file that is not stored locally with id=='._ve($this->file->getID()));
parent::__construct(_('Requested local URL for a file that is not stored locally.'));
}
}

View File

@ -355,6 +355,7 @@ class MediaFile
$unclearTypes = array('application/octet-stream', $unclearTypes = array('application/octet-stream',
'application/vnd.ms-office', 'application/vnd.ms-office',
'application/zip', 'application/zip',
'text/plain',
'text/html', // Ironically, Wikimedia Commons' SVG_logo.svg is identified as text/html 'text/html', // Ironically, Wikimedia Commons' SVG_logo.svg is identified as text/html
// TODO: for XML we could do better content-based sniffing too // TODO: for XML we could do better content-based sniffing too
'text/xml'); 'text/xml');
@ -364,10 +365,12 @@ class MediaFile
// If we didn't match, or it is an unclear match // If we didn't match, or it is an unclear match
if ($originalFilename && (!$mimetype || in_array($mimetype, $unclearTypes))) { if ($originalFilename && (!$mimetype || in_array($mimetype, $unclearTypes))) {
try { try {
$type = common_supported_ext_to_mime($originalFilename); $type = common_supported_filename_to_mime($originalFilename);
return $type; return $type;
} catch (UnknownExtensionMimeException $e) {
// FIXME: I think we should keep the file extension here (supported should be === true here)
} catch (Exception $e) { } catch (Exception $e) {
// Extension not found, so $mimetype is our best guess // Extension parsed but no connected mimetype, so $mimetype is our best guess
} }
} }

View File

@ -0,0 +1,26 @@
<?php
if (!defined('GNUSOCIAL')) { exit(1); }
/**
* Class for unknown extension MIME type exception
*
* Thrown when we don't know the MIME type for a given file extension.
*
* @category Exception
* @package GNUsocial
* @author Mikael Nordfeldth <mmn@hethane.se>
* @license https://www.gnu.org/licenses/agpl-3.0.html
* @link https://gnu.io/social
*/
class UnknownExtensionMimeException extends ServerException
{
public function __construct($ext)
{
// TRANS: We accept the file type (we probably just accept all files)
// TRANS: but don't know the file extension for it. %1$s is the extension.
$msg = sprintf(_('Unknown MIME type for file extension: %1$s'), _ve($ext));
parent::__construct($msg);
}
}

View File

@ -0,0 +1,27 @@
<?php
if (!defined('GNUSOCIAL')) { exit(1); }
/**
* Class for unknown MIME extension exception
*
* Thrown when we don't know the file extension for a given MIME type.
* This generally means that all files are accepted since if we have
* a list of known MIMEs then they have extensions coupled to them.
*
* @category Exception
* @package GNUsocial
* @author Mikael Nordfeldth <mmn@hethane.se>
* @license https://www.gnu.org/licenses/agpl-3.0.html
* @link https://gnu.io/social
*/
class UnknownMimeExtensionException extends ServerException
{
public function __construct($mimetype)
{
// TRANS: We accept the file type (we probably just accept all files)
// TRANS: but don't know the file extension for it.
$msg = sprintf(_('Supported mimetype but unknown extension relation: %1$s'), _ve($mimetype));
parent::__construct($msg);
}
}

View File

@ -1991,16 +1991,23 @@ function common_accept_to_prefs($accept, $def = '*/*')
} }
// Match by our supported file extensions // Match by our supported file extensions
function common_supported_ext_to_mime($fileext) function common_supported_filename_to_mime($filename)
{ {
// Accept a filename and take out the extension // Accept a filename and take out the extension
if (strpos($fileext, '.') !== false) { if (strpos($filename, '.') === false) {
$fileext = substr(strrchr($fileext, '.'), 1); throw new ServerException(sprintf('No extension on filename: %1$s', _ve($filename)));
} }
$fileext = substr(strrchr($filename, '.'), 1);
return common_supported_ext_to_mime($fileext);
}
function common_supported_ext_to_mime($fileext)
{
$supported = common_config('attachments', 'supported'); $supported = common_config('attachments', 'supported');
if ($supported === true) { if ($supported === true) {
throw new ServerException('Supported extension but unknown mimetype relation.'); // FIXME: Should we just accept the extension straight off when supported === true?
throw new UnknownExtensionMimeException($fileext);
} }
foreach($supported as $type => $ext) { foreach($supported as $type => $ext) {
if ($ext === $fileext) { if ($ext === $fileext) {
@ -2015,16 +2022,15 @@ function common_supported_ext_to_mime($fileext)
function common_supported_mime_to_ext($mimetype) function common_supported_mime_to_ext($mimetype)
{ {
$supported = common_config('attachments', 'supported'); $supported = common_config('attachments', 'supported');
if ($supported === true) { if (is_array($supported)) {
throw new ServerException('Supported mimetype but unknown extension relation.'); foreach($supported as $type => $ext) {
} if ($mimetype === $type) {
foreach($supported as $type => $ext) { return $ext;
if ($mimetype === $type) { }
return $ext;
} }
} }
throw new ServerException('Unsupported MIME type'); throw new UnknownMimeExtensionException($mimetype);
} }
// The MIME "media" is the part before the slash (video in video/webm) // The MIME "media" is the part before the slash (video in video/webm)

View File

@ -65,53 +65,52 @@ class ShowfavoritesAction extends ShowstreamAction
return array(new Feed(Feed::JSON, return array(new Feed(Feed::JSON,
common_local_url('ApiTimelineFavorites', common_local_url('ApiTimelineFavorites',
array( array(
'id' => $this->user->nickname, 'id' => $this->getTarget()->getNickname(),
'format' => 'as')), 'format' => 'as')),
// TRANS: Feed link text. %s is a username. // TRANS: Feed link text. %s is a username.
sprintf(_('Feed for favorites of %s (Activity Streams JSON)'), sprintf(_('Feed for favorites of %s (Activity Streams JSON)'),
$this->user->nickname)), $this->getTarget()->getNickname())),
new Feed(Feed::RSS1, new Feed(Feed::RSS1,
common_local_url('favoritesrss', common_local_url('favoritesrss',
array('nickname' => $this->user->nickname)), array('nickname' => $this->getTarget()->getNickname())),
// TRANS: Feed link text. %s is a username. // TRANS: Feed link text. %s is a username.
sprintf(_('Feed for favorites of %s (RSS 1.0)'), sprintf(_('Feed for favorites of %s (RSS 1.0)'),
$this->user->nickname)), $this->getTarget()->getNickname())),
new Feed(Feed::RSS2, new Feed(Feed::RSS2,
common_local_url('ApiTimelineFavorites', common_local_url('ApiTimelineFavorites',
array( array(
'id' => $this->user->nickname, 'id' => $this->getTarget()->getNickname(),
'format' => 'rss')), 'format' => 'rss')),
// TRANS: Feed link text. %s is a username. // TRANS: Feed link text. %s is a username.
sprintf(_('Feed for favorites of %s (RSS 2.0)'), sprintf(_('Feed for favorites of %s (RSS 2.0)'),
$this->user->nickname)), $this->getTarget()->getNickname())),
new Feed(Feed::ATOM, new Feed(Feed::ATOM,
common_local_url('ApiTimelineFavorites', common_local_url('ApiTimelineFavorites',
array( array(
'id' => $this->user->nickname, 'id' => $this->getTarget()->getNickname(),
'format' => 'atom')), 'format' => 'atom')),
// TRANS: Feed link text. %s is a username. // TRANS: Feed link text. %s is a username.
sprintf(_('Feed for favorites of %s (Atom)'), sprintf(_('Feed for favorites of %s (Atom)'),
$this->user->nickname))); $this->getTarget()->getNickname())));
} }
function showEmptyListMessage() function showEmptyListMessage()
{ {
if (common_logged_in()) { if (common_logged_in()) {
$current_user = common_current_user(); if ($this->getTarget()->sameAs($this->getScoped())) {
if ($this->user->id === $current_user->id) {
// TRANS: Text displayed instead of favourite notices for the current logged in user that has no favourites. // TRANS: Text displayed instead of favourite notices for the current logged in user that has no favourites.
$message = _('You haven\'t chosen any favorite notices yet. Click the fave button on notices you like to bookmark them for later or shed a spotlight on them.'); $message = _('You haven\'t chosen any favorite notices yet. Click the fave button on notices you like to bookmark them for later or shed a spotlight on them.');
} else { } else {
// TRANS: Text displayed instead of favourite notices for a user that has no favourites while logged in. // TRANS: Text displayed instead of favourite notices for a user that has no favourites while logged in.
// TRANS: %s is a username. // TRANS: %s is a username.
$message = sprintf(_('%s hasn\'t added any favorite notices yet. Post something interesting they would add to their favorites :)'), $this->user->nickname); $message = sprintf(_('%s hasn\'t added any favorite notices yet. Post something interesting they would add to their favorites :)'), $this->getTarget()->getNickname());
} }
} }
else { else {
// TRANS: Text displayed instead of favourite notices for a user that has no favourites while not logged in. // TRANS: Text displayed instead of favourite notices for a user that has no favourites while not logged in.
// TRANS: %s is a username, %%%%action.register%%%% is a link to the user registration page. // TRANS: %s is a username, %%%%action.register%%%% is a link to the user registration page.
// TRANS: (link text)[link] is a Mark Down link. // TRANS: (link text)[link] is a Mark Down link.
$message = sprintf(_('%s hasn\'t added any favorite notices yet. Why not [register an account](%%%%action.register%%%%) and then post something interesting they would add to their favorites :)'), $this->user->nickname); $message = sprintf(_('%s hasn\'t added any favorite notices yet. Why not [register an account](%%%%action.register%%%%) and then post something interesting they would add to their favorites :)'), $this->getTarget()->getNickname());
} }
$this->elementStart('div', 'guide'); $this->elementStart('div', 'guide');

View File

@ -55,7 +55,7 @@ class StoreRemoteMediaPlugin extends Plugin
switch (common_get_mime_media($file->mimetype)) { switch (common_get_mime_media($file->mimetype)) {
case 'image': case 'image':
// Just to set something for now at least... // Just to set something for now at least...
$file->title = $file->mimetype; //$file->title = $file->mimetype;
break; break;
} }