2013-02-14 12:57:26 -05:00
|
|
|
require_dependency 'discourse_hub'
|
2013-06-06 10:40:10 -04:00
|
|
|
require_dependency 'user_name_suggester'
|
2013-08-18 10:37:38 -04:00
|
|
|
require_dependency 'user_activator'
|
2013-11-11 12:51:14 -05:00
|
|
|
require_dependency 'avatar_upload_service'
|
2013-02-05 14:16:51 -05:00
|
|
|
|
|
|
|
class UsersController < ApplicationController
|
|
|
|
|
2013-08-14 06:20:05 -04:00
|
|
|
skip_before_filter :authorize_mini_profiler, only: [:avatar]
|
|
|
|
skip_before_filter :check_xhr, only: [:show, :password_reset, :update, :activate_account, :authorize_email, :user_preferences_redirect, :avatar]
|
2013-02-05 14:16:51 -05:00
|
|
|
|
2013-08-24 16:37:31 -04:00
|
|
|
before_filter :ensure_logged_in, only: [:username, :update, :change_email, :user_preferences_redirect, :upload_avatar, :toggle_avatar]
|
2013-02-07 10:45:24 -05:00
|
|
|
|
2013-05-22 11:20:16 -04:00
|
|
|
# we need to allow account creation with bad CSRF tokens, if people are caching, the CSRF token on the
|
2013-05-03 02:43:11 -04:00
|
|
|
# page is going to be empty, this means that server will see an invalid CSRF and blow the session
|
|
|
|
# once that happens you can't log in with social
|
2013-06-05 00:01:24 -04:00
|
|
|
skip_before_filter :verify_authenticity_token, only: [:create]
|
2013-07-15 12:12:54 -04:00
|
|
|
skip_before_filter :redirect_to_login_if_required, only: [:check_username,
|
|
|
|
:create,
|
|
|
|
:get_honeypot_value,
|
|
|
|
:activate_account,
|
|
|
|
:send_activation_email,
|
|
|
|
:authorize_email,
|
|
|
|
:password_reset]
|
2013-05-03 02:43:11 -04:00
|
|
|
|
2013-02-07 10:45:24 -05:00
|
|
|
def show
|
2013-02-05 14:16:51 -05:00
|
|
|
@user = fetch_user_from_params
|
2013-03-08 15:04:37 -05:00
|
|
|
user_serializer = UserSerializer.new(@user, scope: guardian, root: 'user')
|
|
|
|
respond_to do |format|
|
|
|
|
format.html do
|
|
|
|
store_preloaded("user_#{@user.username}", MultiJson.dump(user_serializer))
|
|
|
|
end
|
|
|
|
|
|
|
|
format.json do
|
|
|
|
render_json_dump(user_serializer)
|
|
|
|
end
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def user_preferences_redirect
|
|
|
|
redirect_to email_preferences_path(current_user.username_lower)
|
|
|
|
end
|
|
|
|
|
|
|
|
def update
|
2013-10-07 06:19:45 -04:00
|
|
|
user = fetch_user_from_params
|
2013-02-05 14:16:51 -05:00
|
|
|
guardian.ensure_can_edit!(user)
|
2013-04-11 20:07:46 -04:00
|
|
|
json_result(user, serializer: UserSerializer) do |u|
|
2013-11-01 17:06:20 -04:00
|
|
|
updater = UserUpdater.new(user)
|
|
|
|
updater.update(params)
|
2013-02-07 10:45:24 -05:00
|
|
|
end
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def username
|
2013-06-06 03:14:32 -04:00
|
|
|
params.require(:new_username)
|
2013-02-05 14:16:51 -05:00
|
|
|
|
|
|
|
user = fetch_user_from_params
|
2013-08-12 14:54:52 -04:00
|
|
|
guardian.ensure_can_edit_username!(user)
|
2013-02-07 10:45:24 -05:00
|
|
|
|
2013-02-05 14:16:51 -05:00
|
|
|
result = user.change_username(params[:new_username])
|
|
|
|
raise Discourse::InvalidParameters.new(:new_username) unless result
|
|
|
|
|
2013-02-07 10:45:24 -05:00
|
|
|
render nothing: true
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def preferences
|
|
|
|
render nothing: true
|
|
|
|
end
|
|
|
|
|
|
|
|
def invited
|
2013-11-08 14:11:41 -05:00
|
|
|
inviter = fetch_user_from_params
|
2013-11-05 17:52:50 -05:00
|
|
|
|
2013-11-08 14:11:41 -05:00
|
|
|
invites = if guardian.can_see_pending_invites_from?(inviter)
|
|
|
|
Invite.find_all_invites_from(inviter)
|
|
|
|
else
|
|
|
|
Invite.find_redeemed_invites_from(inviter)
|
2013-11-05 17:52:50 -05:00
|
|
|
end
|
|
|
|
|
2013-11-08 14:11:41 -05:00
|
|
|
invites = invites.filter_by(params[:filter])
|
2013-11-05 17:52:50 -05:00
|
|
|
|
2013-11-08 14:11:41 -05:00
|
|
|
render_serialized(invites.to_a, InviteSerializer)
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def is_local_username
|
2013-06-06 03:14:32 -04:00
|
|
|
params.require(:username)
|
2013-02-05 14:16:51 -05:00
|
|
|
u = params[:username].downcase
|
|
|
|
r = User.exec_sql('select 1 from users where username_lower = ?', u).values
|
|
|
|
render json: {valid: r.length == 1}
|
|
|
|
end
|
|
|
|
|
2013-08-25 04:57:12 -04:00
|
|
|
def render_available_true
|
|
|
|
render(json: { available: true })
|
|
|
|
end
|
|
|
|
|
|
|
|
def changing_case_of_own_username(target_user, username)
|
|
|
|
target_user and username.downcase == target_user.username.downcase
|
|
|
|
end
|
|
|
|
|
|
|
|
# Used for checking availability of a username and will return suggestions
|
|
|
|
# if the username is not available.
|
2013-02-05 14:16:51 -05:00
|
|
|
def check_username
|
2013-06-06 03:14:32 -04:00
|
|
|
params.require(:username)
|
2013-08-25 04:57:12 -04:00
|
|
|
username = params[:username]
|
2013-02-05 14:16:51 -05:00
|
|
|
|
2013-08-25 04:57:12 -04:00
|
|
|
target_user = user_from_params_or_current_user
|
2013-07-30 14:13:56 -04:00
|
|
|
|
2013-06-28 16:21:46 -04:00
|
|
|
# The special case where someone is changing the case of their own username
|
2013-08-25 04:57:12 -04:00
|
|
|
return render_available_true if changing_case_of_own_username(target_user, username)
|
2013-06-28 16:21:46 -04:00
|
|
|
|
2013-09-06 05:35:29 -04:00
|
|
|
checker = UsernameCheckerService.new
|
|
|
|
email = params[:email] || target_user.try(:email)
|
|
|
|
render(json: checker.check_username(username, email))
|
2013-08-25 04:57:12 -04:00
|
|
|
rescue RestClient::Forbidden
|
|
|
|
render json: {errors: [I18n.t("discourse_hub.access_token_problem")]}
|
|
|
|
end
|
2013-02-05 14:16:51 -05:00
|
|
|
|
2013-08-25 04:57:12 -04:00
|
|
|
def user_from_params_or_current_user
|
|
|
|
params[:for_user_id] ? User.find(params[:for_user_id]) : current_user
|
|
|
|
end
|
|
|
|
|
2013-02-05 14:16:51 -05:00
|
|
|
def create
|
2013-06-20 15:47:35 -04:00
|
|
|
return fake_success_response if suspicious? params
|
2013-02-06 19:25:21 -05:00
|
|
|
|
2013-04-12 18:46:55 -04:00
|
|
|
user = User.new_from_params(params)
|
2013-10-21 14:49:51 -04:00
|
|
|
user.ip_address = request.ip
|
2013-08-18 10:37:38 -04:00
|
|
|
auth = authenticate_user(user, params)
|
2013-08-23 05:38:01 -04:00
|
|
|
register_nickname(user)
|
2013-02-05 14:16:51 -05:00
|
|
|
|
2013-10-19 05:48:11 -04:00
|
|
|
user.save ? user_create_successful(user, auth) : user_create_failed(user)
|
2013-02-05 14:16:51 -05:00
|
|
|
|
2013-03-07 14:56:28 -05:00
|
|
|
rescue ActiveRecord::StatementInvalid
|
2013-04-12 18:46:55 -04:00
|
|
|
render json: { success: false, message: I18n.t("login.something_already_taken") }
|
2013-10-19 05:48:11 -04:00
|
|
|
rescue DiscourseHub::NicknameUnavailable => e
|
2013-08-25 18:41:17 -04:00
|
|
|
render json: e.response_message
|
2013-02-05 14:16:51 -05:00
|
|
|
rescue RestClient::Forbidden
|
2013-04-12 18:46:55 -04:00
|
|
|
render json: { errors: [I18n.t("discourse_hub.access_token_problem")] }
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
2013-02-06 19:25:21 -05:00
|
|
|
def get_honeypot_value
|
|
|
|
render json: {value: honeypot_value, challenge: challenge_value}
|
|
|
|
end
|
|
|
|
|
2013-02-05 14:16:51 -05:00
|
|
|
def password_reset
|
|
|
|
expires_now()
|
|
|
|
|
|
|
|
@user = EmailToken.confirm(params[:token])
|
|
|
|
if @user.blank?
|
|
|
|
flash[:error] = I18n.t('password_reset.no_token')
|
2013-11-11 06:28:26 -05:00
|
|
|
elsif request.put?
|
|
|
|
raise Discourse::InvalidParameters.new(:password) unless params[:password].present?
|
2013-10-07 06:19:45 -04:00
|
|
|
@user.password = params[:password]
|
|
|
|
logon_after_password_reset if @user.save
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
2013-03-22 14:08:11 -04:00
|
|
|
render layout: 'no_js'
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
2013-02-07 10:45:24 -05:00
|
|
|
|
2013-10-07 06:19:45 -04:00
|
|
|
def logon_after_password_reset
|
2013-11-11 12:51:14 -05:00
|
|
|
message = if Guardian.new(@user).can_access_forum?
|
|
|
|
# Log in the user
|
|
|
|
log_on_user(@user)
|
|
|
|
'password_reset.success'
|
|
|
|
else
|
|
|
|
@requires_approval = true
|
|
|
|
'password_reset.success_unapproved'
|
|
|
|
end
|
|
|
|
|
|
|
|
flash[:success] = I18n.t(message)
|
|
|
|
end
|
2013-10-07 06:19:45 -04:00
|
|
|
|
2013-02-05 14:16:51 -05:00
|
|
|
def change_email
|
2013-06-06 03:14:32 -04:00
|
|
|
params.require(:email)
|
2013-02-05 14:16:51 -05:00
|
|
|
user = fetch_user_from_params
|
2013-09-07 22:42:41 -04:00
|
|
|
guardian.ensure_can_edit_email!(user)
|
2013-04-27 23:02:23 -04:00
|
|
|
lower_email = Email.downcase(params[:email]).strip
|
2013-02-05 14:16:51 -05:00
|
|
|
|
|
|
|
# Raise an error if the email is already in use
|
2013-04-14 20:20:33 -04:00
|
|
|
if User.where("email = ?", lower_email).exists?
|
2013-04-12 18:46:55 -04:00
|
|
|
raise Discourse::InvalidParameters.new(:email)
|
|
|
|
end
|
2013-02-05 14:16:51 -05:00
|
|
|
|
2013-04-14 20:20:33 -04:00
|
|
|
email_token = user.email_tokens.create(email: lower_email)
|
2013-04-12 18:46:55 -04:00
|
|
|
Jobs.enqueue(
|
|
|
|
:user_email,
|
2013-04-14 20:20:33 -04:00
|
|
|
to_address: lower_email,
|
2013-04-12 18:46:55 -04:00
|
|
|
type: :authorize_email,
|
|
|
|
user_id: user.id,
|
|
|
|
email_token: email_token.token
|
|
|
|
)
|
2013-02-05 14:16:51 -05:00
|
|
|
|
2013-02-07 10:45:24 -05:00
|
|
|
render nothing: true
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def authorize_email
|
|
|
|
expires_now()
|
|
|
|
if @user = EmailToken.confirm(params[:token])
|
|
|
|
log_on_user(@user)
|
|
|
|
else
|
|
|
|
flash[:error] = I18n.t('change_email.error')
|
|
|
|
end
|
2013-03-22 14:08:11 -04:00
|
|
|
render layout: 'no_js'
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def activate_account
|
|
|
|
expires_now()
|
|
|
|
if @user = EmailToken.confirm(params[:token])
|
|
|
|
|
|
|
|
# Log in the user unless they need to be approved
|
2013-04-03 12:23:28 -04:00
|
|
|
if Guardian.new(@user).can_access_forum?
|
2013-02-05 14:16:51 -05:00
|
|
|
@user.enqueue_welcome_message('welcome_user') if @user.send_welcome_message
|
|
|
|
log_on_user(@user)
|
2013-04-03 12:23:28 -04:00
|
|
|
else
|
|
|
|
@needs_approval = true
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
else
|
|
|
|
flash[:error] = I18n.t('activation.already_done')
|
|
|
|
end
|
2013-03-22 14:08:11 -04:00
|
|
|
render layout: 'no_js'
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
2013-02-22 11:49:48 -05:00
|
|
|
def send_activation_email
|
|
|
|
@user = fetch_user_from_params
|
|
|
|
@email_token = @user.email_tokens.unconfirmed.active.first
|
2013-10-07 06:19:45 -04:00
|
|
|
enqueue_activation_email if @user
|
2013-02-22 11:49:48 -05:00
|
|
|
render nothing: true
|
|
|
|
end
|
|
|
|
|
2013-10-07 06:19:45 -04:00
|
|
|
def enqueue_activation_email
|
|
|
|
@email_token ||= @user.email_tokens.create(email: @user.email)
|
|
|
|
Jobs.enqueue(:user_email, type: :signup, user_id: @user.id, email_token: @email_token.token)
|
|
|
|
end
|
|
|
|
|
2013-02-05 14:16:51 -05:00
|
|
|
def search_users
|
2013-02-07 05:59:25 -05:00
|
|
|
term = params[:term].to_s.strip
|
2013-02-05 14:16:51 -05:00
|
|
|
topic_id = params[:topic_id]
|
|
|
|
topic_id = topic_id.to_i if topic_id
|
|
|
|
|
2013-10-30 15:45:13 -04:00
|
|
|
results = UserSearch.new(term, topic_id).search
|
2013-02-05 14:16:51 -05:00
|
|
|
|
2013-10-30 15:45:13 -04:00
|
|
|
user_fields = [:username, :use_uploaded_avatar, :upload_avatar_template, :uploaded_avatar_id]
|
|
|
|
user_fields << :name if SiteSetting.enable_names?
|
|
|
|
|
|
|
|
render json: { users: results.as_json(only: user_fields, methods: :avatar_template) }
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
2013-08-14 06:20:05 -04:00
|
|
|
# [LEGACY] avatars in quotes/oneboxes might still be pointing to this route
|
|
|
|
# fixing it requires a rebake of all the posts
|
2013-08-13 16:08:29 -04:00
|
|
|
def avatar
|
2013-08-14 12:26:31 -04:00
|
|
|
user = User.where(username_lower: params[:username].downcase).first
|
2013-08-14 06:20:05 -04:00
|
|
|
if user.present?
|
|
|
|
size = determine_avatar_size(params[:size])
|
|
|
|
url = user.avatar_template.gsub("{size}", size.to_s)
|
|
|
|
expires_in 1.day
|
|
|
|
redirect_to url
|
|
|
|
else
|
|
|
|
raise ActiveRecord::RecordNotFound
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def determine_avatar_size(size)
|
|
|
|
size = size.to_i
|
|
|
|
size = 64 if size == 0
|
|
|
|
size = 10 if size < 10
|
|
|
|
size = 128 if size > 128
|
|
|
|
size
|
|
|
|
end
|
|
|
|
|
|
|
|
def upload_avatar
|
2013-08-13 16:08:29 -04:00
|
|
|
user = fetch_user_from_params
|
|
|
|
guardian.ensure_can_edit!(user)
|
|
|
|
|
|
|
|
file = params[:file] || params[:files].first
|
|
|
|
|
2013-10-18 10:33:19 -04:00
|
|
|
# Only allow url uploading for API users
|
|
|
|
# TODO: Does not protect from huge uploads
|
|
|
|
# https://github.com/discourse/discourse/pull/1512
|
2013-08-13 16:08:29 -04:00
|
|
|
# check the file size (note: this might also be done in the web server)
|
2013-11-11 12:51:14 -05:00
|
|
|
avatar = build_avatar_from(file)
|
|
|
|
avatar_policy = AvatarUploadPolicy.new(avatar)
|
2013-10-18 10:33:19 -04:00
|
|
|
|
2013-11-11 12:51:14 -05:00
|
|
|
if avatar_policy.too_big?
|
|
|
|
return render status: 413, text: I18n.t("upload.images.too_large",
|
|
|
|
max_size_kb: avatar_policy.max_size_kb)
|
|
|
|
end
|
2013-08-13 16:08:29 -04:00
|
|
|
|
2013-11-11 12:51:14 -05:00
|
|
|
raise FastImage::UnknownImageType unless SiteSetting.authorized_image?(avatar.file)
|
2013-08-13 16:08:29 -04:00
|
|
|
|
2013-11-11 12:51:14 -05:00
|
|
|
upload_avatar_for(user, avatar)
|
2013-08-13 16:08:29 -04:00
|
|
|
|
2013-10-18 10:33:19 -04:00
|
|
|
rescue Discourse::InvalidParameters
|
|
|
|
render status: 422, text: I18n.t("upload.images.unknown_image_type")
|
2013-08-13 16:08:29 -04:00
|
|
|
rescue FastImage::ImageFetchFailure
|
|
|
|
render status: 422, text: I18n.t("upload.images.fetch_failure")
|
|
|
|
rescue FastImage::UnknownImageType
|
|
|
|
render status: 422, text: I18n.t("upload.images.unknown_image_type")
|
|
|
|
rescue FastImage::SizeNotFound
|
|
|
|
render status: 422, text: I18n.t("upload.images.size_not_found")
|
|
|
|
end
|
|
|
|
|
|
|
|
def toggle_avatar
|
|
|
|
params.require(:use_uploaded_avatar)
|
|
|
|
user = fetch_user_from_params
|
|
|
|
guardian.ensure_can_edit!(user)
|
|
|
|
|
|
|
|
user.use_uploaded_avatar = params[:use_uploaded_avatar]
|
|
|
|
user.save!
|
|
|
|
|
2013-08-16 18:29:54 -04:00
|
|
|
render nothing: true
|
2013-08-13 16:08:29 -04:00
|
|
|
end
|
|
|
|
|
2013-02-05 14:16:51 -05:00
|
|
|
private
|
|
|
|
|
2013-02-06 19:25:21 -05:00
|
|
|
def honeypot_value
|
|
|
|
Digest::SHA1::hexdigest("#{Discourse.current_hostname}:#{Discourse::Application.config.secret_token}")[0,15]
|
|
|
|
end
|
|
|
|
|
|
|
|
def challenge_value
|
2013-08-23 02:19:23 -04:00
|
|
|
challenge = $redis.get('SECRET_CHALLENGE')
|
|
|
|
unless challenge && challenge.length == 16*2
|
|
|
|
challenge = SecureRandom.hex(16)
|
|
|
|
$redis.set('SECRET_CHALLENGE',challenge)
|
|
|
|
end
|
|
|
|
|
|
|
|
challenge
|
2013-02-06 19:25:21 -05:00
|
|
|
end
|
|
|
|
|
2013-06-05 14:08:21 -04:00
|
|
|
def suspicious?(params)
|
|
|
|
honeypot_or_challenge_fails?(params) || SiteSetting.invite_only?
|
|
|
|
end
|
|
|
|
|
2013-06-20 15:47:35 -04:00
|
|
|
def fake_success_response
|
2013-06-05 14:08:21 -04:00
|
|
|
render(
|
|
|
|
json: {
|
|
|
|
success: true,
|
|
|
|
active: false,
|
|
|
|
message: I18n.t("login.activate_email", email: params[:email])
|
|
|
|
}
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
2013-04-12 18:46:55 -04:00
|
|
|
def honeypot_or_challenge_fails?(params)
|
|
|
|
params[:password_confirmation] != honeypot_value ||
|
|
|
|
params[:challenge] != challenge_value.try(:reverse)
|
|
|
|
end
|
|
|
|
|
|
|
|
def valid_session_authentication?(auth, email)
|
|
|
|
auth && auth[:email] == email && auth[:email_valid]
|
|
|
|
end
|
|
|
|
|
|
|
|
def create_third_party_auth_records(user, auth)
|
2013-08-23 02:20:43 -04:00
|
|
|
return unless auth && auth[:authenticator_name]
|
2013-08-25 16:18:07 -04:00
|
|
|
|
2013-08-23 02:20:43 -04:00
|
|
|
authenticator = Users::OmniauthCallbacksController.find_authenticator(auth[:authenticator_name])
|
|
|
|
authenticator.after_create_account(user, auth)
|
2013-04-12 18:46:55 -04:00
|
|
|
end
|
2013-08-18 00:43:59 -04:00
|
|
|
|
2013-08-23 05:38:01 -04:00
|
|
|
def register_nickname(user)
|
|
|
|
if user.valid? && SiteSetting.call_discourse_hub?
|
|
|
|
DiscourseHub.register_nickname(user.username, user.email)
|
|
|
|
end
|
|
|
|
end
|
2013-10-19 05:48:11 -04:00
|
|
|
|
|
|
|
def user_create_successful(user, auth)
|
|
|
|
activator = UserActivator.new(user, request, session, cookies)
|
|
|
|
create_third_party_auth_records(user, auth)
|
|
|
|
|
|
|
|
# Clear authentication session.
|
|
|
|
session[:authentication] = nil
|
|
|
|
render json: { success: true, active: user.active?, message: activator.activation_message }
|
|
|
|
end
|
|
|
|
|
|
|
|
def user_create_failed(user)
|
|
|
|
render json: {
|
|
|
|
success: false,
|
|
|
|
message: I18n.t("login.errors", errors: user.errors.full_messages.join("\n")),
|
|
|
|
errors: user.errors.to_hash,
|
|
|
|
values: user.attributes.slice("name", "username", "email")
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def authenticate_user(user, params)
|
|
|
|
auth = session[:authentication]
|
|
|
|
user.active = true if valid_session_authentication?(auth, params[:email])
|
|
|
|
user.password_required! unless auth
|
|
|
|
auth
|
|
|
|
end
|
|
|
|
|
2013-11-11 12:51:14 -05:00
|
|
|
def build_avatar_from(file)
|
|
|
|
source = if file.is_a?(String)
|
|
|
|
is_api? ? :url : (raise FastImage::UnknownImageType)
|
|
|
|
else
|
|
|
|
:image
|
|
|
|
end
|
|
|
|
AvatarUploadService.new(file, source)
|
|
|
|
end
|
|
|
|
|
|
|
|
def upload_avatar_for(user, avatar)
|
|
|
|
upload = Upload.create_for(user.id, avatar.file, avatar.filesize)
|
|
|
|
user.upload_avatar(upload)
|
|
|
|
|
|
|
|
Jobs.enqueue(:generate_avatars, user_id: user.id, upload_id: upload.id)
|
|
|
|
render json: { url: upload.url, width: upload.width, height: upload.height }
|
|
|
|
end
|
|
|
|
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|