publify_core 9.0.0.pre1
Sign up to get free protection for your applications and to get access to all the features.
Potentially problematic release.
This version of publify_core might be problematic. Click here for more details.
- checksums.yaml +7 -0
- data/MIT-LICENSE +21 -0
- data/README.rdoc +3 -0
- data/Rakefile +35 -0
- data/app/assets/fonts/open-sans-bold.woff +0 -0
- data/app/assets/fonts/open-sans.woff +0 -0
- data/app/assets/images/admin/loading.gif +0 -0
- data/app/assets/images/admin/typologo.gif +0 -0
- data/app/assets/images/calendar_date_select/calendar.gif +0 -0
- data/app/assets/images/close.gif +0 -0
- data/app/assets/images/closelabel.gif +0 -0
- data/app/assets/images/go.png +0 -0
- data/app/assets/images/loading.gif +0 -0
- data/app/assets/images/nextlabel.gif +0 -0
- data/app/assets/images/overlay.png +0 -0
- data/app/assets/images/powered.gif +0 -0
- data/app/assets/images/prevlabel.gif +0 -0
- data/app/assets/images/spinner-blue.gif +0 -0
- data/app/assets/images/spinner.gif +0 -0
- data/app/assets/images/thumb_blank.jpg +0 -0
- data/app/assets/javascripts/application.js +2 -0
- data/app/assets/javascripts/cookies.js +60 -0
- data/app/assets/javascripts/datetimepicker.js +1470 -0
- data/app/assets/javascripts/lang/da_DK.js +17 -0
- data/app/assets/javascripts/lang/default.js +21 -0
- data/app/assets/javascripts/lang/en_US.js +21 -0
- data/app/assets/javascripts/lang/fr_FR.js +21 -0
- data/app/assets/javascripts/lang/nl_NL.js +21 -0
- data/app/assets/javascripts/lang/zh_TW.js +17 -0
- data/app/assets/javascripts/lightbox.js +350 -0
- data/app/assets/javascripts/observe.js +28 -0
- data/app/assets/javascripts/publify.js +11 -0
- data/app/assets/javascripts/publify_admin.js +138 -0
- data/app/assets/javascripts/quicktags.js +440 -0
- data/app/assets/javascripts/set-timeago-lang.js +3 -0
- data/app/assets/javascripts/sidebar.js +28 -0
- data/app/assets/javascripts/spinnable.js +5 -0
- data/app/assets/javascripts/tagmanager.js +481 -0
- data/app/assets/javascripts/typeahead.js +1139 -0
- data/app/assets/javascripts/widearea.js +486 -0
- data/app/assets/stylesheets/accounts.css.scss +7 -0
- data/app/assets/stylesheets/administration_structure.css.scss +239 -0
- data/app/assets/stylesheets/coderay.css +135 -0
- data/app/assets/stylesheets/datetimepicker.css +306 -0
- data/app/assets/stylesheets/lightbox.css +63 -0
- data/app/assets/stylesheets/publify.css.scss +9 -0
- data/app/assets/stylesheets/publify_admin.css.scss +11 -0
- data/app/assets/stylesheets/rss.css +53 -0
- data/app/assets/stylesheets/sidebar_admin.css.scss +36 -0
- data/app/assets/stylesheets/tagmanager.css +102 -0
- data/app/assets/stylesheets/user-styles.css +29 -0
- data/app/assets/stylesheets/widearea.css +133 -0
- data/app/controllers/accounts_controller.rb +2 -0
- data/app/controllers/admin/base_controller.rb +41 -0
- data/app/controllers/admin/cache_controller.rb +33 -0
- data/app/controllers/admin/content_controller.rb +174 -0
- data/app/controllers/admin/dashboard_controller.rb +87 -0
- data/app/controllers/admin/feedback_controller.rb +159 -0
- data/app/controllers/admin/migrations_controller.rb +20 -0
- data/app/controllers/admin/notes_controller.rb +69 -0
- data/app/controllers/admin/pages_controller.rb +88 -0
- data/app/controllers/admin/post_types_controller.rb +56 -0
- data/app/controllers/admin/profiles_controller.rb +48 -0
- data/app/controllers/admin/redirects_controller.rb +47 -0
- data/app/controllers/admin/resources_controller.rb +30 -0
- data/app/controllers/admin/seo_controller.rb +45 -0
- data/app/controllers/admin/settings_controller.rb +53 -0
- data/app/controllers/admin/sidebar_controller.rb +66 -0
- data/app/controllers/admin/tags_controller.rb +53 -0
- data/app/controllers/admin/textfilters_controller.rb +6 -0
- data/app/controllers/admin/themes_controller.rb +37 -0
- data/app/controllers/admin/users_controller.rb +65 -0
- data/app/controllers/articles_controller.rb +205 -0
- data/app/controllers/authors_controller.rb +27 -0
- data/app/controllers/base_controller.rb +45 -0
- data/app/controllers/comments_controller.rb +69 -0
- data/app/controllers/content_controller.rb +31 -0
- data/app/controllers/feedback_controller.rb +47 -0
- data/app/controllers/notes_controller.rb +37 -0
- data/app/controllers/setup_controller.rb +62 -0
- data/app/controllers/tags_controller.rb +55 -0
- data/app/controllers/text_controller.rb +9 -0
- data/app/controllers/textfilter_controller.rb +3 -0
- data/app/controllers/theme_controller.rb +59 -0
- data/app/controllers/trackbacks_controller.rb +36 -0
- data/app/controllers/xml_controller.rb +70 -0
- data/app/helpers/admin/base_helper.rb +87 -0
- data/app/helpers/admin/feedback_helper.rb +42 -0
- data/app/helpers/articles_helper.rb +8 -0
- data/app/helpers/authors_helper.rb +39 -0
- data/app/helpers/base_helper.rb +246 -0
- data/app/helpers/blog_helper.rb +12 -0
- data/app/helpers/xml_helper.rb +16 -0
- data/app/mailers/notification_mailer.rb +38 -0
- data/app/models/ability.rb +52 -0
- data/app/models/archives_sidebar.rb +45 -0
- data/app/models/article/factory.rb +56 -0
- data/app/models/article/states.rb +178 -0
- data/app/models/article.rb +321 -0
- data/app/models/blog.rb +290 -0
- data/app/models/blog_sweeper.rb +86 -0
- data/app/models/comment.rb +53 -0
- data/app/models/config_manager.rb +81 -0
- data/app/models/content.rb +138 -0
- data/app/models/content_base.rb +95 -0
- data/app/models/feedback/states.rb +256 -0
- data/app/models/feedback.rb +225 -0
- data/app/models/meta_sidebar.rb +8 -0
- data/app/models/note.rb +144 -0
- data/app/models/page.rb +36 -0
- data/app/models/page_cache.rb +33 -0
- data/app/models/page_sidebar.rb +12 -0
- data/app/models/ping.rb +116 -0
- data/app/models/post_type.rb +15 -0
- data/app/models/redirect.rb +45 -0
- data/app/models/redirection.rb +4 -0
- data/app/models/resource.rb +28 -0
- data/app/models/search_sidebar.rb +7 -0
- data/app/models/sidebar.rb +138 -0
- data/app/models/static_sidebar.rb +20 -0
- data/app/models/tag.rb +63 -0
- data/app/models/tag_sidebar.rb +27 -0
- data/app/models/text_filter.rb +61 -0
- data/app/models/trackback.rb +58 -0
- data/app/models/trigger.rb +45 -0
- data/app/models/user.rb +148 -0
- data/app/services/migrator.rb +25 -0
- data/app/services/title_builder.rb +80 -0
- data/app/uploaders/resource_uploader.rb +30 -0
- data/app/views/accounts/confirm.html.erb +8 -0
- data/app/views/admin/cache/show.html.erb +18 -0
- data/app/views/admin/content/_article_list.html.erb +26 -0
- data/app/views/admin/content/_form.html.erb +165 -0
- data/app/views/admin/content/autosave.js.erb +5 -0
- data/app/views/admin/content/edit.html.erb +3 -0
- data/app/views/admin/content/index.html.erb +48 -0
- data/app/views/admin/content/index.js.erb +1 -0
- data/app/views/admin/content/new.html.erb +3 -0
- data/app/views/admin/dashboard/_comment.html.erb +18 -0
- data/app/views/admin/dashboard/_comments.html.erb +10 -0
- data/app/views/admin/dashboard/_drafts.html.erb +25 -0
- data/app/views/admin/dashboard/_inbound.html.erb +31 -0
- data/app/views/admin/dashboard/_overview.html.erb +23 -0
- data/app/views/admin/dashboard/_welcome.html.erb +28 -0
- data/app/views/admin/dashboard/index.html.erb +17 -0
- data/app/views/admin/feedback/_button.html.erb +19 -0
- data/app/views/admin/feedback/_feedback.html.erb +7 -0
- data/app/views/admin/feedback/_ham.html.erb +17 -0
- data/app/views/admin/feedback/_spam.html.erb +23 -0
- data/app/views/admin/feedback/article.html.erb +69 -0
- data/app/views/admin/feedback/edit.html.erb +48 -0
- data/app/views/admin/feedback/ham.js +1 -0
- data/app/views/admin/feedback/index.html.erb +53 -0
- data/app/views/admin/feedback/spam.js +1 -0
- data/app/views/admin/migrations/show.html.erb +39 -0
- data/app/views/admin/notes/_form.html.erb +37 -0
- data/app/views/admin/notes/_header.html.erb +6 -0
- data/app/views/admin/notes/_list.html.erb +13 -0
- data/app/views/admin/notes/_note.html.erb +14 -0
- data/app/views/admin/notes/edit.html.erb +11 -0
- data/app/views/admin/notes/index.html.erb +11 -0
- data/app/views/admin/notes/show.html.erb +14 -0
- data/app/views/admin/pages/_form.html.erb +101 -0
- data/app/views/admin/pages/_pages.html.erb +21 -0
- data/app/views/admin/pages/edit.html.erb +1 -0
- data/app/views/admin/pages/index.html.erb +17 -0
- data/app/views/admin/pages/new.html.erb +1 -0
- data/app/views/admin/post_types/_index_and_form.html.erb +65 -0
- data/app/views/admin/post_types/edit.html.erb +1 -0
- data/app/views/admin/post_types/index.html.erb +1 -0
- data/app/views/admin/profiles/index.html.erb +10 -0
- data/app/views/admin/redirects/_index_and_form.html.erb +68 -0
- data/app/views/admin/redirects/edit.html.erb +1 -0
- data/app/views/admin/redirects/index.html.erb +1 -0
- data/app/views/admin/resources/index.html.erb +68 -0
- data/app/views/admin/seo/_general.html.erb +123 -0
- data/app/views/admin/seo/_permalinks.html.erb +53 -0
- data/app/views/admin/seo/_titles.html.erb +210 -0
- data/app/views/admin/seo/show.html.erb +32 -0
- data/app/views/admin/settings/display.html.erb +110 -0
- data/app/views/admin/settings/feedback.html.erb +125 -0
- data/app/views/admin/settings/index.html.erb +73 -0
- data/app/views/admin/settings/write.html.erb +87 -0
- data/app/views/admin/shared/_edit.html.erb +4 -0
- data/app/views/admin/shared/_menu.html.erb +122 -0
- data/app/views/admin/shared/_twitter_alert.html.erb +3 -0
- data/app/views/admin/sidebar/_available.html.erb +6 -0
- data/app/views/admin/sidebar/_available.json.erb +6 -0
- data/app/views/admin/sidebar/_config.html.erb +27 -0
- data/app/views/admin/sidebar/_target.html.erb +9 -0
- data/app/views/admin/sidebar/_target_sidebar.html.erb +20 -0
- data/app/views/admin/sidebar/destroy.js.erb +1 -0
- data/app/views/admin/sidebar/index.html.erb +31 -0
- data/app/views/admin/sidebar/sortable.js.erb +3 -0
- data/app/views/admin/sidebar/update.js.erb +2 -0
- data/app/views/admin/tags/_index_and_form.html.erb +59 -0
- data/app/views/admin/tags/edit.html.erb +1 -0
- data/app/views/admin/tags/index.html.erb +1 -0
- data/app/views/admin/themes/index.html.erb +27 -0
- data/app/views/admin/users/_form.html.erb +215 -0
- data/app/views/admin/users/edit.html.erb +8 -0
- data/app/views/admin/users/index.html.erb +39 -0
- data/app/views/admin/users/new.html.erb +8 -0
- data/app/views/archives_sidebar/_content.html.erb +13 -0
- data/app/views/articles/_article.html.erb +9 -0
- data/app/views/articles/_article_collection.html.erb +8 -0
- data/app/views/articles/_article_content.html.erb +5 -0
- data/app/views/articles/_article_excerpt.html.erb +13 -0
- data/app/views/articles/_article_links.html.erb +10 -0
- data/app/views/articles/_comment.html.erb +1 -0
- data/app/views/articles/_comment_errors.html.erb +2 -0
- data/app/views/articles/_comment_form.html.erb +48 -0
- data/app/views/articles/_comment_list.html.erb +5 -0
- data/app/views/articles/_comment_preview.html.erb +4 -0
- data/app/views/articles/_full_article_content.html.erb +2 -0
- data/app/views/articles/_password_form.html.erb +10 -0
- data/app/views/articles/_protected_article_content.html.erb +6 -0
- data/app/views/articles/_trackback.html.erb +6 -0
- data/app/views/articles/archives.html.erb +25 -0
- data/app/views/articles/comment.js.erb +5 -0
- data/app/views/articles/comment_failed.js.erb +3 -0
- data/app/views/articles/error.html.erb +3 -0
- data/app/views/articles/feedback_atom_feed.atom.builder +8 -0
- data/app/views/articles/feedback_rss_feed.rss.builder +21 -0
- data/app/views/articles/index.html.erb +1 -0
- data/app/views/articles/index_atom_feed.atom.builder +8 -0
- data/app/views/articles/index_rss_feed.rss.builder +20 -0
- data/app/views/articles/live_search.html.erb +10 -0
- data/app/views/articles/read.html.erb +61 -0
- data/app/views/articles/search.html.erb +8 -0
- data/app/views/articles/trackback.xml.builder +5 -0
- data/app/views/articles/view_page.html.erb +3 -0
- data/app/views/authors/show.html.erb +40 -0
- data/app/views/authors/show_atom_feed.atom.builder +8 -0
- data/app/views/authors/show_rss_feed.rss.builder +20 -0
- data/app/views/comments/_comment.html.erb +16 -0
- data/app/views/comments/index.html.erb +1 -0
- data/app/views/comments/index_atom_feed.atom.builder +8 -0
- data/app/views/comments/index_rss_feed.rss.builder +20 -0
- data/app/views/comments/preview.html.erb +1 -0
- data/app/views/comments/preview.js.erb +3 -0
- data/app/views/devise/mailer/reset_password_instructions.html.erb +13 -0
- data/app/views/devise/passwords/edit.html.erb +28 -0
- data/app/views/devise/passwords/new.html.erb +20 -0
- data/app/views/devise/registrations/new.html.erb +36 -0
- data/app/views/devise/sessions/new.html.erb +32 -0
- data/app/views/devise/shared/_links.html.erb +15 -0
- data/app/views/errors/404.html.erb +2 -0
- data/app/views/layouts/accounts.html.erb +33 -0
- data/app/views/layouts/administration.html.erb +37 -0
- data/app/views/layouts/default.html.erb +32 -0
- data/app/views/layouts/editor.html.erb +31 -0
- data/app/views/meta_sidebar/_content.html.erb +8 -0
- data/app/views/notes/_note.html.erb +15 -0
- data/app/views/notes/error.html.erb +3 -0
- data/app/views/notes/index.html.erb +15 -0
- data/app/views/notes/show.html.erb +5 -0
- data/app/views/notes/show_in_reply.html.erb +16 -0
- data/app/views/notification_mailer/_mail_footer.html.erb +7 -0
- data/app/views/notification_mailer/_mail_header.html.erb +1 -0
- data/app/views/notification_mailer/article.html.erb +6 -0
- data/app/views/notification_mailer/comment.html.erb +11 -0
- data/app/views/notification_mailer/notif_user.html.erb +14 -0
- data/app/views/page_sidebar/_content.html.erb +12 -0
- data/app/views/search_sidebar/_content.html.erb +10 -0
- data/app/views/settings/done.html.erb +2 -0
- data/app/views/settings/install.html.erb +12 -0
- data/app/views/setup/index.html.erb +13 -0
- data/app/views/shared/_atom_header.atom.builder +6 -0
- data/app/views/shared/_atom_item_article.atom.builder +39 -0
- data/app/views/shared/_atom_item_comment.atom.builder +10 -0
- data/app/views/shared/_atom_item_trackback.atom.builder +9 -0
- data/app/views/shared/_flash.erb +10 -0
- data/app/views/shared/_page_header.html.erb +26 -0
- data/app/views/shared/_rss_item_article.rss.builder +35 -0
- data/app/views/shared/_rss_item_comment.rss.builder +8 -0
- data/app/views/shared/_rss_item_trackback.rss.builder +7 -0
- data/app/views/sidebar/_row.html.erb +1 -0
- data/app/views/sidebar/_sidebar.html.erb +5 -0
- data/app/views/sidebar/display_plugins.html.erb +5 -0
- data/app/views/sidebar/show.html.erb +1 -0
- data/app/views/static_sidebar/_content.html.erb +2 -0
- data/app/views/tag_sidebar/_content.html.erb +10 -0
- data/app/views/tags/index.html.erb +15 -0
- data/app/views/tags/show.html.erb +1 -0
- data/app/views/theme/static_view_test.html.erb +1 -0
- data/app/views/trackbacks/index_atom_feed.atom.builder +7 -0
- data/app/views/trackbacks/index_rss_feed.rss.builder +20 -0
- data/app/views/trackbacks/trackback.xml.builder +4 -0
- data/app/views/xml/_googlesitemap_item_article.googlesitemap.builder +5 -0
- data/app/views/xml/_googlesitemap_item_category.googlesitemap.builder +4 -0
- data/app/views/xml/_googlesitemap_item_page.googlesitemap.builder +4 -0
- data/app/views/xml/_googlesitemap_item_tag.googlesitemap.builder +4 -0
- data/app/views/xml/feed.googlesitemap.builder +7 -0
- data/app/views/xml/rsd.rsd.builder +8 -0
- data/config/i18n-tasks.yml +49 -0
- data/config/initializers/devise.rb +265 -0
- data/config/initializers/mime_types.rb +6 -0
- data/config/locales/da.yml +827 -0
- data/config/locales/de.yml +827 -0
- data/config/locales/en.yml +827 -0
- data/config/locales/es-MX.yml +827 -0
- data/config/locales/fr.yml +827 -0
- data/config/locales/he.yml +827 -0
- data/config/locales/it.yml +827 -0
- data/config/locales/ja.yml +827 -0
- data/config/locales/lt.yml +827 -0
- data/config/locales/nb-NO.yml +827 -0
- data/config/locales/nl.yml +827 -0
- data/config/locales/pl.yml +827 -0
- data/config/locales/pt-BR.yml +827 -0
- data/config/locales/ro.yml +827 -0
- data/config/locales/ru.yml +827 -0
- data/config/locales/sidebars.da.yml +20 -0
- data/config/locales/sidebars.de.yml +20 -0
- data/config/locales/sidebars.en.yml +20 -0
- data/config/locales/sidebars.es-MX.yml +20 -0
- data/config/locales/sidebars.fr.yml +20 -0
- data/config/locales/sidebars.he.yml +20 -0
- data/config/locales/sidebars.it.yml +20 -0
- data/config/locales/sidebars.ja.yml +20 -0
- data/config/locales/sidebars.lt.yml +20 -0
- data/config/locales/sidebars.nb-NO.yml +20 -0
- data/config/locales/sidebars.nl.yml +20 -0
- data/config/locales/sidebars.pl.yml +20 -0
- data/config/locales/sidebars.pt-BR.yml +20 -0
- data/config/locales/sidebars.ro.yml +20 -0
- data/config/locales/sidebars.ru.yml +20 -0
- data/config/locales/sidebars.zh-CN.yml +20 -0
- data/config/locales/sidebars.zh-TW.yml +20 -0
- data/config/locales/zh-CN.yml +827 -0
- data/config/locales/zh-TW.yml +827 -0
- data/config/routes.rb +177 -0
- data/db/migrate/113_initial_schema.rb +205 -0
- data/db/migrate/114_fixes_buggy_articles_and_notes.rb +52 -0
- data/db/migrate/115_drops_categories_for_tags.rb +34 -0
- data/db/migrate/20150207131657_add_missing_indexes.rb +19 -0
- data/db/migrate/20150807134129_simplify_redirect_relations.rb +38 -0
- data/db/migrate/20150808052637_add_blog_ids.rb +33 -0
- data/db/migrate/20150808191127_add_blog_id_to_redirects.rb +15 -0
- data/db/migrate/20150810094754_add_blog_id_to_tags.rb +15 -0
- data/db/migrate/20160108111120_add_devise_to_users.rb +53 -0
- data/db/migrate/20160108184201_move_last_connection_to_last_sign_in_at.rb +16 -0
- data/db/migrate/20160110094906_remove_profiles_rights.rb +14 -0
- data/db/migrate/20160605103918_replace_profile_id_with_string.rb +30 -0
- data/db/migrate/20160605154632_remove_profiles.rb +24 -0
- data/db/migrate/20160701061851_demand_blog_id_on_contents.rb +9 -0
- data/db/migrate/20160701062604_add_blog_id_to_resources.rb +28 -0
- data/db/seeds.rb +37 -0
- data/lib/email_notify.rb +26 -0
- data/lib/format.rb +17 -0
- data/lib/publify_core/engine.rb +23 -0
- data/lib/publify_core/lang.rb +5 -0
- data/lib/publify_core/version.rb +3 -0
- data/lib/publify_core.rb +56 -0
- data/lib/publify_guid.rb +9 -0
- data/lib/publify_plugins.rb +72 -0
- data/lib/publify_textfilter_markdown.rb +44 -0
- data/lib/publify_textfilter_none.rb +14 -0
- data/lib/publify_textfilter_smartypants.rb +14 -0
- data/lib/publify_textfilter_textile.rb +21 -0
- data/lib/publify_textfilter_twitterfilter.rb +33 -0
- data/lib/publify_time.rb +30 -0
- data/lib/sidebar_field.rb +115 -0
- data/lib/sidebar_registry.rb +33 -0
- data/lib/spam_protection.rb +101 -0
- data/lib/stateful.rb +106 -0
- data/lib/tasks/publify_core_tasks.rake +4 -0
- data/lib/text_filter_plugin.rb +182 -0
- data/lib/theme.rb +72 -0
- data/lib/transforms.rb +45 -0
- metadata +865 -0
@@ -0,0 +1,225 @@
|
|
1
|
+
require_dependency 'spam_protection'
|
2
|
+
require 'akismet'
|
3
|
+
|
4
|
+
class Feedback < ActiveRecord::Base
|
5
|
+
self.table_name = 'feedback'
|
6
|
+
|
7
|
+
belongs_to :text_filter
|
8
|
+
belongs_to :article
|
9
|
+
|
10
|
+
include PublifyGuid
|
11
|
+
include Stateful
|
12
|
+
include ContentBase
|
13
|
+
include States
|
14
|
+
|
15
|
+
class ContentTextHelpers
|
16
|
+
include ActionView::Helpers::UrlHelper
|
17
|
+
include ActionView::Helpers::TextHelper
|
18
|
+
include ActionView::Helpers::SanitizeHelper
|
19
|
+
end
|
20
|
+
|
21
|
+
validate :feedback_not_closed, on: :create
|
22
|
+
validates :article, presence: true
|
23
|
+
|
24
|
+
before_create :create_guid, :article_allows_this_feedback
|
25
|
+
before_save :correct_url, :before_save_handler
|
26
|
+
after_save :post_trigger, :report_classification, :invalidates_cache?
|
27
|
+
after_initialize :after_initialize_handler
|
28
|
+
after_destroy ->(c) { c.invalidates_cache?(true) }
|
29
|
+
|
30
|
+
scope :ham, -> { where(state: %w(presumed_ham ham)) }
|
31
|
+
scope :spam, -> { where(state: 'spam') }
|
32
|
+
scope :published_since, ->(time) { ham.where('published_at > ?', time) }
|
33
|
+
scope :presumed_ham, -> { where(state: 'presumed_ham') }
|
34
|
+
scope :presumed_spam, -> { where(state: 'presumed_spam') }
|
35
|
+
scope :unapproved, -> { where(status_confirmed: false) }
|
36
|
+
|
37
|
+
scope :published, -> { where(published: true) }
|
38
|
+
scope :oldest_first, -> { order(:created_at) }
|
39
|
+
|
40
|
+
has_state(:state,
|
41
|
+
valid_states: [:unclassified, :presumed_spam, :just_marked_as_spam, :spam, :just_presumed_ham, :presumed_ham, :just_marked_as_ham, :ham],
|
42
|
+
handles: [:published?, :status_confirmed?, :just_published?,
|
43
|
+
:mark_as_ham, :mark_as_spam, :confirm_classification,
|
44
|
+
:withdraw,
|
45
|
+
:before_save_handler, :after_initialize_handler,
|
46
|
+
:send_notifications, :post_trigger, :report_classification])
|
47
|
+
|
48
|
+
def self.paginated(page, per_page)
|
49
|
+
page(page).per(per_page)
|
50
|
+
end
|
51
|
+
|
52
|
+
def self.comments
|
53
|
+
Comment.where(published: true).order('created_at DESC')
|
54
|
+
end
|
55
|
+
|
56
|
+
def self.trackbacks
|
57
|
+
Trackback.where(published: true).order('created_at DESC')
|
58
|
+
end
|
59
|
+
|
60
|
+
def self.from(type, article_id = nil)
|
61
|
+
if article_id.present?
|
62
|
+
Article.find(article_id).send("published_#{type}")
|
63
|
+
else
|
64
|
+
send(type)
|
65
|
+
end
|
66
|
+
end
|
67
|
+
|
68
|
+
def parent
|
69
|
+
article
|
70
|
+
end
|
71
|
+
|
72
|
+
def permalink_url(_anchor = :ignored, only_path = false)
|
73
|
+
article.permalink_url("#{self.class.to_s.downcase}-#{id}", only_path)
|
74
|
+
end
|
75
|
+
|
76
|
+
def html_postprocess(_field, html)
|
77
|
+
helper = ContentTextHelpers.new
|
78
|
+
helper.sanitize(helper.auto_link(html))
|
79
|
+
end
|
80
|
+
|
81
|
+
def correct_url
|
82
|
+
return if url.blank?
|
83
|
+
self.url = 'http://' + url.to_s unless url =~ %r{^https?://}
|
84
|
+
end
|
85
|
+
|
86
|
+
def article_allows_this_feedback
|
87
|
+
article && blog_allows_feedback? && article_allows_feedback?
|
88
|
+
end
|
89
|
+
|
90
|
+
def blog_allows_feedback?
|
91
|
+
true
|
92
|
+
end
|
93
|
+
|
94
|
+
def akismet_options
|
95
|
+
{ type: self.class.to_s.downcase,
|
96
|
+
author: originator,
|
97
|
+
author_email: email,
|
98
|
+
author_url: url,
|
99
|
+
text: body }
|
100
|
+
end
|
101
|
+
|
102
|
+
def spam_fields
|
103
|
+
[:title, :body, :ip, :url]
|
104
|
+
end
|
105
|
+
|
106
|
+
def classify
|
107
|
+
return :ham if user_id
|
108
|
+
return :spam if blog.default_moderate_comments
|
109
|
+
return :ham unless blog.sp_global
|
110
|
+
|
111
|
+
# Yeah, three state logic is evil...
|
112
|
+
case sp_is_spam? || akismet_is_spam?
|
113
|
+
when nil then :spam
|
114
|
+
when true then :spam
|
115
|
+
when false then :ham
|
116
|
+
end
|
117
|
+
end
|
118
|
+
|
119
|
+
def sp_is_spam?(_options = {})
|
120
|
+
sp = SpamProtection.new(blog)
|
121
|
+
Timeout.timeout(30) do
|
122
|
+
spam_fields.any? do |field|
|
123
|
+
sp.is_spam?(send(field))
|
124
|
+
end
|
125
|
+
end
|
126
|
+
rescue Timeout::Error
|
127
|
+
nil
|
128
|
+
end
|
129
|
+
|
130
|
+
def akismet_is_spam?(_options = {})
|
131
|
+
return false if akismet.nil?
|
132
|
+
|
133
|
+
begin
|
134
|
+
Timeout.timeout(60) do
|
135
|
+
akismet.comment_check(ip, user_agent, akismet_options)
|
136
|
+
end
|
137
|
+
rescue Timeout::Error
|
138
|
+
nil
|
139
|
+
end
|
140
|
+
end
|
141
|
+
|
142
|
+
def change_state!
|
143
|
+
result = ''
|
144
|
+
if state.spam? || state.presumed_spam?
|
145
|
+
mark_as_ham
|
146
|
+
result = 'ham'
|
147
|
+
else
|
148
|
+
mark_as_spam
|
149
|
+
result = 'spam'
|
150
|
+
end
|
151
|
+
save!
|
152
|
+
result
|
153
|
+
end
|
154
|
+
|
155
|
+
def mark_as_ham!
|
156
|
+
mark_as_ham
|
157
|
+
save!
|
158
|
+
end
|
159
|
+
|
160
|
+
def mark_as_spam!
|
161
|
+
mark_as_spam
|
162
|
+
save!
|
163
|
+
end
|
164
|
+
|
165
|
+
def report_as_spam
|
166
|
+
return if akismet.nil?
|
167
|
+
begin
|
168
|
+
Timeout.timeout(5) do
|
169
|
+
akismet.submit_spam(ip, user_agent, akismet_options)
|
170
|
+
end
|
171
|
+
rescue Timeout::Error
|
172
|
+
nil
|
173
|
+
end
|
174
|
+
end
|
175
|
+
|
176
|
+
def report_as_ham
|
177
|
+
return if akismet.nil?
|
178
|
+
begin
|
179
|
+
Timeout.timeout(5) do
|
180
|
+
akismet.ham(ip, user_agent, akismet_options)
|
181
|
+
end
|
182
|
+
rescue Timeout::Error
|
183
|
+
nil
|
184
|
+
end
|
185
|
+
end
|
186
|
+
|
187
|
+
def withdraw!
|
188
|
+
withdraw
|
189
|
+
save!
|
190
|
+
end
|
191
|
+
|
192
|
+
def confirm_classification!
|
193
|
+
confirm_classification
|
194
|
+
save
|
195
|
+
end
|
196
|
+
|
197
|
+
def feedback_not_closed
|
198
|
+
errors.add(:article_id, 'Comment are closed') if article.comments_closed?
|
199
|
+
end
|
200
|
+
|
201
|
+
delegate :blog, to: :article
|
202
|
+
|
203
|
+
private
|
204
|
+
|
205
|
+
@@akismet = nil
|
206
|
+
|
207
|
+
def akismet
|
208
|
+
@@akismet = akismet_client if @@akismet.nil?
|
209
|
+
@@akismet == false ? nil : @@akismet
|
210
|
+
end
|
211
|
+
|
212
|
+
def akismet_client
|
213
|
+
return false if blog.sp_akismet_key.blank?
|
214
|
+
client = Akismet::Client.new(blog.sp_akismet_key, blog.base_url)
|
215
|
+
begin
|
216
|
+
return client.verify_key ? client : false
|
217
|
+
rescue SocketError
|
218
|
+
nil
|
219
|
+
end
|
220
|
+
end
|
221
|
+
|
222
|
+
def blog_id
|
223
|
+
article.blog_id if article.present?
|
224
|
+
end
|
225
|
+
end
|
data/app/models/note.rb
ADDED
@@ -0,0 +1,144 @@
|
|
1
|
+
class Note < Content
|
2
|
+
require 'twitter'
|
3
|
+
require 'json'
|
4
|
+
require 'uri'
|
5
|
+
include PublifyGuid
|
6
|
+
include ConfigManager
|
7
|
+
|
8
|
+
serialize :settings, Hash
|
9
|
+
|
10
|
+
setting :twitter_id, :string, ''
|
11
|
+
setting :in_reply_to_status_id, :string, ''
|
12
|
+
setting :in_reply_to_protected, :boolean, false
|
13
|
+
setting :in_reply_to_message, :string, ''
|
14
|
+
|
15
|
+
validates :body, presence: true
|
16
|
+
validates :permalink, :guid, uniqueness: true
|
17
|
+
|
18
|
+
after_create :set_permalink, :shorten_url
|
19
|
+
before_create :create_guid
|
20
|
+
|
21
|
+
default_scope { order('published_at DESC') }
|
22
|
+
|
23
|
+
TWITTER_FTP_URL_LENGTH = 19
|
24
|
+
TWITTER_HTTP_URL_LENGTH = 20
|
25
|
+
TWITTER_HTTPS_URL_LENGTH = 21
|
26
|
+
TWITTER_LINK_LENGTH = 22
|
27
|
+
|
28
|
+
def set_permalink
|
29
|
+
self.permalink = "#{id}-#{body.to_permalink[0..79]}" if permalink.blank?
|
30
|
+
save
|
31
|
+
end
|
32
|
+
|
33
|
+
def categories
|
34
|
+
[]
|
35
|
+
end
|
36
|
+
|
37
|
+
def tags
|
38
|
+
[]
|
39
|
+
end
|
40
|
+
|
41
|
+
def html_preprocess(_field, html)
|
42
|
+
PublifyApp::Textfilter::Twitterfilter.filtertext(html)
|
43
|
+
end
|
44
|
+
|
45
|
+
def truncate(message, length)
|
46
|
+
if message[length + 1] == ' '
|
47
|
+
message[0..length]
|
48
|
+
else
|
49
|
+
message[0..(message[0..length].rindex(' ') - 1)]
|
50
|
+
end
|
51
|
+
end
|
52
|
+
|
53
|
+
def twitter_message
|
54
|
+
base_message = body.strip_html
|
55
|
+
if too_long?("#{base_message} (#{short_link})")
|
56
|
+
max_length = 140 - "... (#{redirect.from_url})".length - 1
|
57
|
+
"#{truncate(base_message, max_length)}... (#{redirect.from_url})"
|
58
|
+
else
|
59
|
+
"#{base_message} (#{short_link})"
|
60
|
+
end
|
61
|
+
end
|
62
|
+
|
63
|
+
# FIXME: This breaks if the user changes or deletes their handle.
|
64
|
+
def twitter_url
|
65
|
+
File.join('https://twitter.com', user.twitter, 'status', twitter_id)
|
66
|
+
end
|
67
|
+
|
68
|
+
def send_to_twitter
|
69
|
+
return false unless blog.has_twitter_configured?
|
70
|
+
return false unless user.has_twitter_configured?
|
71
|
+
|
72
|
+
twitter = Twitter::REST::Client.new do |config|
|
73
|
+
config.consumer_key = blog.twitter_consumer_key
|
74
|
+
config.consumer_secret = blog.twitter_consumer_secret
|
75
|
+
config.access_token = user.twitter_oauth_token
|
76
|
+
config.access_token_secret = user.twitter_oauth_token_secret
|
77
|
+
end
|
78
|
+
|
79
|
+
begin
|
80
|
+
options = {}
|
81
|
+
if in_reply_to_status_id && in_reply_to_status_id != ''
|
82
|
+
options = { in_reply_to_status_id: in_reply_to_status_id }
|
83
|
+
self.in_reply_to_message = twitter.status(in_reply_to_status_id).to_json
|
84
|
+
end
|
85
|
+
tweet = twitter.update(twitter_message, options)
|
86
|
+
self.twitter_id = tweet.attrs[:id_str]
|
87
|
+
save
|
88
|
+
user.update_twitter_profile_image(tweet.attrs[:user][:profile_image_url])
|
89
|
+
true
|
90
|
+
rescue StandardError => e
|
91
|
+
Rails.logger.error("Error while sending to twitter: #{e}")
|
92
|
+
errors.add(:message, e)
|
93
|
+
false
|
94
|
+
end
|
95
|
+
end
|
96
|
+
|
97
|
+
content_fields :body
|
98
|
+
|
99
|
+
def password_protected?
|
100
|
+
false
|
101
|
+
end
|
102
|
+
|
103
|
+
def access_by?(user)
|
104
|
+
user.admin? || user_id == user.id
|
105
|
+
end
|
106
|
+
|
107
|
+
def permalink_url(anchor = nil, only_path = false)
|
108
|
+
blog.url_for(
|
109
|
+
controller: '/notes',
|
110
|
+
action: 'show',
|
111
|
+
permalink: permalink,
|
112
|
+
anchor: anchor,
|
113
|
+
only_path: only_path
|
114
|
+
)
|
115
|
+
end
|
116
|
+
|
117
|
+
def short_link
|
118
|
+
path = redirect.from_path
|
119
|
+
"#{prefix} #{path}"
|
120
|
+
end
|
121
|
+
|
122
|
+
def prefix
|
123
|
+
blog.shortener_url.sub(/^https?\:\/\//, '')
|
124
|
+
end
|
125
|
+
|
126
|
+
private
|
127
|
+
|
128
|
+
def too_long?(message)
|
129
|
+
uris = URI.extract(message, %w(http https ftp))
|
130
|
+
uris << prefix
|
131
|
+
uris.each do |uri|
|
132
|
+
payload = case uri.split(':')[0]
|
133
|
+
when 'https'
|
134
|
+
'-' * TWITTER_HTTPS_URL_LENGTH
|
135
|
+
when 'ftp'
|
136
|
+
'-' * TWITTER_FTP_URL_LENGTH
|
137
|
+
else
|
138
|
+
'-' * TWITTER_HTTP_URL_LENGTH
|
139
|
+
end
|
140
|
+
message = message.gsub(uri, payload)
|
141
|
+
end
|
142
|
+
message.length > 140
|
143
|
+
end
|
144
|
+
end
|
data/app/models/page.rb
ADDED
@@ -0,0 +1,36 @@
|
|
1
|
+
class Page < Content
|
2
|
+
validates :title, :body, presence: true
|
3
|
+
validates :name, uniqueness: true
|
4
|
+
|
5
|
+
include ConfigManager
|
6
|
+
|
7
|
+
serialize :settings, Hash
|
8
|
+
setting :password, :string, ''
|
9
|
+
|
10
|
+
before_save :set_permalink
|
11
|
+
after_save :shorten_url
|
12
|
+
|
13
|
+
def set_permalink
|
14
|
+
self.name = title.to_permalink if name.blank?
|
15
|
+
end
|
16
|
+
|
17
|
+
content_fields :body
|
18
|
+
|
19
|
+
def self.default_order
|
20
|
+
'name ASC'
|
21
|
+
end
|
22
|
+
|
23
|
+
def self.search_with(search_hash)
|
24
|
+
super(search_hash).order('title ASC')
|
25
|
+
end
|
26
|
+
|
27
|
+
def permalink_url(anchor = nil, only_path = false)
|
28
|
+
blog.url_for(
|
29
|
+
controller: '/articles',
|
30
|
+
action: 'view_page',
|
31
|
+
name: name,
|
32
|
+
anchor: anchor,
|
33
|
+
only_path: only_path
|
34
|
+
)
|
35
|
+
end
|
36
|
+
end
|
@@ -0,0 +1,33 @@
|
|
1
|
+
# FIXME: This class is not a model anymore. Move elsewhere?
|
2
|
+
class PageCache
|
3
|
+
def self.logger
|
4
|
+
::Rails.logger
|
5
|
+
end
|
6
|
+
|
7
|
+
def logger
|
8
|
+
::Rails.logger
|
9
|
+
end
|
10
|
+
|
11
|
+
def self.public_path
|
12
|
+
ActionController::Base.page_cache_directory
|
13
|
+
end
|
14
|
+
|
15
|
+
# Delete all file save in path_cache by page_cache system
|
16
|
+
def self.sweep_all
|
17
|
+
zap_pages(%w(*))
|
18
|
+
end
|
19
|
+
|
20
|
+
def self.sweep_theme_cache
|
21
|
+
zap_pages(%w(images/theme/* stylesheets/theme/* javascripts/theme/*))
|
22
|
+
end
|
23
|
+
|
24
|
+
def self.zap_pages(paths)
|
25
|
+
# Ensure no one is going to wipe his own blog public directory
|
26
|
+
# It happened once on a release and was no fun at all
|
27
|
+
return if public_path == "#{::Rails.root}/public"
|
28
|
+
paths.each do |v|
|
29
|
+
FileUtils.rm_rf(Dir.glob(public_path + "/#{v}"))
|
30
|
+
end
|
31
|
+
true
|
32
|
+
end
|
33
|
+
end
|
data/app/models/ping.rb
ADDED
@@ -0,0 +1,116 @@
|
|
1
|
+
require 'rexml/document'
|
2
|
+
require 'xmlrpc/client'
|
3
|
+
|
4
|
+
class Ping < ActiveRecord::Base
|
5
|
+
belongs_to :article
|
6
|
+
|
7
|
+
class Pinger
|
8
|
+
attr_accessor :article
|
9
|
+
attr_accessor :blog
|
10
|
+
|
11
|
+
def send_pingback_or_trackback
|
12
|
+
@response = Net::HTTP.get_response(URI.parse(ping.url))
|
13
|
+
send_pingback or send_trackback
|
14
|
+
rescue Timeout::Error
|
15
|
+
Rails.logger.info 'Sending pingback or trackback timed out'
|
16
|
+
return
|
17
|
+
rescue => err
|
18
|
+
Rails.logger.info "Sending pingback or trackback failed with error: #{err}"
|
19
|
+
end
|
20
|
+
|
21
|
+
def pingback_url
|
22
|
+
if response['X-Pingback']
|
23
|
+
response['X-Pingback']
|
24
|
+
elsif response.body =~ /<link rel="pingback" href="([^"]+)" ?\/?>/
|
25
|
+
Regexp.last_match[1]
|
26
|
+
end
|
27
|
+
end
|
28
|
+
|
29
|
+
attr_reader :origin_url
|
30
|
+
|
31
|
+
attr_reader :response
|
32
|
+
|
33
|
+
attr_reader :ping
|
34
|
+
|
35
|
+
def send_xml_rpc(*args)
|
36
|
+
ping.send(:send_xml_rpc, *args)
|
37
|
+
end
|
38
|
+
|
39
|
+
def trackback_url
|
40
|
+
rdfs = response.body.scan(/<rdf:RDF.*?<\/rdf:RDF>/m)
|
41
|
+
rdfs.each do |rdf|
|
42
|
+
xml = REXML::Document.new(rdf)
|
43
|
+
xml.elements.each('//rdf:Description') do |desc|
|
44
|
+
if rdfs.size == 1 || desc.attributes['dc:identifier'] == ping.url
|
45
|
+
return desc.attributes['trackback:ping']
|
46
|
+
end
|
47
|
+
end
|
48
|
+
end
|
49
|
+
# Didn't find a trackback url, so fall back to the url itself.
|
50
|
+
@ping.url
|
51
|
+
end
|
52
|
+
|
53
|
+
def send_pingback
|
54
|
+
if pingback_url
|
55
|
+
send_xml_rpc(pingback_url, 'pingback.ping', origin_url, ping.url)
|
56
|
+
true
|
57
|
+
else
|
58
|
+
false
|
59
|
+
end
|
60
|
+
end
|
61
|
+
|
62
|
+
def send_trackback
|
63
|
+
do_send_trackback(trackback_url, origin_url)
|
64
|
+
end
|
65
|
+
|
66
|
+
def do_send_trackback(trackback_url, origin_url)
|
67
|
+
trackback_uri = URI.parse(trackback_url)
|
68
|
+
|
69
|
+
post = "title=#{CGI.escape(article.title)}"
|
70
|
+
post << "&excerpt=#{CGI.escape(article.html(:body).strip_html[0..254])}"
|
71
|
+
post << "&url=#{origin_url}"
|
72
|
+
post << "&blog_name=#{CGI.escape(blog.blog_name)}"
|
73
|
+
|
74
|
+
path = trackback_uri.path
|
75
|
+
path += "?#{trackback_uri.query}" if trackback_uri.query
|
76
|
+
|
77
|
+
Net::HTTP.start(trackback_uri.host, trackback_uri.port) do |http|
|
78
|
+
http.post(path, post, 'Content-type' => 'application/x-www-form-urlencoded; charset=utf-8')
|
79
|
+
end
|
80
|
+
end
|
81
|
+
|
82
|
+
private
|
83
|
+
|
84
|
+
def initialize(origin_url, ping)
|
85
|
+
@origin_url = origin_url
|
86
|
+
@ping = ping
|
87
|
+
# Add this call to text filter cause of a strange thing around text_filter. Need to clean text_filter usage !
|
88
|
+
ping.article.default_text_filter
|
89
|
+
ping.article.text_filter
|
90
|
+
# Make sure these are fetched now for thread safety purposes.
|
91
|
+
self.article = ping.article
|
92
|
+
self.blog = article.blog
|
93
|
+
end
|
94
|
+
end
|
95
|
+
|
96
|
+
def send_pingback_or_trackback(origin_url)
|
97
|
+
t = Thread.start(Pinger.new(origin_url, self), &:send_pingback_or_trackback)
|
98
|
+
t
|
99
|
+
end
|
100
|
+
|
101
|
+
def send_weblogupdatesping(server_url, origin_url)
|
102
|
+
t = Thread.start(article.blog.blog_name) do |blog_name|
|
103
|
+
send_xml_rpc(url, 'weblogUpdates.ping', blog_name, server_url, origin_url)
|
104
|
+
end
|
105
|
+
t
|
106
|
+
end
|
107
|
+
|
108
|
+
protected
|
109
|
+
|
110
|
+
def send_xml_rpc(xml_rpc_url, name, *args)
|
111
|
+
server = XMLRPC::Client.new2(URI.parse(xml_rpc_url).to_s)
|
112
|
+
server.call(name, *args)
|
113
|
+
rescue => e
|
114
|
+
logger.error(e)
|
115
|
+
end
|
116
|
+
end
|
@@ -0,0 +1,15 @@
|
|
1
|
+
# coding: utf-8
|
2
|
+
class PostType < ActiveRecord::Base
|
3
|
+
validates :name, uniqueness: true
|
4
|
+
validates :name, presence: true
|
5
|
+
validate :name_is_not_read
|
6
|
+
before_save :sanitize_title
|
7
|
+
|
8
|
+
def name_is_not_read
|
9
|
+
errors.add(:name, I18n.t('errors.article_type_already_exist')) if name == 'read'
|
10
|
+
end
|
11
|
+
|
12
|
+
def sanitize_title
|
13
|
+
self.permalink = name.to_permalink
|
14
|
+
end
|
15
|
+
end
|
@@ -0,0 +1,45 @@
|
|
1
|
+
class Redirect < ActiveRecord::Base
|
2
|
+
belongs_to :contents
|
3
|
+
belongs_to :blog
|
4
|
+
|
5
|
+
validates :from_path, uniqueness: true
|
6
|
+
validates :to_path, presence: true
|
7
|
+
validates :blog, presence: true
|
8
|
+
|
9
|
+
def full_to_path
|
10
|
+
path = to_path
|
11
|
+
return path if path =~ /^(https?):\/\/([^\/]*)(.*)/
|
12
|
+
url_root = blog.root_path
|
13
|
+
path = File.join(url_root, path) unless url_root.nil? || path[0, url_root.length] == url_root
|
14
|
+
path
|
15
|
+
end
|
16
|
+
|
17
|
+
def shorten
|
18
|
+
if (temp_token = random_token) && self.class.find_by(from_path: temp_token).nil?
|
19
|
+
temp_token
|
20
|
+
else
|
21
|
+
shorten
|
22
|
+
end
|
23
|
+
end
|
24
|
+
|
25
|
+
def to_url
|
26
|
+
raise 'Use #from_url'
|
27
|
+
end
|
28
|
+
|
29
|
+
def from_url
|
30
|
+
File.join(blog.shortener_url, from_path)
|
31
|
+
end
|
32
|
+
|
33
|
+
private
|
34
|
+
|
35
|
+
def random_token
|
36
|
+
characters = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz1234567890'
|
37
|
+
temp_token = ''
|
38
|
+
srand
|
39
|
+
6.times do
|
40
|
+
pos = rand(characters.length)
|
41
|
+
temp_token += characters[pos..pos]
|
42
|
+
end
|
43
|
+
temp_token
|
44
|
+
end
|
45
|
+
end
|
@@ -0,0 +1,28 @@
|
|
1
|
+
require 'carrierwave'
|
2
|
+
require 'carrierwave/orm/activerecord'
|
3
|
+
|
4
|
+
class Resource < ActiveRecord::Base
|
5
|
+
belongs_to :blog
|
6
|
+
belongs_to :article
|
7
|
+
|
8
|
+
mount_uploader :upload, ResourceUploader
|
9
|
+
validate :image_mime_type_consistent
|
10
|
+
validates :upload, presence: true
|
11
|
+
|
12
|
+
scope :without_images, -> { where("mime NOT LIKE '%image%'") }
|
13
|
+
scope :images, -> { where("mime LIKE '%image%'") }
|
14
|
+
scope :by_filename, -> { order('upload') }
|
15
|
+
scope :by_created_at, -> { order('created_at DESC') }
|
16
|
+
|
17
|
+
scope :without_images_by_filename, -> { without_images.by_filename }
|
18
|
+
scope :images_by_created_at, -> { images.by_created_at }
|
19
|
+
|
20
|
+
private
|
21
|
+
|
22
|
+
def image_mime_type_consistent
|
23
|
+
if upload.content_type =~ %r{^image/}
|
24
|
+
expected_type = upload.file.send :mime_magic_content_type
|
25
|
+
errors.add(:upload, 'Has MIME type mismatch') unless upload.content_type == expected_type
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|