Skip to content

Latest commit

 

History

History
1299 lines (963 loc) · 41.5 KB

UPGRADE_NOTES.md

File metadata and controls

1299 lines (963 loc) · 41.5 KB

Upgrading from 0.22 to 0.23

For a full diff of necessary changes, please see LuckyDiff.

  • Upgrade Lucky CLI (homebrew)
brew update
brew upgrade lucky
  • Upgrade Lucky CLI (Linux)

Remove the existing Lucky binary and follow the Linux instructions in this section https://luckyframework.org/guides/getting-started/installing#on-linux

  • Update versions in shard.yml

    • Crystal should be 0.35.0
    • Lucky should be ~> 0.23.0
    • Authentic should be ~> 0.6.1
    • LuckyFlow should be ~> 0.7.0
    • jwt should be ~> 1.4.2
  • Run shards update

General updates

  • Update: params.get now strips white space. If you need the raw value, use params.get_raw.
  • Rename: mount to m in all pages that use components. Note: This was reverted in the next version
  • Update: all mounted components to use new signature mount(MyComponent.new(x: 1, y: 2)) -> m(MyComponent, x: 1, y:2).
  • Remove: Lucky::SessionHandler and Lucky::FlashHandler from src/app_server.cr

Optional updates

  • Add: Avram::RecordNotFoundError to the dont_report array in src/actions/errors/show.cr
  • Update: def render(error : Lucky::RouteNotFoundError to def render(error : Lucky::RouteNotFoundError | Avram::RecordNotFoundError) in src/actions/errors/show.cr.
  • Update: any CLI tasks that use ARGV to use the native args See implementation

Upgrading from 0.21 to 0.22

For a full diff of necessary changes, please see LuckyDiff.

  • Upgrade Lucky CLI (homebrew)
brew update
brew upgrade lucky
  • Upgrade Lucky CLI (Linux)

Remove the existing Lucky binary and follow the Linux instructions in this section https://luckyframework.org/guides/getting-started/installing#on-linux

  • Update versions in shard.yml

    • Crystal should be 0.35.0
    • Lucky should be ~> 0.22.0
    • Authentic should be ~> 0.6.0
    • jwt should be ~> 1.4.2
  • Run shards update

Upgrading from 0.20 to 0.21

For a full diff of necessary changes, please see LuckyDiff.

  • Upgrade Lucky CLI (homebrew)
brew update
brew upgrade lucky
  • Upgrade Lucky CLI (Linux)

Remove the existing Lucky binary and follow the Linux instructions in this section https://luckyframework.org/guides/getting-started/installing#on-linux

  • Update versions in shard.yml

    • Crystal should be 0.34.0
    • Lucky should be ~> 0.21.0
    • Authentic should be ~> 0.5.4
    • LuckyFlow should be ~> 0.6.3
  • Run shards update

General updates

  • Rename: config/logger.cr to config/log.cr
  • Update: config/log.cr to use the new Log. See implementation
  • Update: Procfile.dev and update the system_check to script/system_check && sleep 100000.
  • Update: all Lucky.logger.{level}("message") calls to use the new Crystal Log Log.{level} { "message" }
  • Remove: the following lines from config/database.cr
# Uncomment the next line to log all SQL queries
# settings.query_log_level = ::Logger::Severity::DEBUG

Updating Lucky.logger

Before this version, you would log data like this:

Lucky.logger.debug("Logging some message")
Lucky.logger.info({path: @context.request.path})

Now, you would write this like:

# Use the Crystal std-lib log for simple String messages
Log.debug { "Logging some message" }

# Use the Dexter extension for logging key/value data
Log.dexter.info { {path: @context.request.path} }

Upgrading from 0.19 to 0.20

For a full diff of necessary changes, please see LuckyDiff.

  • Update .crystal-version file to 0.34.0
  • Upgrade to crystal 0.34.0
  • Upgrade Lucky CLI (homebrew)
brew update
brew upgrade crystal-lang # Make sure you're up-to-date. Requires 0.34.0
brew upgrade lucky
  • Upgrade Lucky CLI (Linux)

Remove the existing Lucky binary and follow the Linux instructions in this section https://luckyframework.org/guides/getting-started/installing#on-linux

  • Update versions in shard.yml
    • Crystal should be 0.34.0
    • Lucky should be ~> 0.20.0
    • Authentic should be ~> 0.5.2
    • LuckyFlow should be ~> 0.6.2
  • Run shards update

General updates

  • Update: link no longer accepts a String path or URL, it must be an Action. Change link() to an a tag with an href (a "Google", href: "https://google.com"), or use an action class with link (link "Home", to: "/" becomes link("Home", to: Home::Index).
  • Remove: the ? from any needs using a predicate method. e.g. needs signed_in? : Bool -> needs signed_in : Bool. Lucky now automatically creates a method ending with ? for needs with a Bool type.
  • Update: your development ENV["PORT"] to be ENV["DEV_PORT"] if you need to customize the port your local server is running on.
  • Update: all SaveOperation classes where a raw hash is being passed in. e.g. MyOperation.new({"name" => "Gary"}) -> MyOperation.new(name: "Gary"), or if you must use a hash, wrap it in params first: MyOperation.new(Avram::Params.new({"name" => "Gary"})
  • Remove: the on: option from needs inside every Operation class. e.g. needs created_by : String, on: :create -> needs created_by : String. You will need to explicitly pass these when calling new, create, and update.

Optional updates

  • Update: all instance variables called from a needs on a page or component can now just use the method of that name. e.g. @current_user -> current_user
  • Add: include Lucky::CatchUnpermittedAttribute to the class Shared::Field(T) in src/components/shared/field.cr. This will raise a nicer error if you forget to permit a column in your SaveOperation
  • Add: the new Lucky::RemoteIpHandler.new to your app handlers in src/app_server.cr just before Lucky::RouteHandler.new.
  • Add: robots.txt to your public/ directory.
    User-agent: *
    Disallow:
    
  • Update: UserSerializer to inherit from the BaseSerializer if it doesn't already.
  • Add: cookie.http_only(true) to your config/cookies.cr file. This goes inside your settings.on_set block.
  • Update: your node dependencies where needed
  • Update: the setup script in script/setup. See implementation. Be sure to remove the ECR tags.
  • Add: this line system_check: script/system_check && $SHELL to your Procfile.dev
  • Add: the new system_check script in script/system_check. Note: you may need to chmod +x script/system_check. See implementation. Be sure to remove the ECR tags.
  • Add: the new function_helpers script in script/helpers/function_helpers. See implementation
  • Add: the new text_helpers script in script/helpers/text_helpers. See implementation

Upgrading from 0.18 to 0.19

For a full diff of necessary changes, please see LuckyDiff.

  • Update .crystal-version file to 0.33.0
  • Upgrade to crystal 0.33.0
  • Upgrade Lucky CLI (homebrew)
brew update
brew upgrade crystal-lang # Make sure you're up-to-date. Requires 0.33.0
brew upgrade lucky
  • Upgrade Lucky CLI (Linux)

Remove the existing Lucky binary and follow the Linux instructions in this section https://luckyframework.org/guides/getting-started/installing#on-linux

  • Update versions in shard.yml
    • Crystal should be 0.33.0
    • Lucky should be ~> 0.19.0
    • Authentic should be ~> 0.5.1
    • LuckyFlow should be ~> 0.6.2
  • Run shards update

Recommended (but optional) changes

GZip static assets

  • Add the compression plugin to package.json
  • Make these changes to your webpack.mix.js file
  • In src/app_server.cr add Lucky::StaticCompressionHandler.new("./public", file_ext: "gz", content_encoding: "gzip") above the Lucky::StaticFileHandler.new.

GZip text responses

Upgrading from 0.17 to 0.18

For a full diff of necessary changes, please see LuckyDiff.

  • Upgrade to crystal 0.31.1
  • Upgrade Lucky CLI (homebrew)
brew update
brew upgrade crystal-lang # Make sure you're up-to-date. Requires 0.31.1
brew upgrade lucky
  • Upgrade Lucky CLI (Linux)

Remove the existing Lucky binary and follow the Linux instructions in this section https://luckyframework.org/guides/getting-started/installing#on-linux

  • Update .crystal-version to 0.31.1

  • Update versions in shard.yml

    • Crystal should be 0.31.1
    • Lucky should be ~> 0.18
    • Authentic should be ~> 0.5.0
    • LuckyFlow should be ~> 0.6.0
  • Run shards update

General updates

  • Rename: all render calls in actions to html.

  • Update: the src/actions/errors/show.cr file to the new format:

    class Errors::Show < Lucky::ErrorAction
      DEFAULT_MESSAGE = "Something went wrong."
      default_format :html
      dont_report [Lucky::RouteNotFoundError]
    
      def render(error : Lucky::RouteNotFoundError)
        if html?
          error_html "Sorry, we couldn't find that page.", status: 404
        else
          error_json "Not found", status: 404
        end
      end
    
      # When the request is JSON and an InvalidOperationError is raised, show a
      # helpful error with the param that is invalid, and what was wrong with it.
      def render(error : Avram::InvalidOperationError)
        if html?
          error_html DEFAULT_MESSAGE, status: 500
        else
          error_json \
            message: error.renderable_message,
            details: error.renderable_details,
            param: error.invalid_attribute_name,
            status: 400
        end
      end
    
      # Always keep this below other 'render' methods or it may override your
      # custom 'render' methods.
      def render(error : Lucky::RenderableError)
        if html?
          error_html DEFAULT_MESSAGE, status: error.renderable_status
        else
          error_json error.renderable_message, status: error.renderable_status
        end
      end
    
      # If none of the 'render' methods return a response for the raised Exception,
      # Lucky will use this method.
      def default_render(error : Exception) : Lucky::Response
        if html?
          error_html DEFAULT_MESSAGE, status: 500
        else
          error_json DEFAULT_MESSAGE, status: 500
        end
      end
    
      private def error_html(message : String, status : Int)
        context.response.status_code = status
        html Errors::ShowPage, message: message, status: status
      end
    
      private def error_json(message : String, status : Int, details = nil, param = nil)
        json ErrorSerializer.new(message: message, details: details, param: param), status: status
      end
    
      private def report(error : Exception) : Nil
        # Send to Rollbar, send an email, etc.
      end
    end
  • Rename: title to message in src/pages/errors/show_page.cr.

  • Add: BaseSerializer to src/serializers/.

    abstract class BaseSerializer < Lucky::Serializer
      def self.for_collection(collection : Enumerable, *args, **named_args)
        collection.map do |object|
          new(object, *args, **named_args)
        end
      end
    end
  • Add: require "./serializers/base_serializer" to your src/app.cr above require "./serializers/**"

  • Optional: Update all serializers to inherit from BaseSerializer. Also merge Show/Index serializers in to a single class.

    # Merge these two classes
    class Users::IndexSerializer < Lucky::Serializer
    end
    
    class Users::ShowSerializers < Lucky::Serializer
    end
    
    # in to this class
    class UserSerializer < BaseSerializer
      # Same contents as Users::ShowSerializer
      # Calls to Users::IndexSerializer now become:
      #
      #    UserSerializer.for_collection(users)
    end
  • Rename: Errors::ShowSerializer to ErrorSerializer

  • Update: ErrorSerializer to inherit from the new BaseSerializer

  • Update: ErrorSerializer contents with:

    class ErrorSerializer < BaseSerializer
      def initialize(
        @message : String,
        @details : String? = nil,
        @param : String? = nil # If there was a problem with a specific param
      )
      end
    
      def render
        {message: @message, param: @param, details: @details}
      end
    end
    
  • Add: Avram::SchemaEnforcer.ensure_correct_column_mappings! to src/start_server.cr below Avram::Migrator::Runner.new.ensure_migrated!.

  • Update: any mention to renamed errors in this commit. You can likely ignore this as most people do not rescue these specific errors.

  • Add: accepted_formats [:json] to ApiAction in src/actions/api_action.cr.

    abstract class ApiAction < Lucky::Action
      accepted_formats [:json]
    end
  • Add: accepted_formats [:html, :json], default: :html to BrowserAction in src/actions/browser_action.cr

    abstract class BrowserAction < Lucky::Action
      accepted_formats [:html, :json], default: :html
    end
  • Update: src/app_server.cr with explicit return type on the middleware method.

# Add return type here
def middleware : Array(HTTP::Handler)
  [
    # ...
  ] of HTTP::Handler # Add this or app will fail to compile
end
  • Add: include Lucky::RequestExpectations to spec/spec_helper.cr below include Carbon::Expectations
  • Add: Avram::SchemaEnforcer.ensure_correct_column_mappings! to spec/spec_helper.cr below Avram::Migrator::Runner.new.ensure_migrated!
  • Update: Change at_exit do in spec/setup/start_app_server.cr to Spec.after_suite do

Upgrading from 0.16 to 0.17

  • Ensure you've upgraded to crystal 0.30.1
  • Upgrade Lucky CLI (homebrew)
brew update
brew upgrade crystal-lang # Make sure you're up-to-date. Requires 0.30.1
brew upgrade lucky
  • Upgrade Lucky CLI (Linux)

  • Update .crystal-version to 0.30.1

Remove the existing Lucky binary and follow the Linux instructions in this section https://luckyframework.org/guides/getting-started/installing#on-linux

  • Update versions in shard.yml
    • Lucky should be ~> 0.17
  • Run shards update

Example upgrade

If you're not sure about an upgrade step, or simply want to look at an example, see the lucky_bits upgrade.

General updates

  • Rename: Action rendering method text to plain_text.
  • Update: use of number_to_currency now returns a String instead of writing to the view directly.
  • Delete: config/static_file_handler.cr.
  • Add: a new Lucky::LogHandler configure to the bottom of config/logger.cr.
  • Update: Avram::Repo.configure to Avram.configure in config/logger.cr.
config/logger.cr
require "file_utils"

logger =
  if Lucky::Env.test?
    # Logs to `tmp/test.log` so you can see what's happening without having
    # a bunch of log output in your specs results.
    FileUtils.mkdir_p("tmp")
    Dexter::Logger.new(
      io: File.new("tmp/test.log", mode: "w"),
      level: Logger::Severity::DEBUG,
      log_formatter: Lucky::PrettyLogFormatter
    )
  elsif Lucky::Env.production?
    # This sets the log formatter to JSON so you can parse the logs with
    # services like Logentries or Logstash.
    #
    # If you want logs like in development use `Lucky::PrettyLogFormatter`.
    Dexter::Logger.new(
      io: STDOUT,
      level: Logger::Severity::INFO,
      log_formatter: Dexter::Formatters::JsonLogFormatter
    )
  else
    # For development, log everything to STDOUT with the pretty formatter.
    Dexter::Logger.new(
      io: STDOUT,
      level: Logger::Severity::DEBUG,
      log_formatter: Lucky::PrettyLogFormatter
    )
  end

Lucky.configure do |settings|
  settings.logger = logger
end

Lucky::LogHandler.configure do |settings|
  # Skip logging static assets in development
  if Lucky::Env.development?
    settings.skip_if = ->(context : HTTP::Server::Context) {
      context.request.method.downcase == "get" &&
      context.request.resource.starts_with?(/\/css\/|\/js\/|\/assets\//)
    }
  end
end

Avram.configure do |settings|
  settings.logger = logger
end
  • Update: script/setup to include the new postgres checks.
# This must go *after* the 'shards install' step
+ printf "\n▸ Checking that postgres is installed\n"
+ check_postgres | indent
+ printf "✔ Done\n" | indent

+ printf "\n▸ Verifying postgres connection\n"
+ lucky db.verify_connection | indent

printf "\n▸ Setting up the database\n"
lucky db.create | indent

Database updates

  • Add: a new AppDatabase class in src/app_database.cr that inherits from Avram::Database.
class AppDatabase < Avram::Database
end
  • Add: require "./app_database" to src/app.cr right below the require "./shards".
  • Rename: Avram::Repo.configure to AppDatabase.configure in config/database.cr.
  • Add: Avram.configure block.
config/database.cr
database_name = "..."

AppDatabase.configure do |settings|
  if Lucky::Env.production?
    settings.url = ENV.fetch("DATABASE_URL")
  else
    settings.url = ENV["DATABASE_URL"]? || Avram::PostgresURL.build(
      database: database_name,
      hostname: ENV["DB_HOST"]? || "localhost",
      # Some common usernames are "postgres", "root", or your system username (run 'whoami')
      username: ENV["DB_USERNAME"]? || "postgres",
      # Some Postgres installations require no password. Use "" if that is the case.
      password: ENV["DB_PASSWORD"]? || "postgres"
    )
  end
end

Avram.configure do |settings|
  settings.database_to_migrate = AppDatabase

  # this is moved from your old `Avram::Repo.configure` block.
  settings.lazy_load_enabled = Lucky::Env.production?
end
  • Move: the settings.lazy_load_enabled from AppDatabase.configure to Avram.configure block.
  • Add: a database class method to src/models/base_model.cr that returns AppDatabase.
abstract class BaseModel < Avram::Model
  def self.database : Avram::Database.class
    AppDatabase
  end
end
  • Update: Avram::Repo to AppDatabase in spec/setup/clean_database.cr.

  • Avram no longer automatically adds a timestamp and primary key to migrations. Add a primary key and timestamps to your old migrations.

    Also note that the syntax for a UUID has changed. You use primary_key id : UUID instead of an option on 'create'

    def migrate
      create :users do
        # Add these to your 'create' statements in your migrations
        primary_key id : Int64 # Or 'UUID' if using UUID
        add_timestamps
      end
    end
  • Note: Avram now defaults primary keys to Int64 instead of Int32. You can use the change_type macro to migrate your primary keys and foreign keys to Int64 if you need. Run lucky gen.migration UpdatePrimaryKeyTypes.

class UpdatePrimaryKeyTypesV20190723233131 < Avram::Migrator::Migration::V1
  def migrate
    alter table_for(User) do
      change_type id : Int64
    end
    alter table_for(Post) do
      change_type id : Int64
      change_type user_id : Int64
    end
  end
end
  • Update: models now default the primary key to Int64. This can be overridden if your tables uses a different column type for your primary keys, such as Int32 or UUID
abstract class BaseModel < Avram::Model
  macro default_columns
    primary_key id : UUID
    timestamps
  end
end

This also means that any model that uses UUID for a primary key can remove the primary_key_type option

class User < BaseModel
  # 0.16 and earlier
  table :users, primary_key_type: :uuid do
    column email : String
  end

  # Now with 0.17 it will use the 'default_columns' from the 'BaseModel'
  table :users do
    column email : String
  end
end

Updating queries

  • Rename: Query.new.destroy_all to Query.truncate. (e.g. UserQuery.new.destroy_all => UserQuery.truncate)
  • Rename: all association query methods from the association name to where_{association_name}. (e.g. UserQuery.new.posts => UserQuery.new.where_posts)
  • Update: all association query methods no longer take a block. Pass the query in as an argument. (e.g. UserQuery.new.posts { |post_query| } => UserQuery.new.where_posts(PostQuery.new))
  • Update: where_{association_name} methods no longer need to be preceded by a join_{assoc}, unless you need a custom join (i.e. left_join_{assoc}). If you use a custom join, you will need to add the auto_inner_join: false option to your where_{assoc} method.

Moving forms to operations

  • Rename: the src/forms directory to src/operations.
  • Update: require "./forms/mixins/**" and require "./forms/**" to require "./operations/mixins/**" and require "./operations/**" respectively in src/app.cr
  • Rename: BaseForm to SaveOperation in src/operations. (e.g. User::BaseForm => User::SaveOperation)
  • Rename: fillable to permit_columns
  • Rename: form class names to new naming convention. (e.g. class UserForm < User::SaveOperation => class SaveUser < User::SaveOperation). This step is optional, but still recommended to avoid future confusion.
  • Rename: Avram::VirtualForm to Avram::Operation.
  • Rename: virtual form class names to new naming convention VerbNoun. (e.g. class SignInForm < Avram::Operation => class SignInUser < Avram::Operation).
  • Rename: virtual to attribute.
  • Update: all SaveOperation classes to call before_save prepare. The prepare method is no longer called by default, which allows you to rename this method as well.
  • Update: FillableField to PermittedAttribute in src/components/shared/. Check field.cr and field_errors.cr.
  • Update: all authentic classes and modules to use new operation setup. This may require renaming some files to fit the VerbNoun verb_noun.cr convention.
Files in src/operations/
# src/operations/mixins/password_validations.cr
module PasswordValidations
+  macro included
+    before_save run_password_validations
+  end
  #...
end


# src/operations/request_password_reset.cr
- class RequestPasswordReset < Avram::VirtualForm
+ class RequestPasswordReset < Avram::Operation
  #...
end


# src/operations/reset_password.cr
- def prepare
-   run_password_validations
+ before do
    Authentic.copy_and_encrypt password, to: encrypted_password


# src/operations/sign_in_user.cr
- class SignInUser < Avram::VirtualOperation
+ class SignInUser < Avram::Operation


# src/operations/sign_up_user.cr
- def prepare
+ before_save do
    validate_uniqueness_of email
-   run_password_validations
  • Update sign_in_user.cr to match the new template.
  • Rename the FindAuthenticatable mixin to UserFromEmail, again the Lucky CLI template is a helpful guide.

Upgrading from 0.15 to 0.16

  • Upgrade to crystal 0.30.0

No updates to Lucky itself are required. There may be Crystal 0.30.0 related changes you may need to make.

Upgrading from 0.14 to 0.15

  • Upgrade to crystal 0.29.0
  • Upgrade Lucky CLI (macOS)
brew update
brew upgrade crystal-lang # Make sure you're up-to-date. Requires 0.29.0
brew upgrade lucky
  • Upgrade Lucky CLI (Linux)

  • Update .crystal-version to 0.29.0

Remove the existing Lucky binary and follow the Linux instructions in this section https://luckyframework.org/guides/getting-started/installing#on-linux

  • Update versions in shard.yml

    • Lucky should be ~> 0.15
  • Run shards update

  • Rename src/server.cr to src/start_server.cr.

  • Edit src/start_server.cr by changing

    • app to app_server and App to AppServer.
    • delete the line that starts with puts "Listening on
  • Update src/{your app name}.cr to require ./start_server

  • Rename src/dependencies.cr to src/shards.cr

  • Move the App class to a new file in src/app_server.cr

  • Rename App to AppServer and rename Lucky::BaseApp to Lucky::BaseAppServer in your new src/app_server.cr

  • Update src/app.cr to require new ./app_server file

  • Update src/app.cr to require new ./shards file

  • Replace usages of Lucky::Action::Status:: with the respective crystal HTTP::Status::

Upgrading from 0.13 to 0.14

  • Upgrade to crystal 0.28.0
  • Create new file config/force_ssl_handler.cr with the following content:
Lucky::ForceSSLHandler.configure do |settings|
  settings.enabled = Lucky::Env.production?
end

Upgrading from 0.12 to 0.13

  • Upgrade Lucky CLI (macOS)
brew update
brew upgrade crystal-lang # Make sure you're up-to-date. Requires 0.27.2
brew upgrade lucky
  • Upgrade Lucky CLI (Linux)

  • Update .crystal-version to 0.27.2

Remove the existing Lucky binary and follow the Linux instructions in this section https://luckyframework.org/guides/installing/#install-lucky

  • Update versions in shard.yml

    • Lucky should be ~> 0.13
    • LuckyFlow should be ~> 0.4
    • Authentic should be ~> 0.3
  • Run shards update

  • Find and replace LuckyRecord with Avram

  • Add Lucky::AssetHelpers.load_manifest below require "dependencies" in src/app.cr for browser apps. Skip for API only apps.

  • Query#preload with a query now includes the association name -> Query#preload_{{ assoc_name }}

  • Remove unexpose and unexpose_if_exposed from your actions. Pages now ignore unused exposures so these methods have been removed.

  • Change require "lucky_record" to require "avram" in src/dependencies

  • Rename config/log_handler.cr to config/logger.cr

  • Replace config/logger.cr with this:

require "file_utils"

logger =
  if Lucky::Env.test?
    # Logs to `tmp/test.log` so you can see what's happening without having
    # a bunch of log output in your specs results.
    FileUtils.mkdir_p("tmp")
    Dexter::Logger.new(
      io: File.new("tmp/test.log", mode: "w"),
      level: Logger::Severity::DEBUG,
      log_formatter: Lucky::PrettyLogFormatter
    )
  elsif Lucky::Env.production?
    # This sets the log formatter to JSON so you can parse the logs with
    # services like Logentries or Logstash.
    #
    # If you want logs like in development use `Lucky::PrettyLogFormatter`.
    Dexter::Logger.new(
      io: STDOUT,
      level: Logger::Severity::INFO,
      log_formatter: Dexter::Formatters::JsonLogFormatter
    )
  else
    # For development, log everything to STDOUT with the pretty formatter.
    Dexter::Logger.new(
      io: STDOUT,
      level: Logger::Severity::DEBUG,
      log_formatter: Lucky::PrettyLogFormatter
    )
  end

Lucky.configure do |settings|
  settings.logger = logger
end

Avram::Repo.configure do |settings|
  settings.logger = logger
end
  • If using is in queries, rename the calls to eq

  • App in src/app.cr should now inherit from Lucky::BaseApp. See the changes you need to make.

  • Move bin/setup to script/setup

  • In your README replace bin/setup with script/setup

  • Replace bin/lucky in your .gitignore with just /bin/. Lucky projects should now put bash scripts in /script. Binaries go in /bin/ and are ignored.

  • id in actions using route now have the underscored version of the resource name prepended. You'll need to rename your id calls to <resource_name>_id.

# Example from v0.12
class Users::Show < BrowserAction
  route do
    # Using the 'id' param
    UserQuery.find(id)
  end
end

# Would now be
class Users::Show < BrowserAction
  route do
    # Now it is 'user_id'
    UserQuery.find(user_id)
  end
end

And you should now be good to go!

Upgrading from 0.11 to 0.12

  • Upgrade Lucky CLI (macOS)
brew update
brew upgrade crystal-lang # Make sure you're up-to-date. Requires 0.27
brew upgrade lucky
  • Upgrade Lucky CLI (Linux)

Remove the existing Lucky binary and follow the Linux instructions in this section https://luckyframework.org/guides/installing/#install-lucky

Use your package manager to update Crystal to v0.27

  • In db/migrations, change LuckyMigrator::Migration -> LuckyRecord::Migrator::Migration for every migration

  • Remove lucky_migrator from shard.yml

  • Remove lucky_migrator from src/dependencies

  • Remove the LuckyMigrator.configure block from config/database.cr

  • Configuration now requires passing an argument. Find and replace .configure do with .configure do |settings| in all files in config

  • Update config/session.cr

    • Change Lucky::Session::Store.configure to Lucky::Session.configure do |settings|

    • Change your session key because signing/encryption has changed. For example: add _0_12_0 to the end of the key.

    • Remove settings.secret = Lucky::Server.settings.secret_key_base

  • If using cookies[] anywhere in your app, change the key you use. Lucky now signs and encrypts all cookies. Old cookies will not decrypt properly.

  • Change session[]= and cookies[]= to session|cookies.set|get

  • Change session|cookies.destroy to session/cookies.clear

  • cookies.unset(:key) and delete.unset(:key) should be cookies|session.delete(:key)

  • Remove unexpose current_user from src/actions/home/index.cr

  • Query#count has been renamed to Query#select_count. For example: UserQuery.new.count is now UserQuery.new.select_count

  • Change flash.danger to flash.failure in your actions.

  • Update Lucky::Flash::Handler to Lucky::FlashHandler in src/app.cr

  • Update usages of Lucky::Response to Lucky::TextResponse

  • Update usages of LuckyInflector::Inflector to Wordsmith::Inflector

  • Remove config/session.cr and copy config/cookies.cr

  • Replace config/email.cr with this one.

  • Add this line to spec_helper.cr (around line 19) -> LuckyRecord::Migrator::Runner.new.ensure_migrated!

  • In config/server.cr, copy the new block starting at line 15.

  • Update shard versions in shard.yml:

    • Lucky ~> 0.12
    • LuckyRecord ~> 0.7
    • Authentic ~> 0.2
    • LuckyFlow ~> 0.3
  • Change .crystal-version to 0.27.0

  • Run shards update to install the new shards

Upgrading from 0.10 to 0.11

  • Upgrade Lucky CLI (macOS)
brew update
brew upgrade crystal-lang # Make sure you're up-to-date
brew upgrade lucky
  • Upgrade Lucky CLI (Linux)

Remove the existing Lucky binary and follow the Linux instructions in this section https://luckyframework.org/guides/installing/#install-lucky

Use your package manager to update Crystal to v0.25

Upgrading from 0.8 to 0.10

Note: Lucky skipped version 0.9 so that Lucky and Lucky CLI are on the same version.

  • Upgrade Lucky CLI

On macOS:

brew update
brew upgrade crystal-lang # Make sure you're up-to-date
brew upgrade lucky

If you are on Linux, remove the existing Lucky binary and follow the Linux instructions in this section https://luckyframework.org/guides/installing/#install-lucky

  • View the upgrade diff and make changes to your app

In previous upgrade guides (below) every change is listed individually. This was time consuming and error-prone. Now, you can view all changes in this GitHub commit.

  • Ensure node version is at least 6.0 node -v. Install a newer version if yours is older.

  • Move files in src/pipes to src/actions/mixins

  • Change allow to fillable in forms

  • Change allow_virtual to virtual in forms

  • Run shards update

  • Run bin/setup to run new migrations, Laravel Mix and seeds file

If you have any problems or want to add extra details please open an issue or Pull Request. Thank you!

Upgrading from 0.7 to 0.8

  • Upgrade Lucky CLI

On macOS:

brew update
brew upgrade crystal-lang
brew upgrade lucky

If you are on Linux, remove the existing Lucky binary and follow the Linux instructions in this section: https://luckyframework.org/guides/installing/#install-lucky

  • Update dependencies in shard.yml
dependencies:
  lucky:
    github: luckyframework/lucky
    version: "~> 0.8.0"
  lucky_migrator:
    github: luckyframework/lucky_migrator
    version: ~> 0.4.0

Then run shards update

  • Update config/server.cr

You can probably copy this as-is, but if you have made customizations to your config/server.cr then you'll need to customize this:

Lucky::Server.configure do |settings|
  if Lucky::Env.production?
    settings.secret_key_base = secret_key_from_env
    settings.host = "0.0.0.0"
    settings.port = ENV["PORT"].to_i
  else
    settings.secret_key_base = "<%= secret_key_base %>"
    # Change host/port in config/watch.yml
    # Alternatively, you can set the PORT env to set the port
    settings.host = Lucky::ServerSettings.host
    settings.port = Lucky::ServerSettings.port
  end
end

private def secret_key_from_env
  ENV["SECRET_KEY_BASE"]? || raise_missing_secret_key_in_production
end

private def raise_missing_secret_key_in_production
  raise "Please set the SECRET_KEY_BASE environment variable. You can generate a secret key with 'lucky gen.secret_key'"
end
  • Add config/watch.yml

This is used by the watcher so it knows what port the server is running on.

host: 0.0.0.0
port: 5000
  • Update config/database.cr

Put this inside of the LuckyRecord::Repo.configure do |settings| block:

# In development and test, raise an error if you forget to preload associations
settings.lazy_load_enabled = Lucky::Env.production?

See a full example here: https://github.com/luckyframework/lucky_cli/blob/a25472cc7461b1803735d086e57a632f92f93a1c/src/web_app_skeleton/config/database.cr.ecr

  • You will need to preload associations now:

This will make N+1 queries a thing of the past.

# Will now raise a runtime error in dev/test
post = PostQuery.new.find(id)
post.comments # Must preload comments

# Now, you need to preload the comments
post = PostQuery.new.preload_comments.find(id)
post.comments
  • Rename field to column in your models. For example
class Post < BaseModel
  table :posts do
    column title : String # was "field title : String" previously
  end
end
  • Optionally include responsive_meta_tag in MainLayout

You can include this in head to make your app layout responsive.

  • Change abstract def inner to abstract def content in MainLayout

  • Change method call to inner to content in the render method of MainLayout

  • Change instances of def inner to def content in Pages

  • Change form needs to use on: :create

needs in forms should now use on: :save if you want the old behavior.

See https://luckyframework.org/guides/saving-with-forms/#passing-extra-data-to-forms for more info

  • Must pass extra params using create or update

You can no longer pass params to Form#new. You must pass them in the create or update.

UserForm.new(name: "Jane").save!
UserForm.create!(name: "Jane")

More info at https://luckyframework.org/guides/saving-with-forms/#passing-data-without-route-params

  • Change calls from form.save_succeeded? to form.saved?

  • Trap int in src/server.cr

Add this to your src/server.cr before server.listen

Signal::INT.trap do
  server.close
end
  • Add bin/lucky/ to .gitignore
# Add to .gitignore
bin/lucky/
  • Add nice HTML error page

Copy contents of the linked file to src/pages/errors/show_page.cr https://github.com/luckyframework/lucky_cli/blob/a25472cc7461b1803735d086e57a632f92f93a1c/src/web_app_skeleton/src/pages/errors/show_page.cr

  • Add default Error::ShowSerializer

This is used for serializing errors to JSON. Add this to src/serializers/errors/show_serializer.cr

# This is the default error serializer generated by Lucky.
# Feel free to customize it in any way you like.
class Errors::ShowSerializer < Lucky::Serializer
  def initialize(@message : String, @details : String? = nil)
  end

  def render
    {message: @message, details: @details}
  end
end
  • Update Errors::Show action

The error handling action now supports more errors and renders better output.

Copy the contents of the linked file to src/actions/errors/show.cr https://github.com/luckyframework/lucky_cli/blob/a25472cc7461b1803735d086e57a632f92f93a1c/src/web_app_skeleton/src/actions/errors/show.cr

  • Require serializers

Add the following to src/app.cr.

require "./serializers/**"

Upgrading from 0.6 to 0.7

  • Update to Crystal v0.24.1. Lucky will fail on earlier versions
brew update
brew upgrade crystal-lang
brew upgrade lucky

If you are on Linux, remove the existing Lucky binary and follow the Linux instructions in this section: https://luckyframework.org/guides/installing/#install-lucky

  • Update dependencies in shard.yml
dependencies:
  lucky:
    github: luckyframework/lucky
    version: "~> 0.7.0"
  lucky_migrator:
    github: luckyframework/lucky_migrator
    version: ~> 0.4.0

Then run shards update

  • Configure the domain to use for the RouteHelper:
# Add to config/route_helper.cr
Lucky::RouteHelper.configure do |settings|
  if Lucky::Env.production?
    # The APP_DOMAIN is something like https://myapp.com
    settings.domain = ENV.fetch("APP_DOMAIN")
  else
    settings.domain = "http:://localhost:3001"
  end
end
  • Add csrf_meta_tags to your MainLayout
# src/pages/main_layout.cr
# Somewhere in the head tag:
csrf_meta_tags
  • Remove needs flash from MainLayout
# Delete this line
needs flash : Lucky::Flash::Store
  • Remove expose flash from BrowserAction and add forgery protection
# src/actions/browser_action.cr
abstract class BrowserAction < Lucky::Action
  include Lucky::ProtectFromForgery
end
  • Change Shared::FlashComponent to get the flash from @context
# src/components/shared/flash_component.cr
# Change this:
@flash.each
# To:
@context.flash.each
  • Add *.dwarf to the .gitignore
# Add to .gitignore
*.dwarf