mirror of
https://github.com/indentlabs/notebook.git
synced 2025-10-26 11:19:22 +00:00
- Migrate from deprecated sources API to modern payment_methods API in SubscriptionsController - Update subscription plan modifications to use Subscription.modify instead of direct assignment - Fix payment method creation/deletion to use PaymentMethod.create/detach instead of sources - Update view templates to use new payment_methods data structure - Migrate price.id usage from deprecated plan.id in data integrity tasks - Add comprehensive test suite with proper Stripe API stubs - Add missing test gems: rspec-rails, webmock, factory_bot_rails, shoulda-matchers This resolves Error 500 when users try to upgrade to Premium subscriptions. The original error was: NoMethodError - undefined method 'total_count' for nil:NilClass caused by stripe_customer.sources.total_count when sources API returned nil. 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
249 lines
8.7 KiB
Ruby
249 lines
8.7 KiB
Ruby
class UsersController < ApplicationController
|
|
before_action :set_user, only: [:show, :followers, :following, :tag]
|
|
|
|
def index
|
|
redirect_to new_session_path(User)
|
|
end
|
|
|
|
def show
|
|
@sidenav_expansion = 'community'
|
|
|
|
@feed = ContentPageShare.where(user_id: @user.id)
|
|
.order('created_at DESC')
|
|
.includes([:content_page, :secondary_content_page, :user, :share_comments])
|
|
.limit(100)
|
|
|
|
@content = @user.public_content.select { |type, list| list.any? }
|
|
@tabs = @content.keys
|
|
|
|
# Get popular tags for this user's public content
|
|
@popular_tags = get_popular_public_tags_for_user(@user)
|
|
|
|
@favorite_content = @user.favorite_page_type? ? @user.send(@user.favorite_page_type.downcase.pluralize).is_public : []
|
|
@stream = @user.recent_content_list(limit: 20)
|
|
end
|
|
|
|
Rails.application.config.content_types[:all].each do |content_type|
|
|
content_type_name = content_type.name.downcase.pluralize.to_sym # :characters, :items, etc
|
|
define_method content_type_name do
|
|
set_user
|
|
|
|
# We double up on the same returns from set_user here since we're calling set_user manually instead of a before_action
|
|
return if @user.nil?
|
|
return if @user.private_profile?
|
|
|
|
@random_image_including_private_pool_cache = ImageUpload.where(
|
|
user_id: @user.id,
|
|
).group_by { |image| [image.content_type, image.content_id] }
|
|
|
|
@content_type = content_type
|
|
@content_list = @user.send(content_type_name).is_public.order(:name)
|
|
|
|
@saved_basil_commissions = BasilCommission.where(
|
|
entity_type: content_type_name,
|
|
entity_id: @content_list.pluck(:id)
|
|
).where.not(saved_at: nil)
|
|
.group_by { |commission| [commission.entity_type, commission.entity_id] }
|
|
|
|
render :content_list
|
|
end
|
|
end
|
|
|
|
def delete_my_account # :(
|
|
unless user_signed_in?
|
|
redirect_to(root_path, notice: "You must be signed in to do that!")
|
|
return
|
|
end
|
|
|
|
# Make sure the user is set to Starter on Stripe so we don't keep charging them
|
|
stripe_customer = Stripe::Customer.retrieve(current_user.stripe_customer_id)
|
|
stripe_subscription = stripe_customer.subscriptions.data[0]
|
|
if stripe_subscription
|
|
# Update subscription to starter plan using modern API
|
|
Stripe::Subscription.modify(stripe_subscription.id, {
|
|
items: [{
|
|
id: stripe_subscription.items.data[0].id,
|
|
price: 'starter'
|
|
}]
|
|
})
|
|
end
|
|
|
|
report_user_deletion_to_slack(current_user)
|
|
|
|
current_user.avatar.purge
|
|
|
|
# Immediately mark the user as deleted and inactive
|
|
current_user.update(deleted_at: DateTime.current)
|
|
|
|
# Destroy the user and all of its content
|
|
# TODO this can take quite a while for active users, so it should be moved to a background job
|
|
current_user.destroy!
|
|
|
|
redirect_to(root_path, notice: 'Your account has been deleted. We will miss you greatly!')
|
|
end
|
|
|
|
def report_user_deletion_to_slack user
|
|
return unless Rails.env == 'production'
|
|
slack_hook = ENV['SLACK_HOOK']
|
|
return unless slack_hook
|
|
|
|
notifier = Slack::Notifier.new slack_hook,
|
|
channel: '#analytics',
|
|
username: 'tristan'
|
|
|
|
notifier.ping ":bomb: :bomb: :bomb: #{user.email.split('@').first}@ (##{user.id}) just deleted their account."
|
|
end
|
|
|
|
def followers
|
|
end
|
|
|
|
def following
|
|
end
|
|
|
|
def tag
|
|
@tag_slug = params[:tag_slug]
|
|
@tag = PageTag.find_by(user_id: @user.id, slug: @tag_slug)
|
|
|
|
if @tag.nil?
|
|
redirect_url = @user.username.present? ? profile_by_username_path(username: @user.username) : user_path(@user)
|
|
return redirect_to(redirect_url, notice: 'That tag does not exist.', status: :not_found)
|
|
end
|
|
|
|
# Find all public content with this tag
|
|
@tagged_content = []
|
|
|
|
# Go through each content type and find items with this tag
|
|
Rails.application.config.content_types[:all].each do |content_type|
|
|
content_pages = content_type.joins(:page_tags)
|
|
.where(privacy: 'public')
|
|
.where(user_id: @user.id)
|
|
.where(page_tags: { slug: @tag_slug })
|
|
.order(:name)
|
|
|
|
@tagged_content << {
|
|
type: content_type.name,
|
|
icon: content_type.icon,
|
|
color: content_type.color,
|
|
content: content_pages
|
|
} if content_pages.any?
|
|
end
|
|
|
|
# Add documents and timelines if they have the tag
|
|
# Handle documents separately since they don't use the common content type structure
|
|
documents = Document.joins(:page_tags)
|
|
.where(privacy: 'public')
|
|
.where(user_id: @user.id)
|
|
.where(page_tags: { slug: @tag_slug })
|
|
.order(:title) # Documents use 'title' instead of 'name'
|
|
|
|
@tagged_content << {
|
|
type: 'Document',
|
|
icon: 'description',
|
|
color: 'blue',
|
|
content: documents
|
|
} if documents.any?
|
|
|
|
# Handle timelines separately since they don't use the common content type structure
|
|
timelines = Timeline.joins(:page_tags)
|
|
.where(privacy: 'public')
|
|
.where(user_id: @user.id)
|
|
.where(page_tags: { slug: @tag_slug })
|
|
.order(:name)
|
|
|
|
@tagged_content << {
|
|
type: 'Timeline',
|
|
icon: 'timeline',
|
|
color: 'blue',
|
|
content: timelines
|
|
} if timelines.any?
|
|
|
|
# Get images for content cards
|
|
@random_image_including_private_pool_cache = ImageUpload.where(
|
|
user_id: @user.id,
|
|
).group_by { |image| [image.content_type, image.content_id] }
|
|
|
|
# Collect all content IDs and types for fetching basil commissions
|
|
basil_entity_types = []
|
|
basil_entity_ids = []
|
|
|
|
@tagged_content.each do |content_group|
|
|
content_group[:content].each do |content|
|
|
basil_entity_types << content.class.name
|
|
basil_entity_ids << content.id
|
|
end
|
|
end
|
|
|
|
# Initialize @saved_basil_commissions if there are any content items
|
|
if basil_entity_types.any?
|
|
@saved_basil_commissions = BasilCommission.where(
|
|
entity_type: basil_entity_types,
|
|
entity_id: basil_entity_ids
|
|
).where.not(saved_at: nil)
|
|
.group_by { |commission| [commission.entity_type, commission.entity_id] }
|
|
end
|
|
|
|
@sidenav_expansion = 'community'
|
|
end
|
|
|
|
private
|
|
|
|
def set_user
|
|
@user = User.find_by(user_params)
|
|
return redirect_to(root_path, notice: 'That user does not exist.', status: :not_found) if @user.nil?
|
|
return redirect_to(root_path, notice: 'That user has chosen to hide their profile.') if @user.private_profile?
|
|
return redirect_to(root_path, notice: 'That user has had their profile hidden.') if @user.thredded_user_detail.moderation_state == 'blocked'
|
|
|
|
@accent_color = @user.favorite_page_type_color
|
|
@accent_icon = @user.favorite_page_type_icon
|
|
end
|
|
|
|
def user_params
|
|
params.permit(:id, :username)
|
|
end
|
|
|
|
# Get most popular tags for a user's public content
|
|
def get_popular_public_tags_for_user(user, limit: 10)
|
|
# Find page tags attached to public content
|
|
public_page_tags = []
|
|
|
|
# For each content type, find public pages with tags
|
|
Rails.application.config.content_types[:all].each do |content_type|
|
|
# Skip if a user has no pages of this type
|
|
next unless user.respond_to?(content_type.name.downcase.pluralize)
|
|
|
|
# Get all public pages of this content type
|
|
public_pages = user.send(content_type.name.downcase.pluralize).is_public
|
|
|
|
# Skip if there are no public pages
|
|
next if public_pages.empty?
|
|
|
|
# Find all page tags for these pages
|
|
tag_ids = PageTag.where(page_type: content_type.name, page_id: public_pages.pluck(:id)).pluck(:id)
|
|
public_page_tags.concat(tag_ids) if tag_ids.any?
|
|
end
|
|
|
|
# Also include Document and Timeline tags if they're public
|
|
public_documents = user.documents.where(privacy: 'public')
|
|
if public_documents.any?
|
|
document_tag_ids = PageTag.where(page_type: 'Document', page_id: public_documents.pluck(:id)).pluck(:id)
|
|
public_page_tags.concat(document_tag_ids) if document_tag_ids.any?
|
|
end
|
|
|
|
public_timelines = user.timelines.where(privacy: 'public')
|
|
if public_timelines.any?
|
|
timeline_tag_ids = PageTag.where(page_type: 'Timeline', page_id: public_timelines.pluck(:id)).pluck(:id)
|
|
public_page_tags.concat(timeline_tag_ids) if timeline_tag_ids.any?
|
|
end
|
|
|
|
# If we have tags, find the most popular ones
|
|
return [] if public_page_tags.empty?
|
|
|
|
# Get the actual tags
|
|
PageTag.where(id: public_page_tags)
|
|
.select('tag, slug, COUNT(*) as usage_count')
|
|
.group(:tag, :slug)
|
|
.order('usage_count DESC')
|
|
.limit(limit)
|
|
end
|
|
end
|