mirror of
https://github.com/oven-sh/bun
synced 2026-02-10 02:48:50 +00:00
### What does this PR do? ### How did you verify your code works? --------- Co-authored-by: autofix-ci[bot] <114827586+autofix-ci[bot]@users.noreply.github.com> Co-authored-by: Claude Bot <claude-bot@bun.sh> Co-authored-by: Claude <noreply@anthropic.com>
194 lines
6.6 KiB
TypeScript
194 lines
6.6 KiB
TypeScript
import { SQL, randomUUIDv7 } from "bun";
|
|
import { expect, test } from "bun:test";
|
|
import { describeWithContainer } from "harness";
|
|
|
|
describeWithContainer(
|
|
"mysql",
|
|
{
|
|
image: "mysql:8",
|
|
env: {
|
|
MYSQL_ROOT_PASSWORD: "bun",
|
|
},
|
|
},
|
|
(port: number) => {
|
|
const options = {
|
|
url: `mysql://root:bun@localhost:${port}`,
|
|
max: 1,
|
|
bigint: true,
|
|
};
|
|
|
|
test("Transaction works", async () => {
|
|
await using sql = new SQL(options);
|
|
const random_name = ("t_" + randomUUIDv7("hex").replaceAll("-", "")).toLowerCase();
|
|
await sql`CREATE TEMPORARY TABLE IF NOT EXISTS ${sql(random_name)} (a int)`;
|
|
|
|
await sql.begin(async sql => {
|
|
await sql`insert into ${sql(random_name)} values(1)`;
|
|
await sql`insert into ${sql(random_name)} values(2)`;
|
|
});
|
|
|
|
expect((await sql`select a from ${sql(random_name)}`).count).toBe(2);
|
|
await sql.close();
|
|
});
|
|
|
|
test("Throws on illegal transactions", async () => {
|
|
await using sql = new SQL({ ...options, max: 2 });
|
|
try {
|
|
await sql`BEGIN`;
|
|
expect.unreachable();
|
|
} catch (error) {
|
|
expect(error.code).toBe("ERR_MYSQL_UNSAFE_TRANSACTION");
|
|
}
|
|
});
|
|
|
|
test(".catch suppresses uncaught promise rejection", async () => {
|
|
await using sql = new SQL({ ...options, max: 2 });
|
|
const error = await sql`BEGIN`.catch(e => e);
|
|
return expect(error.code).toBe("ERR_MYSQL_UNSAFE_TRANSACTION");
|
|
});
|
|
|
|
test("Transaction throws", async () => {
|
|
await using sql = new SQL(options);
|
|
const random_name = ("t_" + randomUUIDv7("hex").replaceAll("-", "")).toLowerCase();
|
|
await sql`CREATE TEMPORARY TABLE IF NOT EXISTS ${sql(random_name)} (a int)`;
|
|
expect(
|
|
await sql
|
|
.begin(async sql => {
|
|
await sql`insert into ${sql(random_name)} values(1)`;
|
|
await sql`insert into ${sql(random_name)} values('hej')`;
|
|
})
|
|
.catch(e => e.message),
|
|
).toBe("Incorrect integer value: 'hej' for column 'a' at row 1");
|
|
});
|
|
|
|
test("Transaction rolls back", async () => {
|
|
await using sql = new SQL(options);
|
|
const random_name = ("t_" + randomUUIDv7("hex").replaceAll("-", "")).toLowerCase();
|
|
|
|
await sql`CREATE TEMPORARY TABLE IF NOT EXISTS ${sql(random_name)} (a int)`;
|
|
|
|
await sql
|
|
.begin(async sql => {
|
|
await sql`insert into ${sql(random_name)} values(1)`;
|
|
await sql`insert into ${sql(random_name)} values('hej')`;
|
|
})
|
|
.catch(() => {
|
|
/* ignore */
|
|
});
|
|
|
|
expect((await sql`select a from ${sql(random_name)}`).count).toBe(0);
|
|
});
|
|
|
|
test("Transaction throws on uncaught savepoint", async () => {
|
|
await using sql = new SQL(options);
|
|
const random_name = ("t_" + randomUUIDv7("hex").replaceAll("-", "")).toLowerCase();
|
|
await sql`CREATE TEMPORARY TABLE IF NOT EXISTS ${sql(random_name)} (a int)`;
|
|
expect(
|
|
await sql
|
|
.begin(async sql => {
|
|
await sql`insert into ${sql(random_name)} values(1)`;
|
|
await sql.savepoint(async sql => {
|
|
await sql`insert into ${sql(random_name)} values(2)`;
|
|
throw new Error("fail");
|
|
});
|
|
})
|
|
.catch(err => err.message),
|
|
).toBe("fail");
|
|
});
|
|
|
|
test("Transaction throws on uncaught named savepoint", async () => {
|
|
await using sql = new SQL(options);
|
|
const random_name = ("t_" + randomUUIDv7("hex").replaceAll("-", "")).toLowerCase();
|
|
await sql`CREATE TEMPORARY TABLE IF NOT EXISTS ${sql(random_name)} (a int)`;
|
|
expect(
|
|
await sql
|
|
.begin(async sql => {
|
|
await sql`insert into ${sql(random_name)} values(1)`;
|
|
await sql.savepoint("watpoint", async sql => {
|
|
await sql`insert into ${sql(random_name)} values(2)`;
|
|
throw new Error("fail");
|
|
});
|
|
})
|
|
.catch(() => "fail"),
|
|
).toBe("fail");
|
|
});
|
|
|
|
test("Transaction succeeds on caught savepoint", async () => {
|
|
await using sql = new SQL(options);
|
|
const random_name = ("t_" + randomUUIDv7("hex").replaceAll("-", "")).toLowerCase();
|
|
await sql`CREATE TABLE IF NOT EXISTS ${sql(random_name)} (a int)`;
|
|
try {
|
|
await sql.begin(async sql => {
|
|
await sql`insert into ${sql(random_name)} values(1)`;
|
|
await sql
|
|
.savepoint(async sql => {
|
|
await sql`insert into ${sql(random_name)} values(2)`;
|
|
throw new Error("please rollback");
|
|
})
|
|
.catch(() => {
|
|
/* ignore */
|
|
});
|
|
await sql`insert into ${sql(random_name)} values(3)`;
|
|
});
|
|
expect((await sql`select count(1) as count from ${sql(random_name)}`)[0].count).toBe(2);
|
|
} finally {
|
|
await sql`DROP TABLE IF EXISTS ${sql(random_name)}`;
|
|
}
|
|
});
|
|
|
|
test("Savepoint returns Result", async () => {
|
|
let result;
|
|
await using sql = new SQL(options);
|
|
await sql.begin(async t => {
|
|
result = await t.savepoint(s => s`select 1 as x`);
|
|
});
|
|
expect(result[0]?.x).toBe(1);
|
|
});
|
|
|
|
test("Uncaught transaction request errors bubbles to transaction", async () => {
|
|
await using sql = new SQL(options);
|
|
expect(await sql.begin(sql => [sql`select wat`, sql`select 1 as x, ${1} as a`]).catch(e => e.message)).toBe(
|
|
"Unknown column 'wat' in 'field list'",
|
|
);
|
|
});
|
|
|
|
test("Transaction rejects with rethrown error", async () => {
|
|
await using sql = new SQL(options);
|
|
expect(
|
|
await sql
|
|
.begin(async sql => {
|
|
try {
|
|
await sql`select exception`;
|
|
} catch (ex) {
|
|
throw new Error("WAT");
|
|
}
|
|
})
|
|
.catch(e => e.message),
|
|
).toBe("WAT");
|
|
});
|
|
|
|
test("Parallel transactions", async () => {
|
|
await using sql = new SQL({ ...options, max: 2 });
|
|
|
|
expect(
|
|
(await Promise.all([sql.begin(sql => sql`select 1 as count`), sql.begin(sql => sql`select 1 as count`)]))
|
|
.map(x => x[0].count)
|
|
.join(""),
|
|
).toBe("11");
|
|
});
|
|
|
|
test("Many transactions at beginning of connection", async () => {
|
|
await using sql = new SQL({ ...options, max: 2 });
|
|
const xs = await Promise.all(Array.from({ length: 30 }, () => sql.begin(sql => sql`select 1`)));
|
|
return expect(xs.length).toBe(30);
|
|
});
|
|
|
|
test("Transactions array", async () => {
|
|
await using sql = new SQL(options);
|
|
expect(
|
|
(await sql.begin(sql => [sql`select 1 as count`, sql`select 1 as count`])).map(x => x[0].count).join(""),
|
|
).toBe("11");
|
|
});
|
|
},
|
|
);
|