From f0620e7118a76a1faea0ca15ac554818f8bb1bcf Mon Sep 17 00:00:00 2001 From: Guo Xiang Tan Date: Wed, 29 May 2019 09:00:25 +0800 Subject: [PATCH] FEATURE: Support `[description|attachment](upload://)` in MD take 2. Previous attempt was missing `post_uploads` records. --- .../components/composer-editor.js.es6 | 2 +- .../discourse/components/cook-text.js.es6 | 2 +- .../discourse/lib/utilities.js.es6 | 12 +- app/assets/javascripts/markdown-it-bundle.js | 2 +- app/assets/javascripts/pretty-text-bundle.js | 2 +- .../engines/discourse-markdown-it.js.es6 | 22 ++++ .../discourse-markdown/image-protocol.js.es6 | 62 ---------- .../discourse-markdown/upload-protocol.js.es6 | 81 +++++++++++++ .../pretty-text/image-short-url.js.es6 | 68 ----------- .../pretty-text/pretty-text.js.es6 | 4 +- .../pretty-text/upload-short-url.js.es6 | 111 ++++++++++++++++++ app/controllers/uploads_controller.rb | 62 +++++++--- app/models/post.rb | 6 +- app/models/upload.rb | 51 ++++++-- config/routes.rb | 1 + lib/file_store/s3_store.rb | 9 +- lib/pretty_text.rb | 2 +- lib/pretty_text/helpers.rb | 13 +- lib/pretty_text/shims.js | 4 +- spec/components/file_store/s3_store_spec.rb | 19 +-- spec/components/pretty_text_spec.rb | 22 +++- spec/models/post_spec.rb | 35 ++---- spec/models/upload_spec.rb | 10 ++ spec/requests/uploads_controller_spec.rb | 105 +++++++++++++---- .../composer-attachment-test.js.es6 | 39 ++++++ .../lib/image-short-url-test.js.es6 | 56 --------- .../lib/upload-short-url-test.js.es6 | 81 +++++++++++++ test/javascripts/lib/utilities-test.js.es6 | 30 +++-- 28 files changed, 605 insertions(+), 308 deletions(-) delete mode 100644 app/assets/javascripts/pretty-text/engines/discourse-markdown/image-protocol.js.es6 create mode 100644 app/assets/javascripts/pretty-text/engines/discourse-markdown/upload-protocol.js.es6 delete mode 100644 app/assets/javascripts/pretty-text/image-short-url.js.es6 create mode 100644 app/assets/javascripts/pretty-text/upload-short-url.js.es6 create mode 100644 test/javascripts/acceptance/composer-attachment-test.js.es6 delete mode 100644 test/javascripts/lib/image-short-url-test.js.es6 create mode 100644 test/javascripts/lib/upload-short-url-test.js.es6 diff --git a/app/assets/javascripts/discourse/components/composer-editor.js.es6 b/app/assets/javascripts/discourse/components/composer-editor.js.es6 index b59728b5054..3a39a14fd24 100644 --- a/app/assets/javascripts/discourse/components/composer-editor.js.es6 +++ b/app/assets/javascripts/discourse/components/composer-editor.js.es6 @@ -36,7 +36,7 @@ import { import { cacheShortUploadUrl, resolveAllShortUrls -} from "pretty-text/image-short-url"; +} from "pretty-text/upload-short-url"; import { INLINE_ONEBOX_LOADING_CSS_CLASS, diff --git a/app/assets/javascripts/discourse/components/cook-text.js.es6 b/app/assets/javascripts/discourse/components/cook-text.js.es6 index 9fefced7492..3e345df0540 100644 --- a/app/assets/javascripts/discourse/components/cook-text.js.es6 +++ b/app/assets/javascripts/discourse/components/cook-text.js.es6 @@ -13,7 +13,7 @@ const CookText = Ember.Component.extend({ // pretty text may only be loaded now Ember.run.next(() => window - .requireModule("pretty-text/image-short-url") + .requireModule("pretty-text/upload-short-url") .resolveAllShortUrls(ajax) ); }); diff --git a/app/assets/javascripts/discourse/lib/utilities.js.es6 b/app/assets/javascripts/discourse/lib/utilities.js.es6 index c022bfd5c98..7c1e9a38c91 100644 --- a/app/assets/javascripts/discourse/lib/utilities.js.es6 +++ b/app/assets/javascripts/discourse/lib/utilities.js.es6 @@ -444,15 +444,9 @@ export function getUploadMarkdown(upload) { ) { return uploadLocation(upload.url); } else { - return ( - '' + - upload.original_filename + - " (" + - I18n.toHumanSize(upload.filesize) + - ")\n" - ); + return `[${upload.original_filename} (${I18n.toHumanSize( + upload.filesize + )})|attachment](${upload.short_url})`; } } diff --git a/app/assets/javascripts/markdown-it-bundle.js b/app/assets/javascripts/markdown-it-bundle.js index 6f88c5abbe2..2d0ec14ee52 100644 --- a/app/assets/javascripts/markdown-it-bundle.js +++ b/app/assets/javascripts/markdown-it-bundle.js @@ -14,6 +14,6 @@ //= require ./pretty-text/engines/discourse-markdown/newline //= require ./pretty-text/engines/discourse-markdown/html-img //= require ./pretty-text/engines/discourse-markdown/text-post-process -//= require ./pretty-text/engines/discourse-markdown/image-protocol +//= require ./pretty-text/engines/discourse-markdown/upload-protocol //= require ./pretty-text/engines/discourse-markdown/inject-line-number //= require ./pretty-text/engines/discourse-markdown/d-wrap diff --git a/app/assets/javascripts/pretty-text-bundle.js b/app/assets/javascripts/pretty-text-bundle.js index ed5331b6435..23dc0122c3a 100644 --- a/app/assets/javascripts/pretty-text-bundle.js +++ b/app/assets/javascripts/pretty-text-bundle.js @@ -11,4 +11,4 @@ //= require ./pretty-text/sanitizer //= require ./pretty-text/oneboxer //= require ./pretty-text/inline-oneboxer -//= require ./pretty-text/image-short-url +//= require ./pretty-text/upload-short-url diff --git a/app/assets/javascripts/pretty-text/engines/discourse-markdown-it.js.es6 b/app/assets/javascripts/pretty-text/engines/discourse-markdown-it.js.es6 index 4e503d09035..3c91bf03452 100644 --- a/app/assets/javascripts/pretty-text/engines/discourse-markdown-it.js.es6 +++ b/app/assets/javascripts/pretty-text/engines/discourse-markdown-it.js.es6 @@ -1,6 +1,7 @@ import { default as WhiteLister } from "pretty-text/white-lister"; import { sanitize } from "pretty-text/sanitizer"; import guid from "pretty-text/guid"; +import { ATTACHMENT_CSS_CLASS } from "pretty-text/upload-short-url"; function deprecate(feature, name) { return function() { @@ -187,6 +188,26 @@ function setupImageDimensions(md) { md.renderer.rules.image = renderImage; } +function renderAttachment(tokens, idx, options, env, slf) { + const linkOpenToken = tokens[idx]; + const linkTextToken = tokens[idx + 1]; + const split = linkTextToken.content.split("|"); + const isValid = !linkOpenToken.attrs[ + linkOpenToken.attrIndex("data-orig-href") + ]; + + if (isValid && split.length === 2 && split[1] === ATTACHMENT_CSS_CLASS) { + linkOpenToken.attrs.unshift(["class", split[1]]); + linkTextToken.content = split[0]; + } + + return slf.renderToken(tokens, idx, options); +} + +function setupAttachments(md) { + md.renderer.rules.link_open = renderAttachment; +} + let Helpers; export function setup(opts, siteSettings, state) { @@ -276,6 +297,7 @@ export function setup(opts, siteSettings, state) { setupUrlDecoding(opts.engine); setupHoister(opts.engine); setupImageDimensions(opts.engine); + setupAttachments(opts.engine); setupBlockBBCode(opts.engine); setupInlineBBCode(opts.engine); setupTextPostProcessRuler(opts.engine); diff --git a/app/assets/javascripts/pretty-text/engines/discourse-markdown/image-protocol.js.es6 b/app/assets/javascripts/pretty-text/engines/discourse-markdown/image-protocol.js.es6 deleted file mode 100644 index bdcfba414e9..00000000000 --- a/app/assets/javascripts/pretty-text/engines/discourse-markdown/image-protocol.js.es6 +++ /dev/null @@ -1,62 +0,0 @@ -// add image to array if src has an upload -function addImage(images, token) { - if (token.attrs) { - for (let i = 0; i < token.attrs.length; i++) { - if (token.attrs[i][1].indexOf("upload://") === 0) { - images.push([token, i]); - break; - } - } - } -} - -function rule(state) { - let images = []; - - for (let i = 0; i < state.tokens.length; i++) { - let blockToken = state.tokens[i]; - - if (blockToken.tag === "img") { - addImage(images, blockToken); - } - - if (!blockToken.children) { - continue; - } - - for (let j = 0; j < blockToken.children.length; j++) { - let token = blockToken.children[j]; - if (token.tag === "img") { - addImage(images, token); - } - } - } - - if (images.length > 0) { - let srcList = images.map(([token, srcIndex]) => token.attrs[srcIndex][1]); - let lookup = state.md.options.discourse.lookupImageUrls; - let longUrls = (lookup && lookup(srcList)) || {}; - - images.forEach(([token, srcIndex]) => { - let origSrc = token.attrs[srcIndex][1]; - let mapped = longUrls[origSrc]; - if (mapped) { - token.attrs[srcIndex][1] = mapped; - } else { - token.attrs[srcIndex][1] = state.md.options.discourse.getURL( - "/images/transparent.png" - ); - token.attrs.push(["data-orig-src", origSrc]); - } - }); - } -} - -export function setup(helper) { - const opts = helper.getOptions(); - if (opts.previewing) helper.whiteList(["img.resizable"]); - helper.whiteList(["img[data-orig-src]"]); - helper.registerPlugin(md => { - md.core.ruler.push("image-protocol", rule); - }); -} diff --git a/app/assets/javascripts/pretty-text/engines/discourse-markdown/upload-protocol.js.es6 b/app/assets/javascripts/pretty-text/engines/discourse-markdown/upload-protocol.js.es6 new file mode 100644 index 00000000000..27cfb918600 --- /dev/null +++ b/app/assets/javascripts/pretty-text/engines/discourse-markdown/upload-protocol.js.es6 @@ -0,0 +1,81 @@ +// add image to array if src has an upload +function addImage(uploads, token) { + if (token.attrs) { + for (let i = 0; i < token.attrs.length; i++) { + if (token.attrs[i][1].indexOf("upload://") === 0) { + uploads.push([token, i]); + break; + } + } + } +} + +function rule(state) { + let uploads = []; + + for (let i = 0; i < state.tokens.length; i++) { + let blockToken = state.tokens[i]; + + if (blockToken.tag === "img" || blockToken.tag === "a") { + addImage(uploads, blockToken); + } + + if (!blockToken.children) { + continue; + } + + for (let j = 0; j < blockToken.children.length; j++) { + let token = blockToken.children[j]; + if (token.tag === "img" || token.tag === "a") { + addImage(uploads, token); + } + } + } + + if (uploads.length > 0) { + let srcList = uploads.map(([token, srcIndex]) => token.attrs[srcIndex][1]); + let lookup = state.md.options.discourse.lookupUploadUrls; + let longUrls = (lookup && lookup(srcList)) || {}; + + uploads.forEach(([token, srcIndex]) => { + let origSrc = token.attrs[srcIndex][1]; + let mapped = longUrls[origSrc]; + + switch (token.tag) { + case "img": + if (mapped) { + token.attrs[srcIndex][1] = mapped.url; + } else { + token.attrs[srcIndex][1] = state.md.options.discourse.getURL( + "/images/transparent.png" + ); + + token.attrs.push(["data-orig-src", origSrc]); + } + break; + case "a": + if (mapped) { + token.attrs[srcIndex][1] = mapped.short_path; + } else { + token.attrs[srcIndex][1] = state.md.options.discourse.getURL( + "/404" + ); + + token.attrs.push(["data-orig-href", origSrc]); + } + + break; + } + }); + } +} + +export function setup(helper) { + const opts = helper.getOptions(); + if (opts.previewing) helper.whiteList(["img.resizable"]); + helper.whiteList(["img[data-orig-src]", "a[data-orig-href]"]); + + helper.registerPlugin(md => { + md.core.ruler.push("upload-protocol", rule); + }); +} diff --git a/app/assets/javascripts/pretty-text/image-short-url.js.es6 b/app/assets/javascripts/pretty-text/image-short-url.js.es6 deleted file mode 100644 index e457210dd76..00000000000 --- a/app/assets/javascripts/pretty-text/image-short-url.js.es6 +++ /dev/null @@ -1,68 +0,0 @@ -let _cache = {}; - -export function lookupCachedUploadUrl(shortUrl) { - return _cache[shortUrl]; -} - -export function lookupUncachedUploadUrls(urls, ajax) { - return ajax("/uploads/lookup-urls", { - method: "POST", - data: { short_urls: urls } - }).then(uploads => { - uploads.forEach(upload => - cacheShortUploadUrl(upload.short_url, upload.url) - ); - urls.forEach(url => - cacheShortUploadUrl(url, lookupCachedUploadUrl(url) || "missing") - ); - return uploads; - }); -} - -export function cacheShortUploadUrl(shortUrl, url) { - _cache[shortUrl] = url; -} - -export function resetCache() { - _cache = {}; -} - -function _loadCachedShortUrls($images) { - $images.each((idx, image) => { - const $image = $(image); - const url = lookupCachedUploadUrl($image.data("orig-src")); - - if (url) { - $image.removeAttr("data-orig-src"); - if (url !== "missing") { - $image.attr("src", url); - } - } - }); -} - -function _loadShortUrls($images, ajax) { - const urls = $images.toArray().map(img => $(img).data("orig-src")); - return lookupUncachedUploadUrls(urls, ajax).then(() => - _loadCachedShortUrls($images) - ); -} - -export function resolveAllShortUrls(ajax) { - let $shortUploadUrls = $("img[data-orig-src]"); - - if ($shortUploadUrls.length > 0) { - _loadCachedShortUrls($shortUploadUrls); - - $shortUploadUrls = $("img[data-orig-src]"); - if ($shortUploadUrls.length > 0) { - // this is carefully batched so we can do a leading debounce (trigger right away) - return Ember.run.debounce( - null, - () => _loadShortUrls($shortUploadUrls, ajax), - 450, - true - ); - } - } -} diff --git a/app/assets/javascripts/pretty-text/pretty-text.js.es6 b/app/assets/javascripts/pretty-text/pretty-text.js.es6 index da572d891ad..33836cddd3f 100644 --- a/app/assets/javascripts/pretty-text/pretty-text.js.es6 +++ b/app/assets/javascripts/pretty-text/pretty-text.js.es6 @@ -26,7 +26,7 @@ export function buildOptions(state) { lookupPrimaryUserGroupByPostNumber, formatUsername, emojiUnicodeReplacer, - lookupImageUrls, + lookupUploadUrls, previewing, linkify, censoredWords @@ -65,7 +65,7 @@ export function buildOptions(state) { lookupPrimaryUserGroupByPostNumber, formatUsername, emojiUnicodeReplacer, - lookupImageUrls, + lookupUploadUrls, censoredWords, allowedHrefSchemes: siteSettings.allowed_href_schemes ? siteSettings.allowed_href_schemes.split("|") diff --git a/app/assets/javascripts/pretty-text/upload-short-url.js.es6 b/app/assets/javascripts/pretty-text/upload-short-url.js.es6 new file mode 100644 index 00000000000..8b0b4db2b4d --- /dev/null +++ b/app/assets/javascripts/pretty-text/upload-short-url.js.es6 @@ -0,0 +1,111 @@ +let _cache = {}; + +export function lookupCachedUploadUrl(shortUrl) { + return _cache[shortUrl] || {}; +} + +const MISSING = "missing"; + +export function lookupUncachedUploadUrls(urls, ajax) { + return ajax("/uploads/lookup-urls", { + method: "POST", + data: { short_urls: urls } + }).then(uploads => { + uploads.forEach(upload => { + cacheShortUploadUrl(upload.short_url, { + url: upload.url, + short_path: upload.short_path + }); + }); + + urls.forEach(url => + cacheShortUploadUrl(url, { + url: lookupCachedUploadUrl(url).url || MISSING, + short_path: lookupCachedUploadUrl(url).short_path || MISSING + }) + ); + + return uploads; + }); +} + +export function cacheShortUploadUrl(shortUrl, value) { + _cache[shortUrl] = value; +} + +export function resetCache() { + _cache = {}; +} + +export const ATTACHMENT_CSS_CLASS = "attachment"; + +function _loadCachedShortUrls($uploads) { + $uploads.each((idx, upload) => { + const $upload = $(upload); + let url; + + switch (upload.tagName) { + case "A": + url = lookupCachedUploadUrl($upload.data("orig-href")).short_path; + + if (url) { + $upload.removeAttr("data-orig-href"); + + if (url !== MISSING) { + $upload.attr("href", url); + const content = $upload.text().split("|"); + + if (content[1] === ATTACHMENT_CSS_CLASS) { + $upload.addClass(ATTACHMENT_CSS_CLASS); + $upload.text(content[0]); + } + } + } + + break; + case "IMG": + url = lookupCachedUploadUrl($upload.data("orig-src")).url; + + if (url) { + $upload.removeAttr("data-orig-src"); + + if (url !== MISSING) { + $upload.attr("src", url); + } + } + + break; + } + }); +} + +function _loadShortUrls($uploads, ajax) { + const urls = $uploads.toArray().map(upload => { + const $upload = $(upload); + return $upload.data("orig-src") || $upload.data("orig-href"); + }); + + return lookupUncachedUploadUrls(urls, ajax).then(() => + _loadCachedShortUrls($uploads) + ); +} + +export function resolveAllShortUrls(ajax) { + const attributes = "img[data-orig-src], a[data-orig-href]"; + let $shortUploadUrls = $(attributes); + + if ($shortUploadUrls.length > 0) { + _loadCachedShortUrls($shortUploadUrls); + + $shortUploadUrls = $(attributes); + if ($shortUploadUrls.length > 0) { + // this is carefully batched so we can do a leading debounce (trigger right away) + return Ember.run.debounce( + null, + () => _loadShortUrls($shortUploadUrls, ajax), + 450, + true + ); + } + } +} diff --git a/app/controllers/uploads_controller.rb b/app/controllers/uploads_controller.rb index fb1ba6333d6..64852af0160 100644 --- a/app/controllers/uploads_controller.rb +++ b/app/controllers/uploads_controller.rb @@ -5,9 +5,9 @@ require_dependency 'upload_creator' require_dependency "file_store/local_store" class UploadsController < ApplicationController - requires_login except: [:show] + requires_login except: [:show, :show_short] - skip_before_action :preload_json, :check_xhr, :redirect_to_login_if_required, only: [:show] + skip_before_action :preload_json, :check_xhr, :redirect_to_login_if_required, only: [:show, :show_short] def create # capture current user for block later on @@ -56,8 +56,12 @@ class UploadsController < ApplicationController uploads = [] if (params[:short_urls] && params[:short_urls].length > 0) - PrettyText::Helpers.lookup_image_urls(params[:short_urls]).each do |short_url, url| - uploads << { short_url: short_url, url: url } + PrettyText::Helpers.lookup_upload_urls(params[:short_urls]).each do |short_url, paths| + uploads << { + short_url: short_url, + url: paths[:url], + short_path: paths[:short_path] + } end end @@ -76,23 +80,31 @@ class UploadsController < ApplicationController return render_404 unless local_store.has_been_uploaded?(upload.url) end - opts = { - filename: upload.original_filename, - content_type: MiniMime.lookup_by_filename(upload.original_filename)&.content_type, - } - opts[:disposition] = "inline" if params[:inline] - opts[:disposition] ||= "attachment" unless FileHelper.is_supported_image?(upload.original_filename) - - file_path = Discourse.store.path_for(upload) - return render_404 unless file_path - - send_file(file_path, opts) + send_file_local_upload(upload) else render_404 end end end + def show_short + if SiteSetting.prevent_anons_from_downloading_files && current_user.nil? + return render_404 + end + + sha1 = Upload.sha1_from_base62_encoded(params[:base62]) + + if upload = Upload.find_by(sha1: sha1) + if Discourse.store.internal? + send_file_local_upload(upload) + else + redirect_to Discourse.store.path_for(upload) + end + else + render_404 + end + end + def metadata params.require(:url) upload = Upload.get_from_url(params[:url]) @@ -165,4 +177,24 @@ class UploadsController < ApplicationController tempfile&.close! end + private + + def send_file_local_upload(upload) + opts = { + filename: upload.original_filename, + content_type: MiniMime.lookup_by_filename(upload.original_filename)&.content_type + } + + if params[:inline] + opts[:disposition] = "inline" + elsif !FileHelper.is_supported_image?(upload.original_filename) + opts[:disposition] = "attachment" + end + + file_path = Discourse.store.path_for(upload) + return render_404 unless file_path + + send_file(file_path, opts) + end + end diff --git a/app/models/post.rb b/app/models/post.rb index 98a7d3878d8..efb75dd8827 100644 --- a/app/models/post.rb +++ b/app/models/post.rb @@ -889,8 +889,10 @@ class Post < ActiveRecord::Base upload_patterns = [ /\/uploads\/#{RailsMultisite::ConnectionManagement.current_db}\//, /\/original\//, - /\/optimized\// + /\/optimized\//, + /\/uploads\/short-url\/[a-zA-Z0-9]+\..*/ ] + fragments ||= Nokogiri::HTML::fragment(self.cooked) links = fragments.css("a/@href", "img/@src").map { |media| media.value }.uniq @@ -911,7 +913,7 @@ class Post < ActiveRecord::Base if path.include? "optimized" OptimizedImage.extract_sha1(path) else - Upload.extract_sha1(path) + Upload.extract_sha1(path) || Upload.sha1_from_short_path(path) end yield(src, path, sha1) diff --git a/app/models/upload.rb b/app/models/upload.rb index 6a32afcc58a..bfb3fd2eddb 100644 --- a/app/models/upload.rb +++ b/app/models/upload.rb @@ -117,7 +117,28 @@ class Upload < ActiveRecord::Base end def short_url - "upload://#{Base62.encode(sha1.hex)}.#{extension}" + "upload://#{short_url_basename}" + end + + def short_path + self.class.short_path(sha1: self.sha1, extension: self.extension) + end + + def self.short_path(sha1:, extension:) + @url_helpers ||= Rails.application.routes.url_helpers + + @url_helpers.upload_short_path( + base62: self.base62_sha1(sha1), + extension: extension + ) + end + + def self.base62_sha1(sha1) + Base62.encode(sha1.hex) + end + + def base62_sha1 + Upload.base62_sha1(upload.sha1) end def local? @@ -180,15 +201,25 @@ class Upload < ActiveRecord::Base get_dimension(:thumbnail_height) end + def self.sha1_from_short_path(path) + if path =~ /(\/uploads\/short-url\/)([a-zA-Z0-9]+)(\..*)?/ + self.sha1_from_base62_encoded($2) + end + end + def self.sha1_from_short_url(url) if url =~ /(upload:\/\/)?([a-zA-Z0-9]+)(\..*)?/ - sha1 = Base62.decode($2).to_s(16) + self.sha1_from_base62_encoded($2) + end + end - if sha1.length > SHA1_LENGTH - nil - else - sha1.rjust(SHA1_LENGTH, '0') - end + def self.sha1_from_base62_encoded(encoded_sha1) + sha1 = Base62.decode(encoded_sha1).to_s(16) + + if sha1.length > SHA1_LENGTH + nil + else + sha1.rjust(SHA1_LENGTH, '0') end end @@ -322,6 +353,12 @@ class Upload < ActiveRecord::Base problems end + private + + def short_url_basename + "#{Upload.base62_sha1(sha1)}.#{extension}" + end + end # == Schema Information diff --git a/config/routes.rb b/config/routes.rb index 0b42f7fba14..c996544c2e0 100644 --- a/config/routes.rb +++ b/config/routes.rb @@ -486,6 +486,7 @@ Discourse::Application.routes.draw do # used to download original images get "uploads/:site/:sha(.:extension)" => "uploads#show", constraints: { site: /\w+/, sha: /\h{40}/, extension: /[a-z0-9\.]+/i } + get "uploads/short-url/:base62(.:extension)" => "uploads#show_short", constraints: { site: /\w+/, base62: /[a-zA-Z0-9]+/, extension: /[a-z0-9\.]+/i }, as: :upload_short # used to download attachments get "uploads/:site/original/:tree:sha(.:extension)" => "uploads#show", constraints: { site: /\w+/, tree: /([a-z0-9]+\/)+/i, sha: /\h{40}/, extension: /[a-z0-9\.]+/i } # used to download attachments (old route) diff --git a/lib/file_store/s3_store.rb b/lib/file_store/s3_store.rb index 9ca94b98f79..df3296f7abd 100644 --- a/lib/file_store/s3_store.rb +++ b/lib/file_store/s3_store.rb @@ -101,8 +101,13 @@ module FileStore end def path_for(upload) - url = upload.try(:url) - FileStore::LocalStore.new.path_for(upload) if url && url[/^\/[^\/]/] + url = upload&.url + + if url && url[/^\/[^\/]/] + FileStore::LocalStore.new.path_for(upload) + else + url + end end def cdn_url(url) diff --git a/lib/pretty_text.rb b/lib/pretty_text.rb index 32063ad5c4e..79b6e66a4d7 100644 --- a/lib/pretty_text.rb +++ b/lib/pretty_text.rb @@ -159,7 +159,7 @@ module PrettyText __optInput.categoryHashtagLookup = __categoryLookup; __optInput.customEmoji = #{custom_emoji.to_json}; __optInput.emojiUnicodeReplacer = __emojiUnicodeReplacer; - __optInput.lookupImageUrls = __lookupImageUrls; + __optInput.lookupUploadUrls = __lookupUploadUrls; __optInput.censoredWords = #{WordWatcher.words_for_action(:censor).join('|').to_json}; JS diff --git a/lib/pretty_text/helpers.rb b/lib/pretty_text/helpers.rb index 3337282268b..d42ba3749bc 100644 --- a/lib/pretty_text/helpers.rb +++ b/lib/pretty_text/helpers.rb @@ -49,7 +49,7 @@ module PrettyText end end - def lookup_image_urls(urls) + def lookup_upload_urls(urls) map = {} result = {} @@ -66,11 +66,16 @@ module PrettyText reverse_map[value] << key end - Upload.where(sha1: map.values).pluck(:sha1, :url).each do |row| - sha1, url = row + Upload.where(sha1: map.values).pluck(:sha1, :url, :extension).each do |row| + sha1, url, extension = row if short_urls = reverse_map[sha1] - short_urls.each { |short_url| result[short_url] = url } + short_urls.each do |short_url| + result[short_url] = { + url: url, + short_path: Upload.short_path(sha1: sha1, extension: extension) + } + end end end end diff --git a/lib/pretty_text/shims.js b/lib/pretty_text/shims.js index 929dafd13ab..44f80690720 100644 --- a/lib/pretty_text/shims.js +++ b/lib/pretty_text/shims.js @@ -65,8 +65,8 @@ function __getURL(url) { return url; } -function __lookupImageUrls(urls) { - return __helpers.lookup_image_urls(urls); +function __lookupUploadUrls(urls) { + return __helpers.lookup_upload_urls(urls); } function __getTopicInfo(i) { diff --git a/spec/components/file_store/s3_store_spec.rb b/spec/components/file_store/s3_store_spec.rb index 1e497a66976..4474613c50d 100644 --- a/spec/components/file_store/s3_store_spec.rb +++ b/spec/components/file_store/s3_store_spec.rb @@ -303,21 +303,12 @@ describe FileStore::S3Store do end describe ".path_for" do - - def assert_path(path, expected) - upload = Upload.new(url: path) - - path = store.path_for(upload) - expected = FileStore::LocalStore.new.path_for(upload) if expected - - expect(path).to eq(expected) - end - it "correctly falls back to local" do - assert_path("/hello", "/hello") - assert_path("//hello", nil) - assert_path("http://hello", nil) - assert_path("https://hello", nil) + local_upload = Fabricate(:upload) + s3_upload = Fabricate(:upload_s3) + + expect(Discourse.store.path_for(local_upload)).to eq(local_upload.url) + expect(Discourse.store.path_for(s3_upload)).to eq(s3_upload.url) end end diff --git a/spec/components/pretty_text_spec.rb b/spec/components/pretty_text_spec.rb index 9ace739e48a..71da86ead13 100644 --- a/spec/components/pretty_text_spec.rb +++ b/spec/components/pretty_text_spec.rb @@ -1260,9 +1260,11 @@ HTML end - describe "image decoding" do + describe "upload decoding" do it "can decode upload:// for default setup" do + set_cdn_url('https://cdn.com') + upload = Fabricate(:upload) raw = <<~RAW @@ -1274,6 +1276,12 @@ HTML - ![upload](#{upload.short_url}) ![upload](#{upload.short_url.gsub(".png", "")}) + + [some attachment](#{upload.short_url}) + + [some attachment|attachment](#{upload.short_url}) + + [some attachment|random](#{upload.short_url}) RAW cooked = <<~HTML @@ -1290,6 +1298,9 @@ HTML

upload

+

some attachment

+

some attachment

+

some attachment|random

HTML expect(PrettyText.cook(raw)).to eq(cooked.strip) @@ -1297,10 +1308,15 @@ HTML it "can place a blank image if we can not find the upload" do - raw = "![upload](upload://abcABC.png)" + raw = <<~MD + ![upload](upload://abcABC.png) + + [some attachment|attachment](upload://abcdefg.png) + MD cooked = <<~HTML -

upload

+

upload

+

some attachment|attachment

HTML expect(PrettyText.cook(raw)).to eq(cooked.strip) diff --git a/spec/models/post_spec.rb b/spec/models/post_spec.rb index 1b27825fe9b..07cfc8027c6 100644 --- a/spec/models/post_spec.rb +++ b/spec/models/post_spec.rb @@ -1229,29 +1229,11 @@ describe Post do end describe '#link_post_uploads' do - fab!(:video_upload) do - Fabricate(:upload, - url: '/uploads/default/original/1X/1/1234567890123456.mp4' - ) - end - - fab!(:image_upload) do - Fabricate(:upload, - url: '/uploads/default/original/1X/1/1234567890123456.jpg' - ) - end - - fab!(:audio_upload) do - Fabricate(:upload, - url: '/uploads/default/original/1X/1/1234567890123456.ogg' - ) - end - - fab!(:attachment_upload) do - Fabricate(:upload, - url: '/uploads/default/original/1X/1/1234567890123456.csv' - ) - end + fab!(:video_upload) { Fabricate(:upload, extension: "mp4") } + fab!(:image_upload) { Fabricate(:upload) } + fab!(:audio_upload) { Fabricate(:upload, extension: "ogg") } + fab!(:attachment_upload) { Fabricate(:upload, extension: "csv") } + fab!(:attachment_upload_2) { Fabricate(:upload) } let(:base_url) { "#{Discourse.base_url_no_prefix}#{Discourse.base_uri}" } let(:video_url) { "#{base_url}#{video_upload.url}" } @@ -1260,6 +1242,7 @@ describe Post do let(:raw) do <<~RAW Link + [test|attachment](#{attachment_upload_2.short_url})