diff --git a/Gemfile b/Gemfile index 703c5da332a..c2680dc0b06 100644 --- a/Gemfile +++ b/Gemfile @@ -174,7 +174,8 @@ group :development do gem 'binding_of_caller' gem 'yaml-lint' gem 'annotate' - gem 'discourse_dev' + gem 'discourse_dev_assets' + gem 'faker', "~> 2.16" end # this is an optional gem, it provides a high performance replacement diff --git a/Gemfile.lock b/Gemfile.lock index 428edc72973..95ae17fad25 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -115,7 +115,7 @@ GEM railties (>= 3.1) discourse-ember-source (3.12.2.3) discourse-fonts (0.0.8) - discourse_dev (0.2.1) + discourse_dev_assets (0.0.2) faker (~> 2.16) docile (1.4.0) ecma-re-validator (0.3.0) @@ -504,12 +504,13 @@ DEPENDENCIES discourse-ember-rails (= 0.18.6) discourse-ember-source (~> 3.12.2) discourse-fonts - discourse_dev + discourse_dev_assets email_reply_trimmer ember-handlebars-template (= 0.8.0) excon execjs fabrication + faker (~> 2.16) fakeweb fast_blank fast_xs diff --git a/config/dev_defaults.yml b/config/dev_defaults.yml new file mode 100644 index 00000000000..4c5fbd6eaf8 --- /dev/null +++ b/config/dev_defaults.yml @@ -0,0 +1,38 @@ +site_settings: + tagging_enabled: true + verbose_discourse_connect_logging: true + +seed: 1 +start_date: 2020-01-01 +auth_plugin_enabled: true +allow_anonymous_to_impersonate: false + +category: + count: 30 +group: + count: 15 +post: + include_images: false + max_likes_count: 10 +tag: + count: 30 +topic: + count: 30 + replies: + # number of replies per topic between min and max + min: 0 + max: 12 + overrides: + # topic titles can be found in config/locales/faker.en.yml + - title: "Coolest thing you have seen today" + count: 99 + tags: + # number of tags per topic between min and max + min: 0 + max: 3 +user: + count: 30 + +new_user: + username: new_user + email: new_user@example.com diff --git a/config/routes.rb b/config/routes.rb index e395cd0df1f..6946889d644 100644 --- a/config/routes.rb +++ b/config/routes.rb @@ -977,10 +977,6 @@ Discourse::Application.routes.draw do post "/do-not-disturb" => "do_not_disturb#create" delete "/do-not-disturb" => "do_not_disturb#destroy" - if Rails.env.development? - mount DiscourseDev::Engine => "/dev/" - end - get "*url", to: 'permalinks#show', constraints: PermalinkConstraint.new end end diff --git a/lib/discourse_dev.rb b/lib/discourse_dev.rb new file mode 100644 index 00000000000..45774b07733 --- /dev/null +++ b/lib/discourse_dev.rb @@ -0,0 +1,15 @@ +# frozen_string_literal: true + +module DiscourseDev + def self.config + @config ||= Config.new + end + + def self.settings_file + File.join(root, "config", "settings.yml") + end + + def self.root + File.expand_path("..", __dir__) + end +end diff --git a/lib/discourse_dev/category.rb b/lib/discourse_dev/category.rb new file mode 100644 index 00000000000..b13c43d7587 --- /dev/null +++ b/lib/discourse_dev/category.rb @@ -0,0 +1,60 @@ +# frozen_string_literal: true + +require 'discourse_dev/record' +require 'rails' +require 'faker' + +module DiscourseDev + class Category < Record + + def initialize + super(::Category, DiscourseDev.config.category[:count]) + @parent_category_ids = ::Category.where(parent_category_id: nil).pluck(:id) + end + + def data + name = Faker::Discourse.unique.category + parent_category_id = nil + + if Faker::Boolean.boolean(true_ratio: 0.6) + offset = Faker::Number.between(from: 0, to: @parent_category_ids.count - 1) + parent_category_id = @parent_category_ids[offset] + @permissions = ::Category.find(parent_category_id).permissions_params.presence + else + @permissions = nil + end + + { + name: name, + description: Faker::Lorem.paragraph, + user_id: ::Discourse::SYSTEM_USER_ID, + color: Faker::Color.hex_color.last(6), + parent_category_id: parent_category_id + } + end + + def permissions + return @permissions if @permissions.present? + return { everyone: :full } if Faker::Boolean.boolean(true_ratio: 0.75) + + permission = {} + group = Group.random + permission[group.id] = Faker::Number.between(from: 1, to: 3) + + permission + end + + def create! + super do |category| + category.set_permissions(permissions) + category.save! + + @parent_category_ids << category.id if category.parent_category_id.blank? + end + end + + def self.random + super(::Category) + end + end +end diff --git a/lib/discourse_dev/config.rb b/lib/discourse_dev/config.rb new file mode 100644 index 00000000000..4a2142b43c6 --- /dev/null +++ b/lib/discourse_dev/config.rb @@ -0,0 +1,149 @@ +# frozen_string_literal: true + +require 'rails' +require 'highline/import' + +module DiscourseDev + class Config + attr_reader :config, :file_path + + def initialize + default_file_path = File.join(Rails.root, "config", "dev_defaults.yml") + @file_path = File.join(Rails.root, "config", "dev.yml") + default_config = YAML.load_file(default_file_path) + + if File.exists?(file_path) + user_config = YAML.load_file(file_path) + else + puts "I did no detect a custom `config/dev.yml` file, creating one for you where you can amend defaults." + FileUtils.cp(default_file_path, file_path) + user_config = {} + end + + @config = default_config.deep_merge(user_config).deep_symbolize_keys + end + + def update! + update_site_settings + create_admin_user + create_new_user + set_seed + end + + private + + def update_site_settings + puts "Updating site settings..." + + site_settings = config[:site_settings] || {} + + site_settings.each do |key, value| + puts "#{key} = #{value}" + SiteSetting.set(key, value) + end + + SiteSetting.refresh! + end + + def create_admin_user + puts "Creating default admin user account..." + + settings = config[:admin] + + if settings.present? + create_admin_user_from_settings(settings) + else + create_admin_user_from_input + end + end + + def create_new_user + settings = config[:new_user] + + if settings.present? + email = settings[:email] || "new_user@example.com" + + new_user = ::User.create!( + email: email, + username: settings[:username] || UserNameSuggester.suggest(email) + ) + new_user.email_tokens.update_all confirmed: true + new_user.activate + end + end + + def set_seed + seed = self.seed || 1 + Faker::Config.random = Random.new(seed) + end + + def start_date + DateTime.parse(config[:start_date]) + end + + def method_missing(name) + config[name.to_sym] + end + + def create_admin_user_from_settings(settings) + email = settings[:email] + + admin = ::User.with_email(email).first_or_create!( + email: email, + username: settings[:username] || UserNameSuggester.suggest(email), + password: settings[:password] + ) + admin.grant_admin! + if admin.trust_level < 1 + admin.change_trust_level!(1) + end + admin.email_tokens.update_all confirmed: true + admin.activate + end + + def create_admin_user_from_input + begin + email = ask("Email: ") + password = ask("Password (optional, press ENTER to skip): ") + username = UserNameSuggester.suggest(email) + + admin = ::User.new( + email: email, + username: username + ) + + if password.present? + admin.password = password + else + puts "Once site is running use https://localhost:9292/user/#{username}/become to access the account in development" + end + + admin.name = ask("Full name: ") if SiteSetting.full_name_required + saved = admin.save + + if saved + File.open(file_path, 'a') do | file| + file.puts("admin:") + file.puts(" username: #{admin.username}") + file.puts(" email: #{admin.email}") + file.puts(" password: #{password}") if password.present? + end + else + say(admin.errors.full_messages.join("\n")) + end + end while !saved + + admin.active = true + admin.save + + admin.grant_admin! + if admin.trust_level < 1 + admin.change_trust_level!(1) + end + admin.email_tokens.update_all confirmed: true + admin.activate + + say("\nAdmin account created successfully with username #{admin.username}") + end + end +end diff --git a/lib/discourse_dev/group.rb b/lib/discourse_dev/group.rb new file mode 100644 index 00000000000..c346b0c5faa --- /dev/null +++ b/lib/discourse_dev/group.rb @@ -0,0 +1,38 @@ +# frozen_string_literal: true + +require 'discourse_dev/record' +require 'rails' +require 'faker' + +module DiscourseDev + class Group < Record + + def initialize + super(::Group, DiscourseDev.config.group[:count]) + end + + def data + { + name: Faker::Discourse.unique.group, + public_exit: Faker::Boolean.boolean, + public_admission: Faker::Boolean.boolean, + primary_group: Faker::Boolean.boolean, + created_at: Faker::Time.between(from: DiscourseDev.config.start_date, to: DateTime.now), + } + end + + def create! + super do |group| + if Faker::Boolean.boolean + group.add_owner(::Discourse.system_user) + group.allow_membership_requests = true + group.save! + end + end + end + + def self.random + super(::Group) + end + end +end diff --git a/lib/discourse_dev/post.rb b/lib/discourse_dev/post.rb new file mode 100644 index 00000000000..74d9bb85feb --- /dev/null +++ b/lib/discourse_dev/post.rb @@ -0,0 +1,107 @@ +# frozen_string_literal: true + +require 'discourse_dev/record' +require 'faker' + +module DiscourseDev + class Post < Record + + attr_reader :topic + + def initialize(topic, count) + super(::Post, count) + @topic = topic + + category = topic.category + @max_likes_count = DiscourseDev.config.post[:max_likes_count] + unless category.groups.blank? + group_ids = category.groups.pluck(:id) + @user_ids = ::GroupUser.where(group_id: group_ids).pluck(:user_id) + @user_count = @user_ids.count + @max_likes_count = @user_count - 1 + end + end + + def data + { + topic_id: topic.id, + raw: Faker::DiscourseMarkdown.sandwich(sentences: 5), + created_at: Faker::Time.between(from: topic.last_posted_at, to: DateTime.now), + skip_validations: true, + skip_guardian: true + } + end + + def create! + user = self.user + data = Faker::DiscourseMarkdown.with_user(user.id) { self.data } + post = PostCreator.new(user, data).create! + topic.reload + generate_likes(post) + end + + def generate_likes(post) + user_ids = [post.user_id] + + Faker::Number.between(from: 0, to: @max_likes_count).times do + user = self.user + next if user_ids.include?(user.id) + + PostActionCreator.new(user, post, PostActionType.types[:like], created_at: Faker::Time.between(from: post.created_at, to: DateTime.now)).perform + user_ids << user.id + end + end + + def user + return User.random if topic.category.groups.blank? + return Discourse.system_user if @user_ids.blank? + + position = Faker::Number.between(from: 0, to: @user_count - 1) + ::User.find(@user_ids[position]) + end + + def populate! + generate_likes(topic.first_post) + + @count.times do + create! + end + end + + def self.add_replies!(args) + if !args[:topic_id] + puts "Topic ID is required. Aborting." + return + end + + if !::Topic.find_by_id(args[:topic_id]) + puts "Topic ID does not match topic in DB, aborting." + return + end + + topic = ::Topic.find_by_id(args[:topic_id]) + count = args[:count] ? args[:count].to_i : 50 + + puts "Creating #{count} replies in '#{topic.title}'" + + count.times do |i| + begin + user = User.random + reply = Faker::DiscourseMarkdown.with_user(user.id) do + { + topic_id: topic.id, + raw: Faker::DiscourseMarkdown.sandwich(sentences: 5), + skip_validations: true + } + end + PostCreator.new(user, reply).create! + rescue ActiveRecord::RecordNotSaved => e + puts e + end + end + + puts "Done!" + end + + end +end diff --git a/lib/discourse_dev/record.rb b/lib/discourse_dev/record.rb new file mode 100644 index 00000000000..b6196aebcfb --- /dev/null +++ b/lib/discourse_dev/record.rb @@ -0,0 +1,68 @@ +# frozen_string_literal: true + +require 'discourse_dev' +require 'rails' +require 'faker' + +module DiscourseDev + class Record + DEFAULT_COUNT = 30.freeze + + attr_reader :model, :type + + def initialize(model, count = DEFAULT_COUNT) + @@initialized ||= begin + Faker::Discourse.unique.clear + RateLimiter.disable + true + end + + @model = model + @type = model.to_s + @count = count + end + + def create! + record = model.create!(data) + yield(record) if block_given? + end + + def populate! + if current_count >= @count + puts "Already have #{current_count} #{type.downcase} records" + + Rake.application.top_level_tasks.each do |task_name| + Rake::Task[task_name].reenable + end + + Rake::Task['dev:repopulate'].invoke + return + elsif current_count > 0 + @count -= current_count + puts "There are #{current_count} #{type.downcase} records. Creating #{@count} more." + else + puts "Creating #{@count} sample #{type.downcase} records" + end + + @count.times do + create! + putc "." + end + + puts + end + + def current_count + model.count + end + + def self.populate! + self.new.populate! + end + + def self.random(model) + offset = Faker::Number.between(from: 0, to: model.count - 1) + model.offset(offset).first + end + end +end diff --git a/lib/discourse_dev/tag.rb b/lib/discourse_dev/tag.rb new file mode 100644 index 00000000000..987d8656e9c --- /dev/null +++ b/lib/discourse_dev/tag.rb @@ -0,0 +1,32 @@ +# frozen_string_literal: true + +require 'discourse_dev/record' +require 'rails' +require 'faker' + +module DiscourseDev + class Tag < Record + + def initialize + super(::Tag, DiscourseDev.config.tag[:count]) + end + + def create! + super + rescue ActiveRecord::RecordInvalid => e + # If the name is taken, try again + retry + end + + def populate! + return unless SiteSetting.tagging_enabled + super + end + + def data + { + name: Faker::Discourse.unique.tag, + } + end + end +end diff --git a/lib/discourse_dev/topic.rb b/lib/discourse_dev/topic.rb new file mode 100644 index 00000000000..64bc86fd264 --- /dev/null +++ b/lib/discourse_dev/topic.rb @@ -0,0 +1,108 @@ +# frozen_string_literal: true + +require 'discourse_dev/record' +require 'faker' + +module DiscourseDev + class Topic < Record + + def initialize + @settings = DiscourseDev.config.topic + super(::Topic, @settings[:count]) + end + + def data + max_views = 0 + + case Faker::Number.between(from: 0, to: 5) + when 0 + max_views = 10 + when 1 + max_views = 100 + when 2 + max_views = SiteSetting.topic_views_heat_low + when 3 + max_views = SiteSetting.topic_views_heat_medium + when 4 + max_views = SiteSetting.topic_views_heat_high + when 5 + max_views = SiteSetting.topic_views_heat_high + SiteSetting.topic_views_heat_medium + end + + { + title: title[0, SiteSetting.max_topic_title_length], + raw: Faker::DiscourseMarkdown.sandwich(sentences: 5), + category: @category.id, + created_at: Faker::Time.between(from: DiscourseDev.config.start_date, to: DateTime.now), + tags: tags, + topic_opts: { + import_mode: true, + views: Faker::Number.between(from: 1, to: max_views), + custom_fields: { dev_sample: true } + }, + skip_validations: true + } + end + + def title + if current_count < I18n.t("faker.discourse.topics").count + Faker::Discourse.unique.topic + else + Faker::Lorem.unique.sentence(word_count: 5, supplemental: true, random_words_to_add: 4).chomp(".") + end + end + + def tags + return unless SiteSetting.tagging_enabled + + @tags = [] + + Faker::Number.between(from: @settings.dig(:tags, :min), to: @settings.dig(:tags, :max)).times do + @tags << Faker::Discourse.tag + end + + @tags.uniq + end + + def create! + @category = Category.random + user = self.user + topic = Faker::DiscourseMarkdown.with_user(user.id) { data } + post = PostCreator.new(user, topic).create! + + if override = @settings.dig(:replies, :overrides).find { |o| o[:title] == topic[:title] } + reply_count = override[:count] + else + reply_count = Faker::Number.between(from: @settings.dig(:replies, :min), to: @settings.dig(:replies, :max)) + end + + Post.new(post.topic, reply_count).populate! + end + + def populate! + super + delete_unwanted_sidekiq_jobs + end + + def user + return User.random if @category.groups.blank? + + group_ids = @category.groups.pluck(:id) + user_ids = ::GroupUser.where(group_id: group_ids).pluck(:user_id) + user_count = user_ids.count + position = Faker::Number.between(from: 0, to: user_count - 1) + ::User.find(user_ids[position] || Discourse::SYSTEM_USER_ID) + end + + def current_count + category_definition_topic_ids = ::Category.pluck(:topic_id) + ::Topic.where(archetype: :regular).where.not(id: category_definition_topic_ids).count + end + + def delete_unwanted_sidekiq_jobs + Sidekiq::ScheduledSet.new.each do |job| + job.delete if job.item["class"] == "Jobs::UserEmail" + end + end + end +end diff --git a/lib/discourse_dev/user.rb b/lib/discourse_dev/user.rb new file mode 100644 index 00000000000..c64f8dc3466 --- /dev/null +++ b/lib/discourse_dev/user.rb @@ -0,0 +1,93 @@ +# frozen_string_literal: true + +require 'discourse_dev/record' +require 'faker' + +module DiscourseDev + class User < Record + attr_reader :images + + def initialize + super(::User, DiscourseDev.config.user[:count]) + + @images = DiscourseDevAssets.avatars + end + + def data + name = Faker::Name.unique.name + email = Faker::Internet.unique.email(name: name, domain: "faker.invalid") + username = Faker::Internet.unique.username(specifier: ::User.username_length) + username = UserNameSuggester.suggest(username) + username_lower = ::User.normalize_username(username) + + { + name: name, + email: email, + username: username, + username_lower: username_lower, + moderator: Faker::Boolean.boolean(true_ratio: 0.1), + trust_level: Faker::Number.between(from: 1, to: 4), + created_at: Faker::Time.between(from: DiscourseDev.config.start_date, to: DateTime.now), + } + end + + def create! + super do |user| + user.activate + set_random_avatar(user) + Faker::Number.between(from: 0, to: 2).times do + group = Group.random + + group.add(user) + end + end + end + + def self.random + super(::User) + end + + def set_random_avatar(user) + return if images.blank? + return unless Faker::Boolean.boolean + + avatar_index = Faker::Number.between(from: 0, to: images.count - 1) + avatar_path = images[avatar_index] + create_avatar(user, avatar_path) + @images.delete_at(avatar_index) + end + + def create_avatar(user, avatar_path) + tempfile = copy_to_tempfile(avatar_path) + filename = "avatar#{File.extname(avatar_path)}" + upload = UploadCreator.new(tempfile, filename, type: "avatar").create_for(user.id) + + if upload.present? && upload.persisted? + user.create_user_avatar + user.user_avatar.update(custom_upload_id: upload.id) + user.update(uploaded_avatar_id: upload.id) + else + STDERR.puts "Failed to upload avatar for user #{user.username}: #{avatar_path}" + STDERR.puts upload.errors.inspect if upload + end + rescue + STDERR.puts "Failed to create avatar for user #{user.username}: #{avatar_path}" + ensure + tempfile.close! if tempfile + end + + private + + def copy_to_tempfile(source_path) + extension = File.extname(source_path) + tmp = Tempfile.new(['discourse-upload', extension]) + + File.open(source_path) do |source_stream| + IO.copy_stream(source_stream, tmp) + end + + tmp.rewind + tmp + end + end +end diff --git a/lib/faker/discourse.rb b/lib/faker/discourse.rb new file mode 100644 index 00000000000..a2c4720026d --- /dev/null +++ b/lib/faker/discourse.rb @@ -0,0 +1,26 @@ +# frozen_string_literal: true + +require 'faker' + +module Faker + class Discourse < Base + class << self + + def tag + fetch('discourse.tags') + end + + def category + fetch('discourse.categories') + end + + def group + fetch('discourse.groups') + end + + def topic + fetch('discourse.topics') + end + end + end +end diff --git a/lib/faker/discourse_markdown.rb b/lib/faker/discourse_markdown.rb new file mode 100644 index 00000000000..b858e996c61 --- /dev/null +++ b/lib/faker/discourse_markdown.rb @@ -0,0 +1,96 @@ +# frozen_string_literal: true + +require 'faker' +require 'net/http' +require 'json' + +module Faker + class DiscourseMarkdown < Markdown + class << self + attr_writer(:user_id) + + def user_id + @user_id || ::Discourse::SYSTEM_USER_ID + end + + def with_user(user_id) + current_user_id = self.user_id + self.user_id = user_id + begin + yield + ensure + self.user_id = current_user_id + end + end + + def image + image = next_image + image_file = load_image(image) + + upload = ::UploadCreator.new( + image_file, + image[:filename], + origin: image[:url] + ).create_for(user_id) + + ::UploadMarkdown.new(upload).to_markdown if upload.present? && upload.persisted? + rescue => e + STDERR.puts e + STDERR.puts e.backtrace.join("\n") + end + + private + + def next_image + if @images.blank? + if @stop_loading_images + @images = @all_images.dup + else + @next_page = (@next_page || 0) + 1 + url = URI("https://picsum.photos/v2/list?page=#{@next_page}&limit=50") + response = Net::HTTP.get(url) + json = JSON.parse(response) + + if json.blank? + @stop_loading_images = true + @images = @all_images.dup + else + @images = json.sort_by { |image| image["id"] } + @all_images = (@all_images || []).concat(@images) + end + end + end + + image = @images.pop + { filename: "#{image['id']}.jpg", url: "#{image['download_url']}.jpg" } + end + + def image_cache_dir + @image_cache_dir ||= ::File.join(Rails.root, "tmp", "discourse_dev", "images") + end + + def load_image(image) + cache_path = ::File.join(image_cache_dir, image[:filename]) + + if !::File.exists?(cache_path) + FileUtils.mkdir_p(image_cache_dir) + temp_file = ::FileHelper.download( + image[:url], + max_file_size: [SiteSetting.max_image_size_kb.kilobytes, 10.megabytes].max, + tmp_file_name: "image", + follow_redirect: true + ) + FileUtils.cp(temp_file, cache_path) + end + + ::File.open(cache_path) + end + + def available_methods + methods = super + methods << :image if ::DiscourseDev.config.post[:include_images] + methods + end + end + end +end diff --git a/lib/js_locale_helper.rb b/lib/js_locale_helper.rb index c51a8812291..9ab61c941d0 100644 --- a/lib/js_locale_helper.rb +++ b/lib/js_locale_helper.rb @@ -4,7 +4,6 @@ module JsLocaleHelper def self.plugin_client_files(locale_str) files = Dir["#{Rails.root}/plugins/*/config/locales/client*.#{locale_str}.yml"] - files += DiscourseDev.client_locale_files(locale_str) if Rails.env.development? I18n::Backend::DiscourseI18n.sort_locale_files(files) end diff --git a/lib/plugin/instance.rb b/lib/plugin/instance.rb index 6bf2fe0b53e..a7933989c35 100644 --- a/lib/plugin/instance.rb +++ b/lib/plugin/instance.rb @@ -90,8 +90,6 @@ class Plugin::Instance metadata = Plugin::Metadata.parse(source) plugins << self.new(metadata, path) end - - plugins << DiscourseDev.auth_plugin if Rails.env.development? && DiscourseDev.auth_plugin_enabled? } end diff --git a/lib/tasks/dev.rake b/lib/tasks/dev.rake new file mode 100644 index 00000000000..fa6e2f86984 --- /dev/null +++ b/lib/tasks/dev.rake @@ -0,0 +1,47 @@ +# frozen_string_literal: true + +def check_environment! + if !Rails.env.development? + raise "Database commands are only supported in development environment" + end + + ENV['SKIP_TEST_DATABASE'] = "1" + ENV['SKIP_MULTISITE'] = "1" +end + +desc 'Run db:migrate:reset task and populate sample content for development environment' +task 'dev:reset' => ['db:load_config'] do |_, args| + check_environment! + + Rake::Task['db:migrate:reset'].invoke + Rake::Task['dev:config'].invoke + Rake::Task['dev:populate'].invoke +end + +desc 'Initialize development environment' +task 'dev:config' => ['db:load_config'] do |_, args| + DiscourseDev.config.update! +end + +desc 'Populate sample content for development environment' +task 'dev:populate' => ['db:load_config'] do |_, args| + system("redis-cli flushall") + Rake::Task['groups:populate'].invoke + Rake::Task['users:populate'].invoke + Rake::Task['categories:populate'].invoke + Rake::Task['tags:populate'].invoke + Rake::Task['topics:populate'].invoke +end + +desc 'Repopulate sample datas in development environment' +task 'dev:repopulate' => ['db:load_config'] do |_, args| + require 'highline/import' + + answer = ask("Do you want to repopulate the database with fresh data? It will recreate DBs and run migration from scratch before generating all the samples. (Y/n) ") + + if (answer == "" || answer.downcase == 'y') + Rake::Task['dev:reset'].invoke + else + puts "You can run `bin/rails dev:reset` to repopulate anytime." + end +end diff --git a/lib/tasks/populate.rake b/lib/tasks/populate.rake new file mode 100644 index 00000000000..81a3bbae06c --- /dev/null +++ b/lib/tasks/populate.rake @@ -0,0 +1,31 @@ +# frozen_string_literal: true + +desc 'Creates sample categories' +task 'groups:populate' => ['db:load_config'] do |_, args| + DiscourseDev::Group.populate! +end + +desc 'Creates sample user accounts' +task 'users:populate' => ['db:load_config'] do |_, args| + DiscourseDev::User.populate! +end + +desc 'Creates sample categories' +task 'categories:populate' => ['db:load_config'] do |_, args| + DiscourseDev::Category.populate! +end + +desc 'Creates sample tags' +task 'tags:populate' => ['db:load_config'] do |_, args| + DiscourseDev::Tag.populate! +end + +desc 'Creates sample topics' +task 'topics:populate' => ['db:load_config'] do |_, args| + DiscourseDev::Topic.populate! +end + +desc 'Add replies to a topic' +task 'replies:populate', [:topic_id, :count] => ['db:load_config'] do |_, args| + DiscourseDev::Post.add_replies!(args) +end