diff --git a/app/assets/javascripts/admin/models/export_csv.js b/app/assets/javascripts/admin/models/export_csv.js
new file mode 100644
index 00000000000..3afe84a1ce6
--- /dev/null
+++ b/app/assets/javascripts/admin/models/export_csv.js
@@ -0,0 +1,26 @@
+/**
+ Data model for representing an export
+
+ @class ExportCsv
+ @extends Discourse.Model
+ @namespace Discourse
+ @module Discourse
+**/
+Discourse.ExportCsv = Discourse.Model.extend({});
+
+Discourse.ExportCsv.reopenClass({
+ /**
+ Exports user list
+
+ @method export_user_list
+ **/
+ exportUserList: function() {
+ return Discourse.ajax("/admin/export_csv/users.json").then(function(result) {
+ if (result.success) {
+ bootbox.alert(I18n.t("admin.export_csv.success"));
+ } else {
+ bootbox.alert(I18n.t("admin.export_csv.failed"));
+ }
+ });
+ }
+});
diff --git a/app/assets/javascripts/admin/routes/admin_users_list_routes.js b/app/assets/javascripts/admin/routes/admin_users_list_routes.js
index 86b180d39bc..bc1516c182b 100644
--- a/app/assets/javascripts/admin/routes/admin_users_list_routes.js
+++ b/app/assets/javascripts/admin/routes/admin_users_list_routes.js
@@ -9,6 +9,12 @@
Discourse.AdminUsersListRoute = Discourse.Route.extend({
renderTemplate: function() {
this.render('admin/templates/users_list', {into: 'admin/templates/admin'});
+ },
+
+ actions: {
+ exportUsers: function() {
+ Discourse.ExportCsv.exportUserList();
+ }
}
});
@@ -57,7 +63,7 @@ Discourse.AdminUsersListNewRoute = Discourse.Route.extend({
/**
Handles the route that lists pending users.
- @class AdminUsersListNewRoute
+ @class AdminUsersListPendingRoute
@extends Discourse.Route
@namespace Discourse
@module Discourse
diff --git a/app/assets/javascripts/admin/templates/users_list.js.handlebars b/app/assets/javascripts/admin/templates/users_list.js.handlebars
index 61cacc69f1c..131036732a3 100644
--- a/app/assets/javascripts/admin/templates/users_list.js.handlebars
+++ b/app/assets/javascripts/admin/templates/users_list.js.handlebars
@@ -15,6 +15,9 @@
diff --git a/app/controllers/admin/export_csv_controller.rb b/app/controllers/admin/export_csv_controller.rb
new file mode 100644
index 00000000000..f0e574105a9
--- /dev/null
+++ b/app/controllers/admin/export_csv_controller.rb
@@ -0,0 +1,20 @@
+class Admin::ExportCsvController < Admin::AdminController
+
+ skip_before_filter :check_xhr, only: [:download]
+
+ def export_user_list
+ # export csv file in a background thread
+ Jobs.enqueue(:export_csv_file, entity: 'user', user_id: current_user.id)
+ render json: success_json
+ end
+
+ def download
+ filename = params.fetch(:id)
+ if export_csv_path = ExportCsv.get_download_path(filename)
+ send_file export_csv_path
+ else
+ render nothing: true, status: 404
+ end
+ end
+
+end
diff --git a/app/jobs/regular/export_csv_file.rb b/app/jobs/regular/export_csv_file.rb
new file mode 100644
index 00000000000..6e4a0b7979e
--- /dev/null
+++ b/app/jobs/regular/export_csv_file.rb
@@ -0,0 +1,75 @@
+require 'csv'
+require_dependency 'system_message'
+
+module Jobs
+
+ class ExportCsvFile < Jobs::Base
+ sidekiq_options retry: false
+ attr_accessor :current_user
+
+ def initialize
+ @file_name = ""
+ end
+
+ def execute(args)
+ entity = args[:entity]
+ @current_user = User.find_by(id: args[:user_id])
+
+ raise Discourse::InvalidParameters.new(:entity) if entity.blank?
+
+ case entity
+ when 'user'
+ query = ::AdminUserIndexQuery.new
+ user_data = query.find_users_query.to_a
+
+ data = Hash.new do |hash, key|
+ hash[key] = {}
+ end
+
+ user_data.each do |user|
+ id = user['id']
+ email = user['email']
+ data[id] = email
+ end
+ data = data.to_a
+ end
+
+ if data && data.length > 0
+ set_file_path
+ write_csv_file(data)
+ end
+
+ notify_user
+ end
+
+ private
+
+ def set_file_path
+ @file_name = "export_#{SecureRandom.hex(4)}.csv"
+ # ensure directory exists
+ dir = File.dirname("#{ExportCsv.base_directory}/#{@file_name}")
+ FileUtils.mkdir_p(dir) unless Dir.exists?(dir)
+ end
+
+ def write_csv_file(data)
+ # write to CSV file
+ CSV.open(File.expand_path("#{ExportCsv.base_directory}/#{@file_name}", __FILE__), "w") do |csv|
+ data.each do |value|
+ csv << [value[1]]
+ end
+ end
+ end
+
+ def notify_user
+ if @current_user
+ if @file_name != "" && File.exists?("#{ExportCsv.base_directory}/#{@file_name}")
+ SystemMessage.create_from_system_user(@current_user, :csv_export_succeeded, download_link: "#{Discourse.base_url}/admin/export_csv/#{@file_name}/download", file_name: @file_name)
+ else
+ SystemMessage.create_from_system_user(@current_user, :csv_export_failed)
+ end
+ end
+ end
+
+ end
+
+end
diff --git a/app/jobs/scheduled/weekly.rb b/app/jobs/scheduled/weekly.rb
index 156cfd74e45..9a079efb05e 100644
--- a/app/jobs/scheduled/weekly.rb
+++ b/app/jobs/scheduled/weekly.rb
@@ -11,6 +11,7 @@ module Jobs
Post.calculate_avg_time
Topic.calculate_avg_time
ScoreCalculator.new.calculate
+ ExportCsv.remove_old_exports # delete exported CSV files older than 2 days
end
end
end
diff --git a/app/models/export_csv.rb b/app/models/export_csv.rb
new file mode 100644
index 00000000000..ffebf6c1eee
--- /dev/null
+++ b/app/models/export_csv.rb
@@ -0,0 +1,25 @@
+class ExportCsv
+
+ def self.get_download_path(filename)
+ path = File.join(ExportCsv.base_directory, filename)
+ if File.exists?(path)
+ return path
+ else
+ nil
+ end
+ end
+
+ def self.remove_old_exports
+ dir = Dir.new(ExportCsv.base_directory)
+ dir.each do |file|
+ if (File.mtime(File.join(ExportCsv.base_directory, file)) < 2.days.ago)
+ File.delete(File.join(ExportCsv.base_directory, file))
+ end
+ end
+ end
+
+ def self.base_directory
+ File.join(Rails.root, "public", "uploads", "csv_exports", RailsMultisite::ConnectionManagement.current_db)
+ end
+
+end
diff --git a/config/locales/client.en.yml b/config/locales/client.en.yml
index 8f30a72768c..d25b0bbd441 100644
--- a/config/locales/client.en.yml
+++ b/config/locales/client.en.yml
@@ -1603,6 +1603,13 @@ en:
title: "Rollback the database to previous working state"
confirm: "Are your sure you want to rollback the database to the previous working state?"
+ export_csv:
+ users:
+ text: "Export Users"
+ title: "Export user list in a CSV file."
+ success: "Export has been initiated, you will be notified shortly with progress."
+ failed: "Export failed. Please check the logs."
+
customize:
title: "Customize"
long_title: "Site Customizations"
diff --git a/config/locales/server.en.yml b/config/locales/server.en.yml
index 4acc4355f07..512a8a2769e 100644
--- a/config/locales/server.en.yml
+++ b/config/locales/server.en.yml
@@ -1387,6 +1387,17 @@ en:
%{logs}
```
+ csv_export_succeeded:
+ subject_template: "Data Export completed successfully"
+ text_body_template: |
+ The data export was successful.
+
+ Download CSV file:
%{file_name}
+
+ csv_export_failed:
+ subject_template: "Export failed"
+ text_body_template: "The export has failed. Please check the logs."
+
email_reject_trust_level:
subject_template: "Email issue -- Insufficient Trust Level"
text_body_template: |
diff --git a/config/routes.rb b/config/routes.rb
index 89782fe4ca0..add9d1604f1 100644
--- a/config/routes.rb
+++ b/config/routes.rb
@@ -145,6 +145,15 @@ Discourse::Application.routes.draw do
end
end
+ resources :export_csv, constraints: AdminConstraint.new do
+ member do
+ get "download" => "export_csv#download", constraints: { id: /[^\/]+/ }
+ end
+ collection do
+ get "users" => "export_csv#export_user_list"
+ end
+ end
+
resources :badges, constraints: AdminConstraint.new do
collection do
get "types" => "badges#badge_types"
diff --git a/spec/controllers/admin/export_csv_controller_spec.rb b/spec/controllers/admin/export_csv_controller_spec.rb
new file mode 100644
index 00000000000..90863df2bd6
--- /dev/null
+++ b/spec/controllers/admin/export_csv_controller_spec.rb
@@ -0,0 +1,35 @@
+require "spec_helper"
+
+describe Admin::ExportCsvController do
+
+ it "is a subclass of AdminController" do
+ (Admin::ExportCsvController < Admin::AdminController).should be_true
+ end
+
+ let(:export_filename) { "export_b6a2bc87.csv" }
+
+ context "while logged in as an admin" do
+
+ before { @admin = log_in(:admin) }
+
+ describe ".download" do
+
+ it "uses send_file to transmit the export file" do
+ controller.stubs(:render)
+ export = ExportCsv.new()
+ ExportCsv.expects(:get_download_path).with(export_filename).returns(export)
+ subject.expects(:send_file).with(export)
+ get :download, id: export_filename
+ end
+
+ it "returns 404 when the export file does not exist" do
+ ExportCsv.expects(:get_download_path).returns(nil)
+ get :download, id: export_filename
+ response.should be_not_found
+ end
+
+ end
+
+ end
+
+end
diff --git a/spec/jobs/export_csv_file_spec.rb b/spec/jobs/export_csv_file_spec.rb
new file mode 100644
index 00000000000..269fb12b2a9
--- /dev/null
+++ b/spec/jobs/export_csv_file_spec.rb
@@ -0,0 +1,13 @@
+require 'spec_helper'
+
+describe Jobs::ExportCsvFile do
+
+ context '.execute' do
+
+ it 'raises an error when the entity is missing' do
+ lambda { Jobs::ExportCsvFile.new.execute(user_id: "1") }.should raise_error(Discourse::InvalidParameters)
+ end
+
+ end
+end
+