require 'rails_helper'
require 'i18n/backend/discourse_i18n'
require 'translation_override'

describe I18n::Backend::DiscourseI18n do

  let(:backend) { I18n::Backend::DiscourseI18n.new }

  before do
    I18n.reload!
    backend.store_translations(:en, :foo => 'Foo in :en', :bar => 'Bar in :en', :wat => "Hello %{count}")
    backend.store_translations(:en, :items => {:one => 'one item', :other => "%{count} items" })
    backend.store_translations(:de, :bar => 'Bar in :de')
    backend.store_translations(:'de-AT', :baz => 'Baz in :de-AT')
  end

  after do
    I18n.locale = :en
    I18n.reload!
  end

  it 'translates the basics as expected' do
    expect(backend.translate(:en, 'foo')).to eq("Foo in :en")
    expect(backend.translate(:en, 'items', count: 1)).to eq("one item")
    expect(backend.translate(:en, 'items', count: 3)).to eq("3 items")
    expect(backend.translate(:en, 'wat', count: 3)).to eq("Hello 3")
  end

  it 'can be searched by key or value' do
    expect(backend.search(:en, 'fo')).to eq({'foo' => 'Foo in :en'})
    expect(backend.search(:en, 'foo')).to eq({'foo' => 'Foo in :en' })
    expect(backend.search(:en, 'Foo')).to eq({'foo' => 'Foo in :en' })
    expect(backend.search(:en, 'hello')).to eq({'wat' => 'Hello %{count}' })
    expect(backend.search(:en, 'items.one')).to eq({'items.one' => 'one item' })
  end

  it 'can return multiple results' do
    results = backend.search(:en, 'item')

    expect(results['items.one']).to eq('one item')
    expect(results['items.other']).to eq('%{count} items')
  end

  it 'uses fallback locales for searching' do
    expect(backend.search(:de, 'bar')).to eq({'bar' => 'Bar in :de'})
    expect(backend.search(:de, 'foo')).to eq({'foo' => 'Foo in :en'})
  end

  describe '#exists?' do
    it 'returns true when a key is given that exists' do
      expect(backend.exists?(:de, :bar)).to eq(true)
    end

    it 'returns true when a key is given that exists in a fallback locale of the locale' do
      expect(backend.exists?(:de, :foo)).to eq(true)
    end

    it 'returns true when an existing key and an existing locale is given' do
      expect(backend.exists?(:en, :foo)).to eq(true)
      expect(backend.exists?(:de, :bar)).to eq(true)
      expect(backend.exists?(:'de-AT', :baz)).to eq(true)
    end

    it 'returns false when a non-existing key and an existing locale is given' do
      expect(backend.exists?(:en, :bogus)).to eq(false)
      expect(backend.exists?(:de, :bogus)).to eq(false)
      expect(backend.exists?(:'de-AT', :bogus)).to eq(false)
    end

    it 'returns true when a key is given which is missing from the given locale and exists in a fallback locale' do
      expect(backend.exists?(:de, :foo)).to eq(true)
      expect(backend.exists?(:'de-AT', :foo)).to eq(true)
    end

    it 'returns true when a key is given which is missing from the given locale and all its fallback locales' do
      expect(backend.exists?(:de, :baz)).to eq(false)
      expect(backend.exists?(:'de-AT', :bogus)).to eq(false)
    end
  end

  describe 'with overrides' do
    it 'returns the overridden key' do
      TranslationOverride.upsert!('en', 'foo', 'Overwritten foo')
      expect(I18n.translate('foo')).to eq('Overwritten foo')

      TranslationOverride.upsert!('en', 'foo', 'new value')
      expect(I18n.translate('foo')).to eq('new value')
    end

    it 'returns the overridden key after switching the locale' do
      TranslationOverride.upsert!('en', 'foo', 'Overwritten foo in EN')
      TranslationOverride.upsert!('de', 'foo', 'Overwritten foo in DE')

      expect(I18n.translate('foo')).to eq('Overwritten foo in EN')
      I18n.locale = :de
      expect(I18n.translate('foo')).to eq('Overwritten foo in DE')
    end

    it "can be searched" do
      TranslationOverride.upsert!('en', 'wat', 'Overwritten value')
      expect(I18n.search('wat', backend: backend)).to eq({'wat' => 'Overwritten value'})
      expect(I18n.search('Overwritten', backend: backend)).to eq({'wat' => 'Overwritten value'})
      expect(I18n.search('Hello', backend: backend)).to eq({})
    end

    it 'supports disabling' do
      orig_title = I18n.t('title')
      TranslationOverride.upsert!('en', 'title', 'overridden title')

      I18n.overrides_disabled do
        expect(I18n.translate('title')).to eq(orig_title)
      end
      expect(I18n.translate('title')).to eq('overridden title')
    end

    it 'supports interpolation' do
      TranslationOverride.upsert!('en', 'foo', 'hello %{world}')
      I18n.backend.store_translations(:en, foo: 'bar')
      expect(I18n.translate('foo', world: 'foo')).to eq('hello foo')
    end

    it 'supports interpolation named count' do
      TranslationOverride.upsert!('en', 'wat', 'goodbye %{count}')
      I18n.backend.store_translations(:en, wat: 'bar')
      expect(I18n.translate('wat', count: 123)).to eq('goodbye 123')
    end

    it 'ignores interpolation named count if it is not applicable' do
      TranslationOverride.upsert!('en', 'test', 'goodbye')
      I18n.backend.store_translations(:en, test: 'foo')
      I18n.backend.store_translations(:en, wat: 'bar')
      expect(I18n.translate('wat', count: 1)).to eq('bar')
    end

    it 'supports one and other' do
      TranslationOverride.upsert!('en', 'items.one', 'one fish')
      TranslationOverride.upsert!('en', 'items.other', '%{count} fishies')
      I18n.backend.store_translations(:en, items: { one: 'one item', other: "%{count} items" })
      expect(I18n.translate('items', count: 13)).to eq('13 fishies')
      expect(I18n.translate('items', count: 1)).to eq('one fish')
    end

    it 'supports one and other when only a single pluralization key is overridden' do
      TranslationOverride.upsert!('en', 'keys.magic.other', "no magic keys")
      I18n.backend.store_translations(:en, keys: { magic: { one: 'one magic key', other: "%{count} magic keys" } })
      expect(I18n.translate('keys.magic', count: 1)).to eq("one magic key")
      expect(I18n.translate('keys.magic', count: 2)).to eq("no magic keys")
    end

    it 'supports ActiveModel::Naming#human' do
      Fish = Class.new(ActiveRecord::Base)

      TranslationOverride.upsert!('en', 'fish', "fake fish")
      I18n.backend.store_translations(:en, fish: "original fish")

      expect(Fish.model_name.human).to eq('Fish')
    end

    describe "client json" do
      it "is empty by default" do
        expect(I18n.client_overrides_json('en')).to eq("{}")
      end

      it "doesn't return server overrides" do
        TranslationOverride.upsert!('en', 'foo', 'bar')
        expect(I18n.client_overrides_json('en')).to eq("{}")
      end

      it "returns client overrides" do
        TranslationOverride.upsert!('en', 'js.foo', 'bar')
        TranslationOverride.upsert!('en', 'admin_js.beep', 'boop')
        json = ::JSON.parse(I18n.client_overrides_json('en'))

        expect(json).to be_present
        expect(json['js.foo']).to eq('bar')
        expect(json['admin_js.beep']).to eq('boop')
      end
    end
  end

end