decko-commons/decko

View on GitHub
mod/account/spec/set/right/account_spec.rb

Summary

Maintainability
A
0 mins
Test Coverage
# -*- encoding : utf-8 -*-

RSpec.describe Card::Set::Right::Account do
  describe "#create" do
    let(:dummy_account_args) do
      {
        name: "TmpUser",
        "+*account" => {
          "+*email" => "tmpuser@decko.org",
          "+*password" => "Tmp_pas5!"
        }
      }
    end

    context "valid user" do
      # NOTE: - much of this is tested in account_request_spec
      before { Card::Auth.as_bot { user_card } }
      let(:user_card) { Card.create! dummy_account_args.merge(type_code: :user) }

      it "creates an authenticable password" do
        validity = Card::Auth.password_valid? user_card.account, "Tmp_pas5!"
        expect(validity).to be_truthy
      end
    end

    it "checks accountability of 'accounted' card" do
      expect(Card.create(dummy_account_args).errors["+*account"].first)
        .to match(/You don't have permission to create/)
    end

    it "works for any card with +*account permissions -- not just User type" do
      Card::Auth.as_bot do
        Card.create! name: Card::Name[%i[basic account type_plus_right create]],
                     content: "Anyone Signed In"
      end

      expect(Card.create(dummy_account_args).errors).to be_empty
    end

    it "requires email" do
      no_email = Card.create(
        name: "TmpUser",
        type_id: Card::UserID,
        "+*account" => { "+*password" => "tmp_pass" }
      )
      expect(no_email.errors["+*account"].first).to match(/email required/)
    end
  end

  describe "#send_verification_email" do
    before do
      @email = "joe@user.com"
      @account = Card::Auth.find_account_by_email(@email)
      Mail::TestMailer.deliveries.clear
      @account.send_verification_email
      @mail = Mail::TestMailer.deliveries.last
    end

    it "has correct address" do
      expect(@mail.to).to eq([@email])
    end

    it "contains deck title" do
      body = @mail.parts[0].body.raw_source
      expect(body).to match(Card::Rule.global_setting(:title))
    end

    it "contains link to verify account" do
      raw_source = @mail.parts[0].body.raw_source
      ["/update/#{@account.name.url_key}",
       "card%5Btrigger%5D=verify_and_activate",
       "token="].each do |url_part|
        expect(raw_source).to include(url_part)
      end
    end

    it "contains expiry days" do
      msg = "valid for #{Card.config.token_expiry / 1.day} days"
      expect(@mail.parts[0].body.raw_source).to include(msg)
    end
  end

  describe "#verify_and_activate" do
    it "activates account" do
      user = Card.create!(
        name: "TmpUser",
        type_id: Card::UserID,
        "+*account" => { "+*password" => "new Pas5word!",
                         "+*email" => "tmp@decko.org",
                         "+*status" => "unverified" }
      )

      Card::Env.params[:token] = Card::Auth::Token.encode user.id, anonymous: true
      user.account.update! trigger: :verify_and_activate

      expect(user.account).to be_active
    end
  end

  describe "#send_password_reset_email" do
    before do
      @email = "joe@user.com"
      @account = Card::Auth.find_account_by_email(@email)
      Mail::TestMailer.deliveries = []
      @account.send_password_reset_email
      @mail = Mail::TestMailer.deliveries.last
    end

    it "contains deck title" do
      body = @mail.parts[0].body.raw_source
      expect(body).to match(Card::Rule.global_setting(:title))
    end

    it "contains password reset link" do
      raw_source = @mail.parts[0].body.raw_source
      ["/update/#{@account.left.name.url_key}",
       "token=",
       "card%5Btrigger%5D=reset_password"].each do |url_part|
        expect(raw_source).to include(url_part)
      end
    end

    it "contains expiry days" do
      url = "valid for #{Card.config.token_expiry / 1.day} days"
      expect(@mail.parts[0].body.raw_source).to include(url)
    end
  end

  describe "#update" do
    before do
      @account = Card::Auth.find_account_by_email("joe@user.com")
    end

    it "resets password" do
      @account.password_card.update!(content: "new Pas5word!")
      authenticated = Card::Auth.authenticate "joe@user.com", "new Pas5word!"
      assert_equal @account, authenticated
    end

    it "does not rehash password when updating email" do
      @account.email_card.update! content: "joe2@user.com"
      authenticated = Card::Auth.authenticate "joe2@user.com", "joe_pass"
      assert_equal @account, authenticated
    end
  end

  describe "#reset_password" do
    before do
      @email = "joe@user.com"
      @account = Card::Auth.find_account_by_email(@email)
      Card::Auth.signin Card::AnonymousID
    end

    def trigger_reset
      @account.update! trigger: :reset_password
    end

    def auth_token extra_payload={}
      Card::Env.params[:token] = Card::Auth::Token.encode @account.left_id, extra_payload
    end

    it "authenticates with correct token" do
      auth_token
      expect(Card::Auth.current_id).to eq(Card::AnonymousID)
      trigger_reset
      expect(Card::Auth.current_id).to eq(@account.left_id)
    end

    it "does not work if token is expired" do
      auth_token exp: 1.days.ago.to_i
      expect { trigger_reset }.to raise_error(/Signature has expired/)
      # user notified of expired token
    end

    it "does not work if token is wrong" do
      Card::Env.params[:token] = "#{auth_token}xxx"
      expect { trigger_reset }.to raise_error(/Signature verification failed/)
    end
  end
end