import {Fragment} from 'react'; import {render} from 'sentry-test/reactTestingLibrary'; import {SQLishFormatter} from 'sentry/views/starfish/utils/sqlish/SQLishFormatter'; describe('SQLishFormatter', function () { describe('SQLishFormatter.toString()', () => { const formatter = new SQLishFormatter(); it('Falls back to original string if unable to parse', () => { expect(formatter.toString('😤')).toEqual('😤'); }); it('Adds newlines for keywords in SELECTs', () => { expect( formatter.toString('SELECT hello FROM users ORDER BY name DESC LIMIT 1;') ).toMatchInlineSnapshot(` "SELECT hello FROM users ORDER BY name DESC LIMIT 1;" `); }); it('Adds newlines for keywords in INSERTs', () => { expect( formatter.toString('INSERT INTO users (id, name) VALUES (:c0, :c1) RETURNING *') ).toMatchInlineSnapshot(` "INSERT INTO users (id, name) VALUES ( :c0, :c1 ) RETURNING *" `); }); it('Adds indentation for keywords followed by parentheses', () => { expect( formatter.toString('SELECT * FROM (SELECT * FROM users))') ).toMatchInlineSnapshot(` "SELECT * FROM ( SELECT * FROM users ))" `); }); it('Capitalizes lowercase keywords', () => { expect(formatter.toString('select * from users;')).toMatchInlineSnapshot(` "SELECT * FROM users;" `); }); it('Adds indentation for SELECTS in conditions', () => { expect( formatter.toString( 'SELECT * FROM "sentry_users" WHERE (id IN (SELECT VO."id" FROM "sentry_vips" VO LIMIT 1)) AND (id IN (SELECT V1."id" FROM "sentry_currentusers" V1 LIMIT 1)) LIMIT 1' ) ).toMatchInlineSnapshot(` "SELECT * FROM "sentry_users" WHERE ( id IN ( SELECT VO."id" FROM "sentry_vips" VO LIMIT 1 ) ) AND (id IN ( SELECT V1."id" FROM "sentry_currentusers" V1 LIMIT 1 )) LIMIT 1" `); }); it('Reflows long lines to a max length', () => { expect( formatter.toString( 'SELECT "sentry_organization"."id", "sentry_organization"."name", "sentry_organization"."slug", "sentry_organization"."status", "sentry_organization"."date_added", "sentry_organization"."default_role", "sentry_organization"."is_test", "sentry_organization"."flags" FROM "sentry_organization" WHERE "sentry_organization"."id" = %s LIMIT 21' ) ).toMatchInlineSnapshot(` "SELECT "sentry_organization"."id", "sentry_organization"."name", "sentry_organization"."slug", "sentry_organization"."status", "sentry_organization"."date_added", "sentry_organization"."default_role", "sentry_organization"."is_test", "sentry_organization"."flags" FROM "sentry_organization" WHERE "sentry_organization"."id" = %s LIMIT 21" `); }); it('Reflows to specified width', () => { expect( formatter.toString( 'SELECT "sentry_organization"."id", "sentry_organization"."name", "sentry_organization"."slug", "sentry_organization"."status", "sentry_organization"."date_added" FROM "sentry_organization" WHERE "sentry_organization"."id" = %s LIMIT 21', {maxLineLength: 40} ) ).toMatchInlineSnapshot(` "SELECT "sentry_organization"."id", "sentry_organization"."name", "sentry_organization"."slug", "sentry_organization"."status", "sentry_organization"."date_added" FROM "sentry_organization" WHERE "sentry_organization"."id" = %s LIMIT 21" `); }); it('Reflows avoid unnecessary newlines', () => { expect( formatter.toString( 'SELECT "sentry_team"."org_role" FROM "sentry_team" INNER JOIN "sentry_organizationmember_teams" ON ("sentry_team"."id" = "sentry_organizationmember_teams"."team_id" WHERE ( "sentry_organizationmember_teams"."organizationmember_id" = %s AND NOT ("sentry_team"."org_role" IS NULL)' ) ).toMatchInlineSnapshot(` "SELECT "sentry_team"."org_role" FROM "sentry_team" INNER JOIN "sentry_organizationmember_teams" ON ("sentry_team"."id" = "sentry_organizationmember_teams"."team_id" WHERE ( "sentry_organizationmember_teams"."organizationmember_id" = %s AND NOT ("sentry_team"."org_role" IS NULL)" `); }); }); describe('SQLishFormatter.toSimpleMarkup()', () => { const formatter = new SQLishFormatter(); const getMarkup = (markup: any): string => { const {container} = render({markup}); return container.innerHTML; }; beforeEach(() => { // The renderer throws an error because elements in the list do not have // a `"key"` prop, but that's intentional. The list elements are spans // with no semantic meaning, and their keys are not meaningful. jest.spyOn(console, 'error').mockImplementation(jest.fn()); }); it('Capitalizes keywords', () => { expect(getMarkup(formatter.toSimpleMarkup('select hello'))).toMatchInlineSnapshot( `"SELECT hello"` ); }); it('Wraps every token in a `` element', () => { expect(getMarkup(formatter.toSimpleMarkup('SELECT hello;'))).toMatchInlineSnapshot( `"SELECT hello;"` ); }); }); });