import { parseTuples } from '@/store/modules/normalizers'
import _ from 'lodash'

describe('Parse tuples', () => {
  it('parses tuples', () => {
    const tuples = [
      { tuple: [':enabled', false]},
      { tuple: [':host', 'localhost']},
      { tuple: [':port', 389]},
      { tuple: [':tlsopts', ['test', 'test1']]},
      { tuple: [':sslopts', [{ tuple: [':cacertfile', 'path/to/file'] }, { tuple: [':verify', ':verify_peer'] }]]}
    ]
    const expectedResult = {
      ':enabled': false,
      ':host': 'localhost',
      ':port': 389,
      ':tlsopts': ['test', 'test1'],
      ':sslopts': { ':cacertfile': 'path/to/file', ':verify': ':verify_peer' }
    }

    const result = parseTuples(tuples, ':ldap')
    expect(_.isEqual(expectedResult, result)).toBeTruthy()
  })

  it('parses rate limiters setting', () => {
    const tuples = [
      { tuple: [':authentication', { tuple: [60000, 15] }]},
      { tuple: [':app_account_creation', [{ tuple: [100, 55] }, { tuple: [150, 10] }]]}
    ]
    const expectedResult = {
      ':authentication': [60000, 15],
      ':app_account_creation': [[100, 55], [150, 10]]
    }

    const result = parseTuples(tuples, ':rate_limit')
    expect(_.isEqual(expectedResult, result)).toBeTruthy()
  })

  it('parses icons setting', () => {
    const tuples = [
      { tuple: [':icons', [
        { ':src': '/static/logo.png', ':type': 'image/png' },
        { ':icon': '/test/test.png'}
      ]]}
    ]
    const expectedResult = [
      [{ key: ':src', value: '/static/logo.png' }, { key: ':type', value: 'image/png' }],
      [{ key: ':icon', value: '/test/test.png' }]
    ]

    const parsed = parseTuples(tuples, ':manifest')
    expect(typeof parsed).toBe('object')
    expect(':icons' in parsed).toBeTruthy()
    expect('id' in parsed[':icons'][0][0]).toBeTruthy()

    const result = parsed[':icons'].map(icon => {
      const iconWithoutId = icon.map(el => {
        const { id, ...rest } = el
        return rest
      })
      return iconWithoutId
    })
    expect(_.isEqual(expectedResult, result)).toBeTruthy()
  })

  it('parses retries setting', () => {
    const tuples = [
      { tuple: [':retries', [
        { tuple: [':federator_incoming', 5] },
        { tuple: [':federator_outgoing', 10] }
      ]]}
    ]

    const parsed = parseTuples(tuples, ':workers')
    expect(typeof parsed).toBe('object')
    expect(':retries' in parsed).toBeTruthy()
    expect(Array.isArray(parsed[':retries'])).toBeTruthy()
    expect(':federator_incoming' in parsed[':retries'][0]).toBeTruthy()
    expect('id' in parsed[':retries'][0][':federator_incoming']).toBeTruthy()
    expect(parsed[':retries'][0][':federator_incoming']['value']).toEqual(5)
  })

  it('parses objects', () => {
    const tuples = [
      { tuple: [':pleroma_fe', { ':alwaysShowSubjectInput': true, ':redirectRootNoLogin': '/main/all' }]},
      { tuple: [':masto_fe', { ':showInstanceSpecificPanel': true }]}
    ]
    const expectedResult = {
      ':masto_fe': { ':showInstanceSpecificPanel': true },
      ':pleroma_fe': { ':alwaysShowSubjectInput': true, ':redirectRootNoLogin': '/main/all' }
    }

    const result = parseTuples(tuples, ':frontend_configurations')
    expect(_.isEqual(expectedResult, result)).toBeTruthy()
  })

  it('parses ip', () => {
    const tuples = [
      { tuple: [':enabled', false]},
      { tuple: [':ip', { tuple: [0, 0, 0, 0] }]}
    ]
    const expectedResult = { ':enabled': false, ':ip': '0.0.0.0' }

    const result = parseTuples(tuples, ':gopher')
    expect(_.isEqual(expectedResult, result)).toBeTruthy()
  })

  it('parses prune setting that is a tuple', () => {
    const tuples = [
      { tuple: [':verbose', false]},
      { tuple: [':prune', { tuple: [':maxlen', 1500] }]},
      { tuple: [':queues', [
        { tuple: [':activity_expiration', 10]},
        { tuple: [':federator_incoming', 50]}
      ] ]}
    ]
    const expectedResult = {
      ':verbose': false,
      ':prune': [':maxlen', 1500],
      ':queues': { ':activity_expiration': 10, ':federator_incoming': 50 } }

    const result = parseTuples(tuples, 'Oban')
    expect(_.isEqual(expectedResult, result)).toBeTruthy()
  })

  it('parses prune setting that is an atom', () => {
    const tuples = [{ tuple: [':prune', ':disabled' ]}]
    const expectedResult = { ':prune': [':disabled'] }

    const result = parseTuples(tuples, 'Oban')
    expect(_.isEqual(expectedResult, result)).toBeTruthy()
  })

  it('parses mrf_keyword settings', () => {
    const tuples = [
      { tuple: [':reject', ['foo', '~r/foo/iu'] ]},
      { tuple: [':replace', [{ tuple: ['pattern', 'replacement']}, { tuple: ['foo', 'bar']}]]}
    ]
    const expectedResult = {
      ':reject': ['foo', '~r/foo/iu'],
      ':replace': [{ 'pattern': { value:'replacement' }}, { 'foo': { value:'bar' }}]
    }

    const parsed = parseTuples(tuples, ':mrf_keyword')
    expect(typeof parsed).toBe('object')
    expect(':reject' in parsed).toBeTruthy()
    expect(':replace' in parsed).toBeTruthy()

    const result = { ...parsed, ':replace': parsed[':replace'].map(el => {
      const key = Object.keys(el)[0]
      const { id, ...rest } = el[key]
      return { [key]: rest }
    })}
    expect(_.isEqual(expectedResult, result)).toBeTruthy()
  })

  it('parses assets settings', () => {
    const tuples = [
      { tuple: [':mascots', [
        { tuple: [':pleroma_fox_tan', { ':mime_type': 'image/png', ':url': '/images/pleroma-fox-tan-smol.png'}]},
        { tuple: [':pleroma_fox_tan_shy', { ':mime_type': 'image/png', ':url': '/images/pleroma-fox-tan-shy.png'}]}
      ]]},
      { tuple: [':default_mascot', ':pleroma_fox_tan']}
    ]
    const expectedResult = {
      ':default_mascot': ':pleroma_fox_tan',
      ':mascots': [
        { ':pleroma_fox_tan': { ':mime_type': 'image/png', ':url':'/images/pleroma-fox-tan-smol.png' }},
        { ':pleroma_fox_tan_shy': { ':mime_type': 'image/png', ':url':'/images/pleroma-fox-tan-shy.png' }}]
    }

    const parsed = parseTuples(tuples, ':assets')
    expect(typeof parsed).toBe('object')
    expect(':default_mascot' in parsed).toBeTruthy()
    expect(':mascots' in parsed).toBeTruthy()

    const result = { ...parsed, ':mascots': parsed[':mascots'].map(el => {
      const key = Object.keys(el)[0]
      const { id, ...rest } = el[key]
      return { [key]: rest }
    })}
    expect(_.isEqual(expectedResult, result)).toBeTruthy()
  })

  it('parses groups setting in emoji group', () => {
    const tuples = [{ tuple: [':groups', [{ tuple: [':Custom', ['/emoji/*.png', '/emoji/**/*.png']]}]]}]
    const expectedResult = { ':groups': [{ ':Custom': { value: ['/emoji/*.png', '/emoji/**/*.png']}}] }

    const parsed = parseTuples(tuples, ':emoji')

    expect(typeof parsed).toBe('object')
    expect(':groups' in parsed).toBeTruthy()
    const result = { ...parsed, ':groups': parsed[':groups'].map(el => {
      const key = Object.keys(el)[0]
      const { id, ...rest } = el[key]
      return { [key]: rest }
    })}
    expect(_.isEqual(expectedResult, result)).toBeTruthy()
  })

  it('parses crontab setting', () => {
    const tuples = [{ tuple: [':crontab', [
      { tuple: ['0 0 * * *', 'Pleroma.Workers.Cron.ClearOauthTokenWorker'] },
      { tuple: ['0 * * * *', 'Pleroma.Workers.Cron.StatsWorker'] },
      { tuple: ['* * * * *', 'Pleroma.Workers.Cron.PurgeExpiredActivitiesWorker']}
    ]]}]
    const expectedResult = { ':crontab': [
      { 'Pleroma.Workers.Cron.ClearOauthTokenWorker': { value: '0 0 * * *'}},
      { 'Pleroma.Workers.Cron.StatsWorker': { value: '0 * * * *'}},
      { 'Pleroma.Workers.Cron.PurgeExpiredActivitiesWorker': { value: '* * * * *'}}
    ]}

    const parsed = parseTuples(tuples, 'Oban')

    expect(typeof parsed).toBe('object')
    expect(':crontab' in parsed).toBeTruthy()
    const result = { ...parsed, ':crontab': parsed[':crontab'].map(el => {
      const key = Object.keys(el)[0]
      const { id, ...rest } = el[key]
      return { [key]: rest }
    })}
    expect(_.isEqual(expectedResult, result)).toBeTruthy()
  })

  it('parses match_actor setting in mrf_subchain group', () => {
    const tuples = [{ tuple: [':match_actor',
      { '~r/https:\/\/example.com/s': ['Elixir.Pleroma.Web.ActivityPub.MRF.DropPolicy']}]}]
    const expectedResult = { ':match_actor':
      [{ '~r/https:\/\/example.com/s': { value: ['Elixir.Pleroma.Web.ActivityPub.MRF.DropPolicy'] }}]}

    const parsed = parseTuples(tuples, ':mrf_subchain')

    expect(typeof parsed).toBe('object')
    expect(':match_actor' in parsed).toBeTruthy()
    const result = { ...parsed, ':match_actor': parsed[':match_actor'].map(el => {
      const key = Object.keys(el)[0]
      const { id, ...rest } = el[key]
      return { [key]: rest }
    })}
    expect(_.isEqual(expectedResult, result)).toBeTruthy()
  })

  it('parses options setting in MediaProxy.Invalidation.Http group', () => {
    const tuples = [{ tuple: [':options', [{ tuple: [':params', { xxx: 'zzz', aaa: 'bbb' }]}]]}]
    const expectedResult = { ':options': { ':params': 
      [ { xxx: { value: 'zzz' }}, { aaa: { value: 'bbb' }}]
    }}

    const parsed = parseTuples(tuples, 'Pleroma.Web.MediaProxy.Invalidation.Http')

    expect(typeof parsed).toBe('object')
    expect(':options' in parsed).toBeTruthy()
    
    const idRemoved = parsed[':options'][':params'].map(el => {
      const key = Object.keys(el)[0]
      const { id, ...rest } = el[key]
      return { [key]: rest }
    })
    parsed[':options'][':params'] = idRemoved
    expect(_.isEqual(expectedResult, parsed)).toBeTruthy()
  })

  it('parses sender setting in :welcome', () => {
    const senderEmpty = [{ tuple: [':sender', ''] }]
    const senderTuple = [{ tuple: [':sender', { tuple: ['test', 'test@email.com'] }]}]
    const senderString = [{ tuple: [':sender', 'test@email.com'] }]

    const expectedSenderEmpty = { ':sender': { email: '' }}
    const expectedSenderTuple = { ':sender': { email: 'test@email.com', nickname: 'test' }}
    const expectedSenderString = { ':sender': { email: 'test@email.com' }}

    expect(_.isEqual(expectedSenderEmpty, parseTuples(senderEmpty, ':welcome'))).toBeTruthy()
    expect(_.isEqual(expectedSenderTuple, parseTuples(senderTuple, ':welcome'))).toBeTruthy()
    expect(_.isEqual(expectedSenderString, parseTuples(senderString, ':welcome'))).toBeTruthy()
  })

  it('parses proxy_url', () => {
    const proxyUrlNull = [{ tuple: [':proxy_url', null] }]
    const proxyUrlTuple = [{ tuple: [':proxy_url', { tuple: [':socks5', ':localhost', 3090] }]}]
    const proxyUrlString = [{ tuple: [':proxy_url', 'localhost:9020'] }]

    const expectedProxyUrlNull = { ':proxy_url': { socks5: false, host: null, port: null }}
    const expectedProxyUrlTuple = { ':proxy_url': { socks5: true, host: ':localhost', port: 3090 }}
    const expectedProxyUrlString = { ':proxy_url': { socks5: false, host: 'localhost', port: '9020' }}

    expect(_.isEqual(expectedProxyUrlNull, parseTuples(proxyUrlNull, ':http'))).toBeTruthy()
    expect(_.isEqual(expectedProxyUrlTuple, parseTuples(proxyUrlTuple, ':http'))).toBeTruthy()
    expect(_.isEqual(expectedProxyUrlString, parseTuples(proxyUrlString, ':http'))).toBeTruthy()
  })

  it('parses args setting in Pleroma.Upload.Filter.Mogrify', () => {
    const tuples = [{ tuple: [':args', ['strip', '{ "implode", "1" }']]}]
    const expectedResult = { ':args': ['strip', '{ "implode", "1" }'] }

    const result = parseTuples(tuples, 'Pleroma.Upload.Filter.Mogrify')
    expect(_.isEqual(expectedResult, result)).toBeTruthy()
  })

  it('parses nested tuples', () => {
    const tuples = [{ tuple: [':proxy_opts', [
      { tuple: [':redirect_on_failure', false] },
      { tuple: [':max_body_length', 26214400] },
      { tuple: [':http', [
        { tuple: [':follow_redirect', true] },
        { tuple: [':pool', ':media'] }
      ]]},
    ]]}]
    const expectedResult = { ':proxy_opts': {
      ':redirect_on_failure': false,
      ':max_body_length': 26214400,
      ':http': {
        ':follow_redirect': true,
        ':pool': ':media'
      }
    }}
    const result = parseTuples(tuples, ':media_proxy')
    expect(_.isEqual(expectedResult, result)).toBeTruthy()
  })

  it('parses tuples with arrays', () => {
    const tuples = [{ tuple: [':ignore_hosts', []]}, { tuple: [':ignore_tld', ['local', 'localdomain', 'lan']]}]
    const expectedResult = { ':ignore_hosts': [], ':ignore_tld': ['local', 'localdomain', 'lan'] }

    const result = parseTuples(tuples, ':rich_media')
    expect(_.isEqual(expectedResult, result)).toBeTruthy()
  })
})