_a['table'] = 'idf_uploads'; $this->_a['model'] = __CLASS__; $this->_a['cols'] = array( // It is mandatory to have an "id" column. 'id' => array( 'type' => 'Pluf_DB_Field_Sequence', 'blank' => true, ), 'project' => array( 'type' => 'Pluf_DB_Field_Foreignkey', 'model' => 'IDF_Project', 'blank' => false, 'verbose' => __('project'), 'relate_name' => 'downloads', ), 'summary' => array( 'type' => 'Pluf_DB_Field_Varchar', 'blank' => false, 'size' => 250, 'verbose' => __('summary'), ), 'ext_file' => array( 'type' => 'Pluf_DB_Field_Varchar', 'blank' => false, 'size' => 250, 'verbose' => __('External File URL'), ), 'changelog' => array( 'type' => 'Pluf_DB_Field_Text', 'blank' => true, 'verbose' => __('changes'), ), 'file' => array( 'type' => 'Pluf_DB_Field_File', 'blank' => false, 'default' => 0, 'verbose' => __('file'), 'help_text' => __('The path is relative to the upload path.'), ), 'filesize' => array( 'type' => 'Pluf_DB_Field_Integer', 'blank' => false, 'default' => 0, 'verbose' => __('file size in bytes'), ), 'md5' => array( 'type' => 'Pluf_DB_Field_Text', 'blank' => true, 'verbose' => __('MD5'), ), 'submitter' => array( 'type' => 'Pluf_DB_Field_Foreignkey', 'model' => 'Pluf_User', 'blank' => false, 'verbose' => __('submitter'), 'relate_name' => 'submitted_downloads', ), 'tags' => array( 'type' => 'Pluf_DB_Field_Manytomany', 'blank' => true, 'model' => 'IDF_Tag', 'verbose' => __('labels'), ), 'downloads' => array( 'type' => 'Pluf_DB_Field_Integer', 'blank' => false, 'default' => 0, 'verbose' => __('number of downloads'), ), 'creation_dtime' => array( 'type' => 'Pluf_DB_Field_Datetime', 'blank' => true, 'verbose' => __('creation date'), ), 'modif_dtime' => array( 'type' => 'Pluf_DB_Field_Datetime', 'blank' => true, 'verbose' => __('modification date'), ), ); $this->_a['idx'] = array( 'modif_dtime_idx' => array( 'col' => 'modif_dtime', 'type' => 'normal', ), ); $table = $this->_con->pfx.'idf_tag_idf_upload_assoc'; $this->_a['views'] = array( 'join_tags' => array( 'join' => 'LEFT JOIN '.$table .' ON idf_upload_id=id', ), ); } function __toString() { return $this->file; } /** * Get the file size of any remote resource (using get_headers()), * either in bytes or - default - as human-readable formatted string. * * @author Stephan Schmitz * @license MIT * @url * * @param string $url Takes the remote object's URL. * @param boolean $formatSize Whether to return size in bytes or formatted. * @param boolean $useHead Whether to use HEAD requests. If false, uses GET. * @return string Returns human-readable formatted size * or size in bytes (default: formatted). * * * //example * echo getRemoteFilesize('https://github.com/eyecatchup/SEOstats/archive/master.zip'); * */ private function getRemoteFilesize($url, $formatSize = true, $useHead = true) { if (false !== $useHead) { stream_context_set_default(array('http' => array('method' => 'HEAD'))); } $head = array_change_key_case(get_headers($url, 1)); // content-length of download (in bytes), read from Content-Length: field $clen = isset($head['content-length']) ? $head['content-length'] : 0; // cannot retrieve file size, return "-1" if (!$clen) { return -1; } if (!$formatSize) { return $clen; // return size in bytes } $size = $clen; switch ($clen) { case $clen < 1024: $size = $clen .' B'; break; case $clen < 1048576: $size = round($clen / 1024, 2) .' KiB'; break; case $clen < 1073741824: $size = round($clen / 1048576, 2) . ' MiB'; break; case $clen < 1099511627776: $size = round($clen / 1073741824, 2) . ' GiB'; break; } return $size; // return formatted size } function _toIndex() { return ''; } function preSave($create=false) { if ($this->id == '') { $this->creation_dtime = gmdate('Y-m-d H:i:s'); $this->modif_dtime = gmdate('Y-m-d H:i:s'); if ($this->ext_file == "") { $this->md5 = md5_file ($this->getFullPath()); } else { $this->filesize = $this->getRemoteFilesize($this->ext_file, false); } } } function postSave($create=false) { if ($create) { IDF_Timeline::insert($this, $this->get_project(), $this->get_submitter(), $this->creation_dtime); } } function getAbsoluteUrl($project) { return Pluf::f('url_upload').'/'.$project->shortname.'/files/'.$this->file; } function getFullPath() { return(Pluf::f('upload_path').'/'.$this->get_project()->shortname.'/files/'.$this->file); } /** * We drop the information from the timeline. */ function preDelete() { IDF_Timeline::remove($this); @unlink(Pluf::f('upload_path').'/'.$this->project->shortname.'/files/'.$this->file); } /** * Returns the timeline fragment for the file. * * * @param Pluf_HTTP_Request * @return Pluf_Template_SafeString */ public function timelineFragment($request) { $url = Pluf_HTTP_URL_urlForView('IDF_Views_Download::view', array($request->project->shortname, $this->id)); $out = ''. Pluf_esc(Pluf_Template_dateAgo($this->creation_dtime, 'without')). ''; $stag = new IDF_Template_ShowUser(); $user = $stag->start($this->get_submitter(), $request, '', false); $out .= sprintf(__('Download %2$d, %3$s'), $url, $this->id, Pluf_esc($this->summary)).''; $out .= ''; $out .= "\n".'
'.sprintf(__('Addition of download %2$d, by %3$s'), $url, $this->id, $user).'
'; return Pluf_Template::markSafe($out); } public function feedFragment($request) { $url = Pluf::f('url_base') .Pluf_HTTP_URL_urlForView('IDF_Views_Download::view', array($request->project->shortname, $this->id)); $title = sprintf(__('%1$s: Download %2$d added - %3$s'), $request->project->name, $this->id, $this->summary); $date = Pluf_Date::gmDateToGmString($this->creation_dtime); $context = new Pluf_Template_Context_Request( $request, array('url' => $url, 'title' => $title, 'file' => $this, 'date' => $date) ); $tmpl = new Pluf_Template('idf/downloads/feedfragment.xml'); return $tmpl->render($context); } /** * Notification of change of the object. * * @param IDF_Conf Current configuration * @param bool Creation (true) */ public function notify($conf, $create=true) { $project = $this->get_project(); $url = str_replace(array('%p', '%d'), array($project->shortname, $this->id), $conf->getVal('upload_webhook_url', '')); $tags = array(); foreach ($this->get_tags_list() as $tag) { $tags[] = $tag->class.':'.$tag->name; } $submitter = $this->get_submitter(); $payload = array( 'to_send' => array( 'project' => $project->shortname, 'id' => $this->id, 'summary' => $this->summary, 'changelog' => $this->changelog, 'filename' => $this->file, 'filesize' => $this->filesize, 'md5sum' => $this->md5, 'submitter_login' => $submitter->login, 'submitter_email' => $submitter->email, 'tags' => $tags, ), 'project_id' => $project->id, 'authkey' => $project->getWebHookKey(), 'url' => $url, ); if ($create === true) { $payload['method'] = 'PUT'; $payload['to_send']['creation_date'] = $this->creation_dtime; } else { $payload['method'] = 'POST'; $payload['to_send']['update_date'] = $this->modif_dtime; } $item = new IDF_Queue(); $item->type = 'upload'; $item->payload = $payload; $item->create(); $current_locale = Pluf_Translation::getLocale(); $from_email = Pluf::f('from_email'); $messageId = '<'.md5('upload'.$this->id.md5(Pluf::f('secret_key'))).'@'.Pluf::f('mail_host', 'localhost').'>'; $recipients = $project->getNotificationRecipientsForTab('downloads'); foreach ($recipients as $address => $language) { if ($this->get_submitter()->email === $address) { continue; } Pluf_Translation::loadSetLocale($language); $context = new Pluf_Template_Context(array( 'file' => $this, 'urlfile' => $this->getAbsoluteUrl($project), 'project' => $project, 'tags' => $this->get_tags_list(), )); $tplfile = 'idf/downloads/download-created-email.txt'; $subject = __('New download - %1$s (%2$s)'); $headers = array('Message-ID' => $messageId); if (!$create) { $tplfile = 'idf/downloads/download-updated-email.txt'; $subject = __('Updated download - %1$s (%2$s)'); $headers = array('References' => $messageId); } $tmpl = new Pluf_Template($tplfile); $text_email = $tmpl->render($context); $email = new Pluf_Mail($from_email, $address, sprintf($subject, $this->summary, $project->shortname)); $email->addTextMessage($text_email); $email->addHeaders($headers); $email->sendMail(); } Pluf_Translation::loadSetLocale($current_locale); } }