2019-05-02 18:17:27 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2018-06-26 18:02:03 -04:00
|
|
|
require_relative 'base'
|
|
|
|
require 'tiny_tds'
|
|
|
|
|
2018-08-13 09:27:51 -04:00
|
|
|
# Import script for Telligent communities
|
|
|
|
#
|
2019-05-30 11:37:51 -04:00
|
|
|
# It's really hard to find all attachments, but the script tries to do it anyway.
|
2020-03-11 20:39:16 -04:00
|
|
|
#
|
|
|
|
# You can supply a JSON file if you need to map and ignore categories during the import
|
|
|
|
# by providing the path to the file in the `CATEGORY_MAPPING` environment variable.
|
|
|
|
# You can also add tags to remapped categories and remap multiple old forums into one
|
|
|
|
# category. Here's an example of such a `mapping.json` file:
|
|
|
|
#
|
|
|
|
# {
|
|
|
|
# "ignored_forum_ids": [41, 360, 378],
|
|
|
|
#
|
|
|
|
# "mapping": [
|
|
|
|
# {
|
|
|
|
# "category": ["New Category 1"],
|
|
|
|
# "forums": [
|
|
|
|
# { "id": 348, "tag": "some_tag" },
|
|
|
|
# { "id": 347, "tag": "another_tag" }
|
|
|
|
# ]
|
|
|
|
# },
|
|
|
|
# {
|
|
|
|
# "category": ["New Category 2"],
|
|
|
|
# "forums": [
|
|
|
|
# { "id": 9 }
|
|
|
|
# ]
|
|
|
|
# },
|
|
|
|
# {
|
|
|
|
# "category": ["Nested", "Category"],
|
|
|
|
# "forums": [
|
|
|
|
# { "id": 322 }
|
|
|
|
# ]
|
|
|
|
# }
|
|
|
|
# ]
|
|
|
|
# }
|
2018-08-13 09:27:51 -04:00
|
|
|
|
2018-06-26 18:02:03 -04:00
|
|
|
class ImportScripts::Telligent < ImportScripts::Base
|
|
|
|
BATCH_SIZE ||= 1000
|
|
|
|
LOCAL_AVATAR_REGEX ||= /\A~\/.*(?<directory>communityserver-components-(?:selectable)?avatars)\/(?<path>[^\/]+)\/(?<filename>.+)/i
|
|
|
|
REMOTE_AVATAR_REGEX ||= /\Ahttps?:\/\//i
|
2020-03-11 20:39:16 -04:00
|
|
|
EMBEDDED_ATTACHMENT_REGEX ||= /<a href="\/cfs-file(?:\.ashx)?\/__key\/(?<directory>[^\/]+)\/(?<path>[^\/]+)\/(?<filename1>.+?)".*?>(?<filename2>.*?)<\/a>/i
|
2018-06-26 18:02:03 -04:00
|
|
|
|
|
|
|
CATEGORY_LINK_NORMALIZATION = '/.*?(f\/\d+)$/\1'
|
|
|
|
TOPIC_LINK_NORMALIZATION = '/.*?(f\/\d+\/t\/\d+)$/\1'
|
|
|
|
|
|
|
|
def initialize
|
|
|
|
super()
|
|
|
|
|
|
|
|
@client = TinyTds::Client.new(
|
|
|
|
host: ENV["DB_HOST"],
|
|
|
|
username: ENV["DB_USERNAME"],
|
|
|
|
password: ENV["DB_PASSWORD"],
|
2019-05-30 11:37:51 -04:00
|
|
|
database: ENV["DB_NAME"],
|
|
|
|
timeout: 60 # the user query is very slow
|
2018-06-26 18:02:03 -04:00
|
|
|
)
|
2020-03-11 20:39:16 -04:00
|
|
|
|
|
|
|
SiteSetting.tagging_enabled = true
|
2018-06-26 18:02:03 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def execute
|
|
|
|
add_permalink_normalizations
|
|
|
|
import_categories
|
2020-03-11 20:39:16 -04:00
|
|
|
import_users
|
2018-06-26 18:02:03 -04:00
|
|
|
import_topics
|
|
|
|
import_posts
|
|
|
|
mark_topics_as_solved
|
|
|
|
end
|
|
|
|
|
|
|
|
def import_users
|
|
|
|
puts "", "Importing users..."
|
|
|
|
|
|
|
|
user_conditions = <<~SQL
|
|
|
|
(
|
|
|
|
EXISTS(SELECT 1
|
|
|
|
FROM te_Forum_Threads t
|
|
|
|
WHERE t.UserId = u.UserID) OR
|
|
|
|
EXISTS(SELECT 1
|
|
|
|
FROM te_Forum_ThreadReplies r
|
|
|
|
WHERE r.UserId = u.UserID)
|
|
|
|
)
|
|
|
|
SQL
|
|
|
|
|
|
|
|
last_user_id = -1
|
|
|
|
total_count = count(<<~SQL)
|
|
|
|
SELECT COUNT(1) AS count
|
2019-05-30 11:37:51 -04:00
|
|
|
FROM cs_Users u
|
2018-06-26 18:02:03 -04:00
|
|
|
WHERE #{user_conditions}
|
|
|
|
SQL
|
2019-05-30 11:37:51 -04:00
|
|
|
import_count = 0
|
2018-06-26 18:02:03 -04:00
|
|
|
|
2019-05-30 11:37:51 -04:00
|
|
|
loop do
|
2018-06-26 18:02:03 -04:00
|
|
|
rows = query(<<~SQL)
|
2019-05-30 11:37:51 -04:00
|
|
|
SELECT *
|
|
|
|
FROM (
|
|
|
|
SELECT TOP #{BATCH_SIZE}
|
2020-03-11 20:39:16 -04:00
|
|
|
u.UserID, u.Email, u.UserName,u.CreateDate, p.PropertyName, p.PropertyValue
|
2019-05-30 11:37:51 -04:00
|
|
|
FROM cs_Users u
|
2020-03-11 20:39:16 -04:00
|
|
|
LEFT OUTER JOIN (
|
|
|
|
SELECT NULL AS UserID, ap.UserId AS MembershipID, x.PropertyName, x.PropertyValue
|
|
|
|
FROM aspnet_Profile ap
|
|
|
|
CROSS APPLY dbo.GetProperties(ap.PropertyNames, ap.PropertyValuesString) x
|
|
|
|
WHERE ap.PropertyNames NOT LIKE '%:-1%' AND
|
|
|
|
x.PropertyName IN ('bio', 'commonName', 'location', 'webAddress')
|
|
|
|
UNION
|
|
|
|
SELECT up.UserID, NULL AS MembershipID, x.PropertyName, CAST(x.PropertyValue AS NVARCHAR) AS PropertyValue
|
|
|
|
FROM cs_UserProfile up
|
|
|
|
CROSS APPLY dbo.GetProperties(up.PropertyNames, up.PropertyValues) x
|
|
|
|
WHERE up.PropertyNames NOT LIKE '%:-1%' AND
|
2019-05-30 11:37:51 -04:00
|
|
|
x.PropertyName IN ('avatarUrl', 'BannedUntil', 'UserBanReason')
|
2020-03-11 20:39:16 -04:00
|
|
|
) p ON p.UserID = u.UserID OR p.MembershipID = u.MembershipID
|
2018-06-26 18:02:03 -04:00
|
|
|
WHERE u.UserID > #{last_user_id} AND #{user_conditions}
|
2019-05-30 11:37:51 -04:00
|
|
|
ORDER BY u.UserID
|
|
|
|
) x
|
2020-03-11 20:39:16 -04:00
|
|
|
PIVOT (
|
|
|
|
MAX(PropertyValue)
|
|
|
|
FOR PropertyName
|
|
|
|
IN (bio, commonName, location, webAddress, avatarUrl, BannedUntil, UserBanReason)
|
|
|
|
) Y
|
2018-06-26 18:02:03 -04:00
|
|
|
ORDER BY UserID
|
|
|
|
SQL
|
|
|
|
|
|
|
|
break if rows.blank?
|
|
|
|
last_user_id = rows[-1]["UserID"]
|
|
|
|
|
2019-05-30 11:37:51 -04:00
|
|
|
if all_records_exist?(:users, rows.map { |row| row["UserID"] })
|
|
|
|
import_count += rows.size
|
|
|
|
next
|
|
|
|
end
|
|
|
|
|
|
|
|
create_users(rows, total: total_count, offset: import_count) do |row|
|
2018-06-26 18:02:03 -04:00
|
|
|
{
|
|
|
|
id: row["UserID"],
|
|
|
|
email: row["Email"],
|
|
|
|
username: row["UserName"],
|
2019-05-30 11:37:51 -04:00
|
|
|
name: row["commonName"],
|
2018-06-26 18:02:03 -04:00
|
|
|
created_at: row["CreateDate"],
|
|
|
|
bio_raw: html_to_markdown(row["bio"]),
|
2019-05-30 11:37:51 -04:00
|
|
|
location: row["location"],
|
2018-06-26 18:02:03 -04:00
|
|
|
website: row["webAddress"],
|
|
|
|
post_create_action: proc do |user|
|
|
|
|
import_avatar(user, row["avatarUrl"])
|
|
|
|
suspend_user(user, row["BannedUntil"], row["UserBanReason"])
|
|
|
|
end
|
|
|
|
}
|
|
|
|
end
|
2019-05-30 11:37:51 -04:00
|
|
|
|
|
|
|
import_count += rows.size
|
2018-06-26 18:02:03 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# TODO move into base importer (create_user) and use consistent error handling
|
|
|
|
def import_avatar(user, avatar_url)
|
2019-05-30 11:37:51 -04:00
|
|
|
return if ENV["FILE_BASE_DIR"].blank? || avatar_url.blank? || avatar_url.include?("anonymous")
|
2018-06-26 18:02:03 -04:00
|
|
|
|
|
|
|
if match_data = avatar_url.match(LOCAL_AVATAR_REGEX)
|
|
|
|
avatar_path = File.join(ENV["FILE_BASE_DIR"],
|
|
|
|
match_data[:directory].gsub("-", "."),
|
|
|
|
match_data[:path].split("-"),
|
|
|
|
match_data[:filename])
|
|
|
|
|
2020-03-11 20:39:16 -04:00
|
|
|
if File.file?(avatar_path)
|
2018-06-26 18:02:03 -04:00
|
|
|
@uploader.create_avatar(user, avatar_path)
|
|
|
|
else
|
|
|
|
STDERR.puts "Could not find avatar: #{avatar_path}"
|
|
|
|
end
|
|
|
|
elsif avatar_url.match?(REMOTE_AVATAR_REGEX)
|
|
|
|
UserAvatar.import_url_for_user(avatar_url, user) rescue nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def suspend_user(user, banned_until, ban_reason)
|
|
|
|
return if banned_until.blank?
|
|
|
|
|
|
|
|
if banned_until = DateTime.parse(banned_until) > DateTime.now
|
|
|
|
user.suspended_till = banned_until
|
|
|
|
user.suspended_at = DateTime.now
|
|
|
|
user.save!
|
|
|
|
|
|
|
|
StaffActionLogger.new(Discourse.system_user).log_user_suspend(user, ban_reason)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def import_categories
|
2020-03-11 20:39:16 -04:00
|
|
|
if ENV['CATEGORY_MAPPING']
|
|
|
|
import_mapped_forums_as_categories
|
|
|
|
else
|
|
|
|
import_groups_and_forums_as_categories
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def import_mapped_forums_as_categories
|
|
|
|
puts "", "Importing categories..."
|
|
|
|
|
|
|
|
json = JSON.parse(File.read(ENV['CATEGORY_MAPPING']))
|
|
|
|
|
|
|
|
categories = []
|
|
|
|
@forum_ids_to_tags = {}
|
|
|
|
@ignored_forum_ids = json["ignored_forum_ids"]
|
|
|
|
|
|
|
|
json["mapping"].each do |m|
|
|
|
|
parent_id = nil
|
|
|
|
last_index = m["category"].size - 1
|
|
|
|
forum_ids = []
|
|
|
|
|
|
|
|
m["forums"].each do |f|
|
|
|
|
forum_ids << f["id"]
|
|
|
|
@forum_ids_to_tags[f["id"]] = f["tag"] if f["tag"].present?
|
|
|
|
end
|
|
|
|
|
|
|
|
m["category"].each_with_index do |name, index|
|
|
|
|
id = Digest::MD5.hexdigest(name)
|
|
|
|
categories << {
|
|
|
|
id: id,
|
|
|
|
name: name,
|
|
|
|
parent_id: parent_id,
|
|
|
|
forum_ids: index == last_index ? forum_ids : nil
|
|
|
|
}
|
|
|
|
parent_id = id
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
create_categories(categories) do |c|
|
|
|
|
if category_id = category_id_from_imported_category_id(c[:id])
|
|
|
|
map_forum_ids(category_id, c[:forum_ids])
|
|
|
|
nil
|
|
|
|
else
|
|
|
|
{
|
|
|
|
id: c[:id],
|
|
|
|
name: c[:name],
|
|
|
|
parent_category_id: category_id_from_imported_category_id(c[:parent_id]),
|
|
|
|
post_create_action: proc do |category|
|
|
|
|
map_forum_ids(category.id, c[:forum_ids])
|
|
|
|
end
|
|
|
|
}
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def map_forum_ids(category_id, forum_ids)
|
|
|
|
return if forum_ids.blank?
|
|
|
|
|
|
|
|
forum_ids.each do |id|
|
|
|
|
url = "f/#{id}"
|
|
|
|
Permalink.create(url: url, category_id: category_id) unless Permalink.exists?(url: url)
|
|
|
|
add_category(id, Category.find_by_id(category_id))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def import_groups_and_forums_as_categories
|
2018-06-26 18:02:03 -04:00
|
|
|
puts "", "Importing parent categories..."
|
|
|
|
parent_categories = query(<<~SQL)
|
2020-03-11 20:39:16 -04:00
|
|
|
SELECT GroupID, Name, HtmlDescription, DateCreated, SortOrder
|
2018-06-26 18:02:03 -04:00
|
|
|
FROM cs_Groups g
|
|
|
|
WHERE (SELECT COUNT(1)
|
|
|
|
FROM te_Forum_Forums f
|
|
|
|
WHERE f.GroupId = g.GroupID) > 1
|
|
|
|
ORDER BY SortOrder, Name
|
|
|
|
SQL
|
|
|
|
|
|
|
|
create_categories(parent_categories) do |row|
|
|
|
|
{
|
|
|
|
id: "G#{row['GroupID']}",
|
|
|
|
name: clean_category_name(row["Name"]),
|
|
|
|
description: html_to_markdown(row["HtmlDescription"]),
|
|
|
|
position: row["SortOrder"]
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
puts "", "Importing child categories..."
|
|
|
|
child_categories = query(<<~SQL)
|
2020-03-11 20:39:16 -04:00
|
|
|
SELECT ForumId, GroupId, Name, Description, DateCreated, SortOrder
|
2018-06-26 18:02:03 -04:00
|
|
|
FROM te_Forum_Forums
|
|
|
|
ORDER BY GroupId, SortOrder, Name
|
|
|
|
SQL
|
|
|
|
|
|
|
|
create_categories(child_categories) do |row|
|
|
|
|
parent_category_id = parent_category_id_for(row)
|
|
|
|
|
2019-05-30 11:37:51 -04:00
|
|
|
if category_id = replace_with_category_id(child_categories, parent_category_id)
|
2018-06-26 18:02:03 -04:00
|
|
|
add_category(row['ForumId'], Category.find_by_id(category_id))
|
2018-08-13 09:27:51 -04:00
|
|
|
url = "f/#{row['ForumId']}"
|
|
|
|
Permalink.create(url: url, category_id: category_id) unless Permalink.exists?(url: url)
|
2018-06-26 18:02:03 -04:00
|
|
|
nil
|
|
|
|
else
|
|
|
|
{
|
|
|
|
id: row['ForumId'],
|
|
|
|
parent_category_id: parent_category_id,
|
|
|
|
name: clean_category_name(row["Name"]),
|
|
|
|
description: html_to_markdown(row["Description"]),
|
2020-03-11 20:39:16 -04:00
|
|
|
position: row["SortOrder"],
|
|
|
|
post_create_action: proc do |category|
|
|
|
|
url = "f/#{row['ForumId']}"
|
|
|
|
Permalink.create(url: url, category_id: category.id) unless Permalink.exists?(url: url)
|
|
|
|
end
|
2018-06-26 18:02:03 -04:00
|
|
|
}
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def parent_category_id_for(row)
|
2019-05-30 11:37:51 -04:00
|
|
|
category_id_from_imported_category_id("G#{row['GroupId']}") if row.key?("GroupId")
|
2018-06-26 18:02:03 -04:00
|
|
|
end
|
|
|
|
|
2019-05-30 11:37:51 -04:00
|
|
|
def replace_with_category_id(child_categories, parent_category_id)
|
|
|
|
parent_category_id if only_child?(child_categories, parent_category_id)
|
2018-06-26 18:02:03 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def only_child?(child_categories, parent_category_id)
|
|
|
|
count = 0
|
|
|
|
|
|
|
|
child_categories.each do |row|
|
|
|
|
count += 1 if parent_category_id_for(row) == parent_category_id
|
|
|
|
end
|
|
|
|
|
|
|
|
count == 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def clean_category_name(name)
|
|
|
|
CGI.unescapeHTML(name)
|
|
|
|
.strip
|
|
|
|
end
|
|
|
|
|
|
|
|
def import_topics
|
|
|
|
puts "", "Importing topics..."
|
|
|
|
|
|
|
|
last_topic_id = -1
|
2020-03-11 20:39:16 -04:00
|
|
|
total_count = count("SELECT COUNT(1) AS count FROM te_Forum_Threads t WHERE #{ignored_forum_sql_condition}")
|
2018-06-26 18:02:03 -04:00
|
|
|
|
|
|
|
batches do |offset|
|
|
|
|
rows = query(<<~SQL)
|
|
|
|
SELECT TOP #{BATCH_SIZE}
|
2020-03-11 20:39:16 -04:00
|
|
|
t.ThreadId, t.ForumId, t.UserId, t.TotalViews,
|
2018-06-26 18:02:03 -04:00
|
|
|
t.Subject, t.Body, t.DateCreated, t.IsLocked, t.StickyDate,
|
2020-03-11 20:39:16 -04:00
|
|
|
a.ApplicationTypeId, a.ApplicationId, a.ApplicationContentTypeId, a.ContentId, a.FileName, a.IsRemote
|
2018-06-26 18:02:03 -04:00
|
|
|
FROM te_Forum_Threads t
|
|
|
|
LEFT JOIN te_Attachments a
|
|
|
|
ON (a.ApplicationId = t.ForumId AND a.ApplicationTypeId = 0 AND a.ContentId = t.ThreadId AND
|
|
|
|
a.ApplicationContentTypeId = 0)
|
2020-03-11 20:39:16 -04:00
|
|
|
WHERE t.ThreadId > #{last_topic_id} AND #{ignored_forum_sql_condition}
|
2018-06-26 18:02:03 -04:00
|
|
|
ORDER BY t.ThreadId
|
|
|
|
SQL
|
|
|
|
|
|
|
|
break if rows.blank?
|
|
|
|
last_topic_id = rows[-1]["ThreadId"]
|
|
|
|
next if all_records_exist?(:post, rows.map { |row| import_topic_id(row["ThreadId"]) })
|
|
|
|
|
|
|
|
create_posts(rows, total: total_count, offset: offset) do |row|
|
|
|
|
user_id = user_id_from_imported_user_id(row["UserId"]) || Discourse::SYSTEM_USER_ID
|
|
|
|
|
|
|
|
post = {
|
|
|
|
id: import_topic_id(row["ThreadId"]),
|
|
|
|
title: CGI.unescapeHTML(row["Subject"]),
|
2020-03-11 20:39:16 -04:00
|
|
|
raw: raw_with_attachment(row, user_id, :topic),
|
2018-06-26 18:02:03 -04:00
|
|
|
category: category_id_from_imported_category_id(row["ForumId"]),
|
|
|
|
user_id: user_id,
|
|
|
|
created_at: row["DateCreated"],
|
|
|
|
closed: row["IsLocked"],
|
2020-03-11 20:39:16 -04:00
|
|
|
views: row["TotalViews"],
|
2018-09-03 22:16:21 -04:00
|
|
|
post_create_action: proc do |action_post|
|
|
|
|
topic = action_post.topic
|
2018-06-26 18:02:03 -04:00
|
|
|
Jobs.enqueue_at(topic.pinned_until, :unpin_topic, topic_id: topic.id) if topic.pinned_until
|
2018-08-13 09:27:51 -04:00
|
|
|
url = "f/#{row['ForumId']}/t/#{row['ThreadId']}"
|
|
|
|
Permalink.create(url: url, topic_id: topic.id) unless Permalink.exists?(url: url)
|
2018-06-26 18:02:03 -04:00
|
|
|
end
|
|
|
|
}
|
|
|
|
|
|
|
|
if row["StickyDate"] > Time.now
|
|
|
|
post[:pinned_until] = row["StickyDate"]
|
|
|
|
post[:pinned_at] = row["DateCreated"]
|
|
|
|
end
|
|
|
|
|
|
|
|
post
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def import_topic_id(topic_id)
|
|
|
|
"T#{topic_id}"
|
|
|
|
end
|
|
|
|
|
2020-03-11 20:39:16 -04:00
|
|
|
def ignored_forum_sql_condition
|
|
|
|
@ignored_forum_sql_condition ||= @ignored_forum_ids.present? \
|
|
|
|
? "t.ForumId NOT IN (#{@ignored_forum_ids.join(',')})" \
|
|
|
|
: "1 = 1"
|
|
|
|
end
|
|
|
|
|
2018-06-26 18:02:03 -04:00
|
|
|
def import_posts
|
|
|
|
puts "", "Importing posts..."
|
|
|
|
|
|
|
|
last_post_id = -1
|
2020-03-11 20:39:16 -04:00
|
|
|
total_count = count(<<~SQL)
|
|
|
|
SELECT COUNT(1) AS count
|
|
|
|
FROM te_Forum_ThreadReplies tr
|
|
|
|
JOIN te_Forum_Threads t ON (tr.ThreadId = t.ThreadId)
|
|
|
|
WHERE #{ignored_forum_sql_condition}
|
|
|
|
SQL
|
2018-06-26 18:02:03 -04:00
|
|
|
|
|
|
|
batches do |offset|
|
|
|
|
rows = query(<<~SQL)
|
|
|
|
SELECT TOP #{BATCH_SIZE}
|
2020-03-11 20:39:16 -04:00
|
|
|
tr.ThreadReplyId, tr.ThreadId, tr.UserId, pr.ThreadReplyId AS ParentReplyId,
|
2018-06-26 18:02:03 -04:00
|
|
|
tr.Body, tr.ThreadReplyDate,
|
|
|
|
CONVERT(BIT,
|
|
|
|
CASE WHEN tr.AnswerVerifiedUtcDate IS NOT NULL AND NOT EXISTS(
|
|
|
|
SELECT 1
|
|
|
|
FROM te_Forum_ThreadReplies x
|
|
|
|
WHERE
|
|
|
|
x.ThreadId = tr.ThreadId AND x.ThreadReplyId < tr.ThreadReplyId AND x.AnswerVerifiedUtcDate IS NOT NULL
|
|
|
|
)
|
|
|
|
THEN 1
|
|
|
|
ELSE 0 END) AS IsFirstVerifiedAnswer,
|
2020-03-11 20:39:16 -04:00
|
|
|
a.ApplicationTypeId, a.ApplicationId, a.ApplicationContentTypeId, a.ContentId, a.FileName, a.IsRemote
|
2018-06-26 18:02:03 -04:00
|
|
|
FROM te_Forum_ThreadReplies tr
|
|
|
|
JOIN te_Forum_Threads t ON (tr.ThreadId = t.ThreadId)
|
2020-03-11 20:39:16 -04:00
|
|
|
LEFT JOIN te_Forum_ThreadReplies pr ON (tr.ParentReplyId = pr.ThreadReplyId AND tr.ParentReplyId < tr.ThreadReplyId AND tr.ThreadId = pr.ThreadId)
|
2018-06-26 18:02:03 -04:00
|
|
|
LEFT JOIN te_Attachments a
|
|
|
|
ON (a.ApplicationId = t.ForumId AND a.ApplicationTypeId = 0 AND a.ContentId = tr.ThreadReplyId AND
|
|
|
|
a.ApplicationContentTypeId = 1)
|
2020-03-11 20:39:16 -04:00
|
|
|
WHERE tr.ThreadReplyId > #{last_post_id} AND #{ignored_forum_sql_condition}
|
2018-06-26 18:02:03 -04:00
|
|
|
ORDER BY tr.ThreadReplyId
|
|
|
|
SQL
|
|
|
|
|
|
|
|
break if rows.blank?
|
|
|
|
last_post_id = rows[-1]["ThreadReplyId"]
|
|
|
|
next if all_records_exist?(:post, rows.map { |row| row["ThreadReplyId"] })
|
|
|
|
|
|
|
|
create_posts(rows, total: total_count, offset: offset) do |row|
|
2020-03-11 20:39:16 -04:00
|
|
|
imported_parent_id = row["ParentReplyId"]&.nonzero? ? row["ParentReplyId"] : import_topic_id(row["ThreadId"])
|
2018-06-26 18:02:03 -04:00
|
|
|
parent_post = topic_lookup_from_imported_post_id(imported_parent_id)
|
|
|
|
user_id = user_id_from_imported_user_id(row["UserId"]) || Discourse::SYSTEM_USER_ID
|
|
|
|
|
|
|
|
if parent_post
|
|
|
|
post = {
|
|
|
|
id: row["ThreadReplyId"],
|
2020-03-11 20:39:16 -04:00
|
|
|
raw: raw_with_attachment(row, user_id, :post),
|
2018-06-26 18:02:03 -04:00
|
|
|
user_id: user_id,
|
|
|
|
topic_id: parent_post[:topic_id],
|
|
|
|
created_at: row["ThreadReplyDate"],
|
|
|
|
reply_to_post_number: parent_post[:post_number]
|
|
|
|
}
|
|
|
|
|
|
|
|
post[:custom_fields] = { is_accepted_answer: "true" } if row["IsFirstVerifiedAnswer"]
|
|
|
|
post
|
|
|
|
else
|
|
|
|
puts "Failed to import post #{row['ThreadReplyId']}. Parent was not found."
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-03-11 20:39:16 -04:00
|
|
|
def raw_with_attachment(row, user_id, type)
|
2018-08-13 09:27:51 -04:00
|
|
|
raw, embedded_paths, upload_ids = replace_embedded_attachments(row["Body"], user_id)
|
2018-06-26 18:02:03 -04:00
|
|
|
raw = html_to_markdown(raw) || ""
|
|
|
|
|
|
|
|
filename = row["FileName"]
|
2019-05-30 11:37:51 -04:00
|
|
|
return raw if ENV["FILE_BASE_DIR"].blank? || filename.blank?
|
2018-06-26 18:02:03 -04:00
|
|
|
|
2020-03-11 20:39:16 -04:00
|
|
|
if row["IsRemote"]
|
|
|
|
return "#{raw}\n#{filename}"
|
|
|
|
end
|
|
|
|
|
2018-06-26 18:02:03 -04:00
|
|
|
path = File.join(
|
|
|
|
ENV["FILE_BASE_DIR"],
|
|
|
|
"telligent.evolution.components.attachments",
|
|
|
|
"%02d" % row["ApplicationTypeId"],
|
|
|
|
"%02d" % row["ApplicationId"],
|
|
|
|
"%02d" % row["ApplicationContentTypeId"],
|
|
|
|
("%010d" % row["ContentId"]).scan(/.{2}/),
|
2018-08-13 09:27:51 -04:00
|
|
|
clean_filename(filename)
|
2018-06-26 18:02:03 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
unless embedded_paths.include?(path)
|
2020-03-11 20:39:16 -04:00
|
|
|
if File.file?(path)
|
2018-06-26 18:02:03 -04:00
|
|
|
upload = @uploader.create_upload(user_id, path, filename)
|
2018-08-13 09:27:51 -04:00
|
|
|
|
|
|
|
if upload.present? && upload.persisted? && !upload_ids.include?(upload.id)
|
2019-05-30 11:37:51 -04:00
|
|
|
raw = "#{raw}\n#{@uploader.html_for_upload(upload, filename)}"
|
2018-08-13 09:27:51 -04:00
|
|
|
end
|
2018-06-26 18:02:03 -04:00
|
|
|
else
|
2020-03-11 20:39:16 -04:00
|
|
|
id = type == :topic ? row['ThreadId'] : row['ThreadReplyId']
|
|
|
|
STDERR.puts "Could not find file for #{type} #{id}: #{path}"
|
2018-06-26 18:02:03 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
raw
|
|
|
|
end
|
|
|
|
|
|
|
|
def replace_embedded_attachments(raw, user_id)
|
|
|
|
paths = []
|
2018-08-13 09:27:51 -04:00
|
|
|
upload_ids = []
|
2018-06-26 18:02:03 -04:00
|
|
|
|
2020-03-11 20:39:16 -04:00
|
|
|
return [raw, paths, upload_ids] if ENV["FILE_BASE_DIR"].blank?
|
|
|
|
|
2018-06-26 18:02:03 -04:00
|
|
|
raw = raw.gsub(EMBEDDED_ATTACHMENT_REGEX) do
|
2018-08-13 09:27:51 -04:00
|
|
|
filename, path = attachment_path(Regexp.last_match)
|
2018-06-26 18:02:03 -04:00
|
|
|
|
2020-03-11 20:39:16 -04:00
|
|
|
if File.file?(path)
|
2018-06-26 18:02:03 -04:00
|
|
|
upload = @uploader.create_upload(user_id, path, filename)
|
|
|
|
|
|
|
|
if upload.present? && upload.persisted?
|
|
|
|
paths << path
|
2018-08-13 09:27:51 -04:00
|
|
|
upload_ids << upload.id
|
2018-06-26 18:02:03 -04:00
|
|
|
@uploader.html_for_upload(upload, filename)
|
|
|
|
end
|
|
|
|
else
|
|
|
|
STDERR.puts "Could not find file: #{path}"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-08-13 09:27:51 -04:00
|
|
|
[raw, paths, upload_ids]
|
|
|
|
end
|
|
|
|
|
|
|
|
def clean_filename(filename)
|
2020-03-11 20:39:16 -04:00
|
|
|
filename = CGI.unescapeHTML(filename)
|
|
|
|
filename.gsub!(/[\x00\/\\:\*\?\"<>\|]/, '_')
|
|
|
|
|
|
|
|
%w|( ) # % - _ [ ] = , ' ~ ! + { } & @ #|.each do |c|
|
|
|
|
number = "#{c.ord.to_s(16).upcase}00"
|
|
|
|
filename.gsub!("_#{number}_", c)
|
|
|
|
filename.gsub!("_#{number}#{number}_", c * 2)
|
|
|
|
end
|
|
|
|
|
|
|
|
filename
|
2018-08-13 09:27:51 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def attachment_path(match_data)
|
|
|
|
filename, path = join_attachment_path(match_data, filename_index: 2)
|
2020-03-11 20:39:16 -04:00
|
|
|
filename, path = join_attachment_path(match_data, filename_index: 1) unless File.file?(path)
|
2018-08-13 09:27:51 -04:00
|
|
|
[filename, path]
|
|
|
|
end
|
|
|
|
|
|
|
|
# filenames are a total mess - try to guess the correct filename
|
|
|
|
# works for 70% of all files
|
|
|
|
def join_attachment_path(match_data, filename_index:)
|
|
|
|
filename = clean_filename(match_data[:"filename#{filename_index}"])
|
|
|
|
base_path = File.join(
|
|
|
|
ENV["FILE_BASE_DIR"],
|
2020-03-11 20:39:16 -04:00
|
|
|
match_data[:directory].gsub("-", ".").downcase,
|
|
|
|
match_data[:path].gsub("+", " ").gsub(/_\h{4}_/, "?").split(/[\.\-]/).map(&:strip)
|
2018-08-13 09:27:51 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
original_filename = filename.dup
|
|
|
|
|
2020-03-11 20:39:16 -04:00
|
|
|
filename.gsub!(/[_\- ]/, "?")
|
2018-08-13 09:27:51 -04:00
|
|
|
path = File.join(base_path, filename)
|
2020-03-11 20:39:16 -04:00
|
|
|
paths = Dir.glob(path, File::FNM_CASEFOLD)
|
|
|
|
if (path = paths.first) && paths.size == 1
|
|
|
|
filename = File.basename(path)
|
|
|
|
return [filename, path]
|
|
|
|
end
|
2018-08-13 09:27:51 -04:00
|
|
|
|
2020-03-11 20:39:16 -04:00
|
|
|
filename = original_filename.dup
|
|
|
|
filename.gsub!(/_\h{2}00_/, "?")
|
|
|
|
filename.gsub!(/_\h{2}00\h{2}00_/, "??")
|
|
|
|
filename.gsub!(/_\h{2}00\h{2}00\h{2}00_/, "???")
|
|
|
|
filename.gsub!(/[_\- ]/, "?")
|
2018-08-13 09:27:51 -04:00
|
|
|
path = File.join(base_path, filename)
|
2020-03-11 20:39:16 -04:00
|
|
|
paths = Dir.glob(path, File::FNM_CASEFOLD)
|
|
|
|
if (path = paths.first) && paths.size == 1
|
|
|
|
filename = File.basename(path)
|
|
|
|
return [filename, path]
|
|
|
|
end
|
2018-06-26 18:02:03 -04:00
|
|
|
|
2020-03-11 20:39:16 -04:00
|
|
|
filename = original_filename.dup
|
|
|
|
filename.gsub!(/_\h{4}_/, "?")
|
|
|
|
filename.gsub!(/_\h{4}\h{4}_/, "??")
|
|
|
|
filename.gsub!(/_\h{4}\h{4}\h{4}_/, "???")
|
|
|
|
filename.gsub!(/[_\- ]/, "?")
|
|
|
|
path = File.join(base_path, filename)
|
|
|
|
paths = Dir.glob(path, File::FNM_CASEFOLD)
|
|
|
|
if (path = paths.first) && paths.size == 1
|
|
|
|
filename = File.basename(path)
|
|
|
|
return [filename, path]
|
|
|
|
end
|
2018-06-26 18:02:03 -04:00
|
|
|
|
2020-03-11 20:39:16 -04:00
|
|
|
[original_filename, File.join(base_path, original_filename)]
|
2018-06-26 18:02:03 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def html_to_markdown(html)
|
2020-03-11 20:39:16 -04:00
|
|
|
return html if html.blank?
|
|
|
|
|
|
|
|
md = HtmlToMarkdown.new(html).to_markdown
|
|
|
|
md.gsub!(/\[quote.*?\]/, "\n" + '\0' + "\n")
|
|
|
|
md.gsub!(/(?<!^)\[\/quote\]/, "\n[/quote]\n")
|
|
|
|
md.strip!
|
|
|
|
md
|
2018-06-26 18:02:03 -04:00
|
|
|
end
|
|
|
|
|
2020-03-14 17:10:19 -04:00
|
|
|
def mark_topics_as_solved
|
|
|
|
puts "", "Marking topics as solved..."
|
|
|
|
|
|
|
|
DB.exec <<~SQL
|
|
|
|
INSERT INTO topic_custom_fields (name, value, topic_id, created_at, updated_at)
|
|
|
|
SELECT 'accepted_answer_post_id', pcf.post_id, p.topic_id, p.created_at, p.created_at
|
|
|
|
FROM post_custom_fields pcf
|
|
|
|
JOIN posts p ON p.id = pcf.post_id
|
|
|
|
WHERE pcf.name = 'is_accepted_answer' AND pcf.value = 'true'
|
|
|
|
SQL
|
|
|
|
end
|
|
|
|
|
2018-06-26 18:02:03 -04:00
|
|
|
def add_permalink_normalizations
|
|
|
|
normalizations = SiteSetting.permalink_normalizations
|
|
|
|
normalizations = normalizations.blank? ? [] : normalizations.split('|')
|
|
|
|
|
|
|
|
add_normalization(normalizations, CATEGORY_LINK_NORMALIZATION)
|
|
|
|
add_normalization(normalizations, TOPIC_LINK_NORMALIZATION)
|
|
|
|
|
|
|
|
SiteSetting.permalink_normalizations = normalizations.join('|')
|
|
|
|
end
|
|
|
|
|
|
|
|
def add_normalization(normalizations, normalization)
|
|
|
|
normalizations << normalization unless normalizations.include?(normalization)
|
|
|
|
end
|
|
|
|
|
|
|
|
def batches
|
|
|
|
super(BATCH_SIZE)
|
|
|
|
end
|
|
|
|
|
|
|
|
def query(sql)
|
|
|
|
@client.execute(sql).to_a
|
|
|
|
end
|
|
|
|
|
|
|
|
def count(sql)
|
|
|
|
query(sql).first["count"]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
ImportScripts::Telligent.new.perform
|