Skip to content

Building a Blog with Plutonium

Quick Links:

This tutorial walks through building a blog application using Plutonium. You'll learn how to:

  • Set up authentication using Rodauth
  • Create a blog feature package
  • Build a dashboard/portal
  • Implement posts and comments
  • Add interactive actions
  • Configure scoping and authorization

[Rest of the tutorial content remains exactly the same...]

Initial Setup

Let's start by creating a new Rails application with Plutonium:

bash
rails new plutonium_app -a propshaft -j esbuild -c tailwind \
  -m https://radioactive-labs.github.io/plutonium-core/templates/plutonium.rb

This creates a new Rails application with:

  • Propshaft as the asset pipeline
  • esbuild for JavaScript bundling
  • Tailwind CSS for styling
  • Plutonium's base configuration

Setting Up Authentication

Next, let's add authentication using Rodauth:

bash
# Generate Rodauth configuration
rails generate pu:rodauth:install

# Generate user account setup
rails generate pu:rodauth:account user
rails db:migrate

This sets up:

  • A User model with authentication
  • Login/logout functionality
  • Account verification via email
  • Password reset capabilities

Plutonium Login Page

The key files created include:

  • app/models/user.rb - The User model
  • app/rodauth/user_rodauth_plugin.rb - Rodauth configuration
  • app/controllers/rodauth_controller.rb - Base controller for auth pages
  • Database migrations for users and auth-related tables

Creating the Blog Feature Package

Let's create a dedicated package for our blogging functionality:

bash
rails generate pu:pkg:package blogging

This generates a new feature package under packages/blogging/ with:

  • Base controllers
  • Model and policy base classes
  • Package-specific views directory
  • Engine configuration

Adding Posts Resource

Now we can add our first resource - blog posts:

bash
rails generate pu:res:scaffold post user:belongs_to title:string \
  content:text 'published_at:datetime?'

TIP

Unlike Rails, Plutonium generates fields as non-null by default. Appending ? to the type e.g. published_at:datetime? makes published_at a nullable datetime.

When prompted, select the blogging feature package.

This creates:

  • Post model with associations
  • Policy class with basic permissions
  • Controllers for CRUD operations
  • Database migration

The generated post model includes:

ruby
class Blogging::Post < Blogging::ResourceRecord
  belongs_to :user
  validates :title, presence: true
  validates :content, presence: true
end

Remember to apply the new migrations:

bash
rails db:migrate

Creating the Dashboard Portal

Let's add a portal to manage our blog:

bash
rails generate pu:pkg:portal dashboard

When prompted, select user for authentication.

This creates a new portal package under packages/dashboard_portal/ with:

  • Portal-specific controllers
  • Resource presentation layer
  • Dashboard views
  • Authenticated routes

Configure Routes

Let's configure our application routes to properly handle authentication and dashboard access:

ruby
# ==> Redirects

# Redirect to home after login.
create_account_redirect "/"
create_account_redirect "/dashboard"

# Redirect to home after login.
login_redirect "/"
login_redirect "/dashboard"

# Redirect to home page after logout.
logout_redirect "/"
logout_redirect "/dashboard"

# Redirect to wherever login redirects to after account verification.
verify_account_redirect { login_redirect }
ruby
Rails.application.routes.draw do
  # Other routes...

  # Defines the root path route ("/")
  # root "posts#index"
  root to: redirect("/dashboard") 
end

These changes ensure that:

  • Users are directed to the dashboard after signing up
  • Users are directed to the dashboard after logging in
  • Users are directed to the dashboard after logging out
  • The root URL (/) redirects to the dashboard
  • Account verification follows the same flow as login

Connecting Posts to Dashboard

We can now connect our blog posts to the dashboard:

bash
rails generate pu:res:conn

Select:

  1. Source feature: blogging
  2. Resources: Blogging::Post
  3. Portal: dashboard_portal

This configures the routing and controllers to display posts in the dashboard.

Plutonium Posts Dashboard

Adding Comments

Let's add commenting functionality:

bash
rails generate pu:res:scaffold comment blogging/post:belongs_to \
  user:belongs_to body:text

rails db:migrate

rails generate pu:res:conn

TIP

Note the use of the namespaced model in the association blogging/post.

Plutonium has inbuilt support for handling namespaces, generating:

ruby
# model
belongs_to :post, class_name: "Blogging::Post"

# migration
t.belongs_to :post, null: false, foreign_key: {:to_table=>:blogging_posts}

This creates:

  • Comment model with associations
  • Policy controlling access
  • CRUD interface in dashboard
  • Proper routing configuration

Customizing the Interface

Post Table Customization

We can customize how posts appear in the table view:

ruby
# packages/blogging/app/policies/blogging/post_policy.rb
def permitted_attributes_for_index
  [:user, :title, :published_at, :created_at]
end

Plutonium Posts Dashboard (Customized)

Post Detail View

Customize the post detail layout using display grids:

ruby
# packages/blogging/app/definitions/blogging/post_definition.rb
class Blogging::PostDefinition < Blogging::ResourceDefinition
  display :user, wrapper: {class: "col-span-full"}
  display :title, wrapper: {class: "col-span-full"}
  display :content, wrapper: {class: "col-span-full"}
  display :published_at, wrapper: {class: "col-span-full"}
end

Plutonium Posts Detail (Customized)

Adding Publishing Functionality

Let's add the ability to publish posts:

ruby
# packages/blogging/app/definitions/blogging/post_definition.rb
action :publish,
       interaction: Blogging::Posts::Publish,
       collection_record_action: false # do not show this on the table
ruby
# packages/blogging/app/interactions/blogging/posts/publish.rb
module Blogging
  module Posts
    class Publish < ResourceInteraction
      presents label: "Publish Post", icon: Phlex::TablerIcons::Send
      attribute :resource

      def execute
        if resource.update(published_at: Time.current)
          succeed(resource)
            .with_message("Post was successfully published")
        else
          failed(resource.errors)
        end
      end
    end
  end
end
ruby
# packages/blogging/app/policies/blogging/post_policy.rb

def publish?
  !record.published_at 
end


def permitted_attributes_for_create
  [:user, :title, :content, :published_at] 
  [:user, :title, :content] 
end

Plutonium Publish Post

Adding Comments Panel

Enable viewing comments on posts:

ruby
# packages/blogging/app/models/blogging/post.rb
has_many :comments
ruby
# packages/blogging/app/policies/blogging/post_policy.rb
def permitted_associations
  %i[comments]
end

Plutonium Comments Panel

Scoping Resources

Ensure users only see their own content:

ruby
# packages/dashboard_portal/lib/engine.rb
module DashboardPortal
  class Engine < Rails::Engine
    include Plutonium::Portal::Engine

    config.after_initialize do
      scope_to_entity User, strategy: :current_user
    end
  end
end

Adding Nested Comments

Enable adding comments directly from the post form:

ruby
# packages/blogging/app/definitions/blogging/post_definition.rb
nested_input :comments,
  using: Blogging::CommentDefinition,
  fields: %i[user body]
ruby
# packages/blogging/app/models/blogging/post.rb
has_many :comments
accepts_nested_attributes_for :comments
ruby
# packages/blogging/app/policies/blogging/post_policy.rb
def permitted_attributes_for_create
  [:user, :title, :content] 
  [:user, :title, :content, :comments] 
end

Plutonium Nested Comments

Running the Application

  1. Start the Development server:
bash
bin/dev
  1. Visit http://localhost:3000

  2. Create a user account and start managing your blog!

What's Next?

Some ideas for extending the application:

  • Add categories/tags for posts
  • Implement comment moderation
  • Add rich text editing for post content
  • Create a public-facing blog view
  • Add image attachments for posts
  • Implement social sharing

Conclusion

In this tutorial, we've built a full-featured blog application with:

  • User authentication
  • Post management
  • Commenting system
  • Publishing workflow
  • Proper authorization
  • User-scoped content

Plutonium helped us quickly scaffold and connect the pieces while maintaining clean separation of concerns through its package system.

Released under the MIT License.