. */ if (!defined('GNUSOCIAL')) { exit(1); } /** * Table Definition for file_thumbnail */ class File_thumbnail extends Managed_DataObject { public $__table = 'file_thumbnail'; // table name public $file_id; // int(4) primary_key not_null public $url; // text public $filename; // varchar(191) not 255 because utf8mb4 takes more space public $width; // int(4) primary_key public $height; // int(4) primary_key public $modified; // timestamp() not_null default_CURRENT_TIMESTAMP public static function schemaDef() { return array( 'fields' => array( 'file_id' => array('type' => 'int', 'not null' => true, 'description' => 'thumbnail for what URL/file'), 'url' => array('type' => 'text', 'description' => 'URL of thumbnail'), 'filename' => array('type' => 'varchar', 'length' => 191, 'description' => 'if stored locally, filename is put here'), 'width' => array('type' => 'int', 'description' => 'width of thumbnail'), 'height' => array('type' => 'int', 'description' => 'height of thumbnail'), 'modified' => array('type' => 'timestamp', 'not null' => true, 'description' => 'date this record was modified'), ), 'primary key' => array('file_id', 'width', 'height'), 'indexes' => array( 'file_thumbnail_file_id_idx' => array('file_id'), ), 'foreign keys' => array( 'file_thumbnail_file_id_fkey' => array('file', array('file_id' => 'id')), ) ); } /** * Save oEmbed-provided thumbnail data * * @param object $data * @param int $file_id */ public static function saveNew($data, $file_id) { if (!empty($data->thumbnail_url)) { // Non-photo types such as video will usually // show us a thumbnail, though it's not required. self::saveThumbnail($file_id, $data->thumbnail_url, $data->thumbnail_width, $data->thumbnail_height); } else if ($data->type == 'photo') { // The inline photo URL given should also fit within // our requested thumbnail size, per oEmbed spec. self::saveThumbnail($file_id, $data->url, $data->width, $data->height); } } /** * Fetch an entry by using a File's id */ static function byFile(File $file) { $file_thumbnail = self::getKV('file_id', $file->id); if (!$file_thumbnail instanceof File_thumbnail) { throw new ServerException(sprintf('No File_thumbnail entry for File id==%u', $file->id)); } return $file_thumbnail; } /** * Save a thumbnail record for the referenced file record. * * FIXME: Add error handling * * @param int $file_id * @param string $url * @param int $width * @param int $height */ static function saveThumbnail($file_id, $url, $width, $height, $filename=null) { $tn = new File_thumbnail; $tn->file_id = $file_id; $tn->url = $url; $tn->filename = $filename; $tn->width = intval($width); $tn->height = intval($height); $tn->insert(); return $tn; } static function path($filename) { // TODO: Store thumbnails in their own directory and don't use File::path here return File::path($filename); } public function getPath() { $filepath = self::path($this->filename); if (!file_exists($filepath)) { throw new FileNotFoundException($filepath); } return $filepath; } public function getUrl() { if (!empty($this->getFile()->filename)) { // A locally stored File, so let's generate a URL for our instance. $url = File::url($this->filename); if ($url != $this->url) { // For indexing purposes, in case we do a lookup on the 'url' field. // also we're fixing possible changes from http to https, or paths $this->updateUrl($url); } return $url; } // No local filename available, return the URL we have stored return $this->url; } public function updateUrl($url) { $file = File_thumbnail::getKV('url', $url); if ($file instanceof File_thumbnail) { throw new ServerException('URL already exists in DB'); } $sql = 'UPDATE %1$s SET url=%2$s WHERE url=%3$s;'; $result = $this->query(sprintf($sql, $this->__table, $this->_quote((string)$url), $this->_quote((string)$this->url))); if ($result === false) { common_log_db_error($this, 'UPDATE', __FILE__); throw new ServerException("Could not UPDATE {$this->__table}.url"); } return $result; } public function delete($useWhere=false) { if (!empty($this->filename) && file_exists(File_thumbnail::path($this->filename))) { $deleted = @unlink(self::path($this->filename)); if (!$deleted) { common_log(LOG_ERR, sprintf('Could not unlink existing file: "%s"', self::path($this->filename))); } } return parent::delete($useWhere); } public function getFile() { $file = new File(); $file->id = $this->file_id; if (!$file->find(true)) { throw new NoResultException($file); } return $file; } }