fix: Updating store currencies (#984)
* fix: Adds default currency to store currencies on create * fix: Adds integration tests + currency update error handling * add test * Lowercase currency codes * fix tests * fix: Await drop database to stop Jest from complaining * revert itnegration tests fix * fix store tests
This commit is contained in:
committed by
GitHub
parent
3bf32e5dc9
commit
59bb413245
@@ -0,0 +1,144 @@
|
|||||||
|
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
||||||
|
|
||||||
|
exports[`/admin/store POST /admin/store fails to update default currency if not in store currencies 1`] = `
|
||||||
|
Object {
|
||||||
|
"message": "Store does not have currency: eur",
|
||||||
|
"type": "invalid_data",
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
exports[`/admin/store POST /admin/store successfully updates and store currencies 1`] = `
|
||||||
|
Object {
|
||||||
|
"created_at": Any<String>,
|
||||||
|
"currencies": Array [
|
||||||
|
Object {
|
||||||
|
"code": "jpy",
|
||||||
|
"name": "Japanese Yen",
|
||||||
|
"symbol": "¥",
|
||||||
|
"symbol_native": "¥",
|
||||||
|
},
|
||||||
|
Object {
|
||||||
|
"code": "usd",
|
||||||
|
"name": "US Dollar",
|
||||||
|
"symbol": "$",
|
||||||
|
"symbol_native": "$",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"default_currency_code": "usd",
|
||||||
|
"id": Any<String>,
|
||||||
|
"invite_link_template": null,
|
||||||
|
"metadata": null,
|
||||||
|
"name": "Medusa Store",
|
||||||
|
"payment_link_template": null,
|
||||||
|
"swap_link_template": null,
|
||||||
|
"updated_at": Any<String>,
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
exports[`/admin/store POST /admin/store successfully updates default currency and store currencies 1`] = `
|
||||||
|
Object {
|
||||||
|
"created_at": Any<String>,
|
||||||
|
"currencies": Array [
|
||||||
|
Object {
|
||||||
|
"code": "jpy",
|
||||||
|
"name": "Japanese Yen",
|
||||||
|
"symbol": "¥",
|
||||||
|
"symbol_native": "¥",
|
||||||
|
},
|
||||||
|
Object {
|
||||||
|
"code": "usd",
|
||||||
|
"name": "US Dollar",
|
||||||
|
"symbol": "$",
|
||||||
|
"symbol_native": "$",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"default_currency": Object {
|
||||||
|
"code": "jpy",
|
||||||
|
"name": "Japanese Yen",
|
||||||
|
"symbol": "¥",
|
||||||
|
"symbol_native": "¥",
|
||||||
|
},
|
||||||
|
"default_currency_code": "jpy",
|
||||||
|
"id": Any<String>,
|
||||||
|
"invite_link_template": null,
|
||||||
|
"metadata": null,
|
||||||
|
"name": "Medusa Store",
|
||||||
|
"payment_link_template": null,
|
||||||
|
"swap_link_template": null,
|
||||||
|
"updated_at": Any<String>,
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
exports[`/admin/store POST /admin/store successfully updates default currency code 1`] = `
|
||||||
|
Object {
|
||||||
|
"created_at": Any<String>,
|
||||||
|
"currencies": Array [
|
||||||
|
Object {
|
||||||
|
"code": "usd",
|
||||||
|
"name": "US Dollar",
|
||||||
|
"symbol": "$",
|
||||||
|
"symbol_native": "$",
|
||||||
|
},
|
||||||
|
Object {
|
||||||
|
"code": "dkk",
|
||||||
|
"name": "Danish Krone",
|
||||||
|
"symbol": "Dkr",
|
||||||
|
"symbol_native": "kr",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"default_currency": Object {
|
||||||
|
"code": "dkk",
|
||||||
|
"name": "Danish Krone",
|
||||||
|
"symbol": "Dkr",
|
||||||
|
"symbol_native": "kr",
|
||||||
|
},
|
||||||
|
"default_currency_code": "dkk",
|
||||||
|
"id": Any<String>,
|
||||||
|
"invite_link_template": null,
|
||||||
|
"metadata": null,
|
||||||
|
"name": "Medusa Store",
|
||||||
|
"payment_link_template": null,
|
||||||
|
"swap_link_template": null,
|
||||||
|
"updated_at": Any<String>,
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
exports[`/admin/store Store creation has created store with default currency 1`] = `
|
||||||
|
Object {
|
||||||
|
"created_at": Any<String>,
|
||||||
|
"currencies": Array [
|
||||||
|
Object {
|
||||||
|
"code": "usd",
|
||||||
|
"name": "US Dollar",
|
||||||
|
"symbol": "$",
|
||||||
|
"symbol_native": "$",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"default_currency": Object {
|
||||||
|
"code": "usd",
|
||||||
|
"name": "US Dollar",
|
||||||
|
"symbol": "$",
|
||||||
|
"symbol_native": "$",
|
||||||
|
},
|
||||||
|
"default_currency_code": "usd",
|
||||||
|
"fulfillment_providers": Array [
|
||||||
|
Object {
|
||||||
|
"id": "test-ful",
|
||||||
|
"is_installed": true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"id": Any<String>,
|
||||||
|
"invite_link_template": null,
|
||||||
|
"metadata": null,
|
||||||
|
"name": "Medusa Store",
|
||||||
|
"payment_link_template": null,
|
||||||
|
"payment_providers": Array [
|
||||||
|
Object {
|
||||||
|
"id": "test-pay",
|
||||||
|
"is_installed": true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"swap_link_template": null,
|
||||||
|
"updated_at": Any<String>,
|
||||||
|
}
|
||||||
|
`;
|
||||||
221
integration-tests/api/__tests__/admin/store.js
Normal file
221
integration-tests/api/__tests__/admin/store.js
Normal file
@@ -0,0 +1,221 @@
|
|||||||
|
const { Store } = require("@medusajs/medusa")
|
||||||
|
const path = require("path")
|
||||||
|
|
||||||
|
const setupServer = require("../../../helpers/setup-server")
|
||||||
|
const { useApi } = require("../../../helpers/use-api")
|
||||||
|
const { initDb, useDb } = require("../../../helpers/use-db")
|
||||||
|
|
||||||
|
const adminSeeder = require("../../helpers/admin-seeder")
|
||||||
|
|
||||||
|
jest.setTimeout(30000)
|
||||||
|
|
||||||
|
describe("/admin/store", () => {
|
||||||
|
let dbConnection
|
||||||
|
const cwd = path.resolve(path.join(__dirname, "..", ".."))
|
||||||
|
|
||||||
|
beforeAll(async () => {
|
||||||
|
dbConnection = await initDb({ cwd })
|
||||||
|
})
|
||||||
|
|
||||||
|
afterAll(async () => {
|
||||||
|
const db = useDb()
|
||||||
|
await db.shutdown()
|
||||||
|
})
|
||||||
|
|
||||||
|
describe("Store creation", () => {
|
||||||
|
let medusaProcess
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
await adminSeeder(dbConnection)
|
||||||
|
medusaProcess = await setupServer({ cwd })
|
||||||
|
})
|
||||||
|
|
||||||
|
afterEach(async () => {
|
||||||
|
const db = useDb()
|
||||||
|
db.teardown()
|
||||||
|
medusaProcess.kill()
|
||||||
|
})
|
||||||
|
|
||||||
|
it("has created store with default currency", async () => {
|
||||||
|
const api = useApi()
|
||||||
|
|
||||||
|
const response = await api.get("/admin/store", {
|
||||||
|
headers: { Authorization: "Bearer test_token " },
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(response.status).toEqual(200)
|
||||||
|
expect(response.data.store).toMatchSnapshot({
|
||||||
|
id: expect.any(String),
|
||||||
|
name: "Medusa Store",
|
||||||
|
currencies: [
|
||||||
|
{
|
||||||
|
code: "usd",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
default_currency_code: "usd",
|
||||||
|
created_at: expect.any(String),
|
||||||
|
updated_at: expect.any(String),
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe("POST /admin/store", () => {
|
||||||
|
let medusaProcess
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
await adminSeeder(dbConnection)
|
||||||
|
medusaProcess = await setupServer({ cwd })
|
||||||
|
|
||||||
|
const manager = dbConnection.manager
|
||||||
|
const store = await manager.findOne(Store, { name: "Medusa Store" })
|
||||||
|
await manager.query(
|
||||||
|
`INSERT INTO store_currencies (store_id, currency_code) VALUES ('${store.id}', 'dkk')`
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
afterEach(async () => {
|
||||||
|
const db = useDb()
|
||||||
|
await db.teardown()
|
||||||
|
medusaProcess.kill()
|
||||||
|
})
|
||||||
|
|
||||||
|
it("fails to update default currency if not in store currencies", async () => {
|
||||||
|
const api = useApi()
|
||||||
|
|
||||||
|
try {
|
||||||
|
await api.post(
|
||||||
|
"/admin/store",
|
||||||
|
{
|
||||||
|
default_currency_code: "eur",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
headers: { Authorization: "Bearer test_token " },
|
||||||
|
}
|
||||||
|
)
|
||||||
|
} catch (e) {
|
||||||
|
expect(e.response.data).toMatchSnapshot({
|
||||||
|
type: "invalid_data",
|
||||||
|
message: "Store does not have currency: eur",
|
||||||
|
})
|
||||||
|
expect(e.response.status).toBe(400)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
it("fails to remove default currency from currencies without replacing it", async () => {
|
||||||
|
const api = useApi()
|
||||||
|
|
||||||
|
try {
|
||||||
|
await api.post(
|
||||||
|
"/admin/store",
|
||||||
|
{
|
||||||
|
currencies: ["usd"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
headers: { Authorization: "Bearer test_token " },
|
||||||
|
}
|
||||||
|
)
|
||||||
|
} catch (e) {
|
||||||
|
expect(e.response.data).toMatchSnapshot({
|
||||||
|
type: "invalid_data",
|
||||||
|
message:
|
||||||
|
"You are not allowed to remove default currency from store currencies without replacing it as well",
|
||||||
|
})
|
||||||
|
expect(e.response.status).toBe(400)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
it("successfully updates default currency code", async () => {
|
||||||
|
const api = useApi()
|
||||||
|
|
||||||
|
const response = await api.post(
|
||||||
|
"/admin/store",
|
||||||
|
{
|
||||||
|
default_currency_code: "dkk",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
headers: { Authorization: "Bearer test_token " },
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
expect(response.status).toEqual(200)
|
||||||
|
expect(response.data.store).toMatchSnapshot({
|
||||||
|
id: expect.any(String),
|
||||||
|
name: "Medusa Store",
|
||||||
|
currencies: [
|
||||||
|
{
|
||||||
|
code: "usd",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
code: "dkk",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
default_currency_code: "dkk",
|
||||||
|
created_at: expect.any(String),
|
||||||
|
updated_at: expect.any(String),
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it("successfully updates default currency and store currencies", async () => {
|
||||||
|
const api = useApi()
|
||||||
|
|
||||||
|
const response = await api.post(
|
||||||
|
"/admin/store",
|
||||||
|
{
|
||||||
|
default_currency_code: "jpy",
|
||||||
|
currencies: ["jpy", "usd"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
headers: { Authorization: "Bearer test_token " },
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
expect(response.status).toEqual(200)
|
||||||
|
expect(response.data.store).toMatchSnapshot({
|
||||||
|
id: expect.any(String),
|
||||||
|
name: "Medusa Store",
|
||||||
|
currencies: [
|
||||||
|
{
|
||||||
|
code: "jpy",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
code: "usd",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
default_currency_code: "jpy",
|
||||||
|
created_at: expect.any(String),
|
||||||
|
updated_at: expect.any(String),
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it("successfully updates and store currencies", async () => {
|
||||||
|
const api = useApi()
|
||||||
|
|
||||||
|
const response = await api.post(
|
||||||
|
"/admin/store",
|
||||||
|
{
|
||||||
|
currencies: ["jpy", "usd"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
headers: { Authorization: "Bearer test_token " },
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
expect(response.status).toEqual(200)
|
||||||
|
expect(response.data.store).toMatchSnapshot({
|
||||||
|
id: expect.any(String),
|
||||||
|
name: "Medusa Store",
|
||||||
|
currencies: [
|
||||||
|
{
|
||||||
|
code: "jpy",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
code: "usd",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
default_currency_code: "usd",
|
||||||
|
created_at: expect.any(String),
|
||||||
|
updated_at: expect.any(String),
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -11,9 +11,9 @@ const pgGodCredentials = {
|
|||||||
password: DB_PASSWORD,
|
password: DB_PASSWORD,
|
||||||
}
|
}
|
||||||
|
|
||||||
afterAll(() => {
|
afterAll(async () => {
|
||||||
const workerId = parseInt(process.env.JEST_WORKER_ID || "1")
|
const workerId = parseInt(process.env.JEST_WORKER_ID || "1")
|
||||||
dropDatabase(
|
await dropDatabase(
|
||||||
{ databaseName: `medusa-integration-${workerId}` },
|
{ databaseName: `medusa-integration-${workerId}` },
|
||||||
pgGodCredentials
|
pgGodCredentials
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -53,7 +53,11 @@ describe("StoreService", () => {
|
|||||||
describe("update", () => {
|
describe("update", () => {
|
||||||
const storeRepository = MockRepository({
|
const storeRepository = MockRepository({
|
||||||
findOne: () =>
|
findOne: () =>
|
||||||
Promise.resolve({ id: IdMap.getId("store"), name: "Medusa" }),
|
Promise.resolve({
|
||||||
|
id: IdMap.getId("store"),
|
||||||
|
name: "Medusa",
|
||||||
|
default_currency_code: "usd",
|
||||||
|
}),
|
||||||
})
|
})
|
||||||
|
|
||||||
const currencyRepository = MockRepository({})
|
const currencyRepository = MockRepository({})
|
||||||
@@ -79,13 +83,14 @@ describe("StoreService", () => {
|
|||||||
expect(storeRepository.save).toHaveBeenCalledWith({
|
expect(storeRepository.save).toHaveBeenCalledWith({
|
||||||
id: IdMap.getId("store"),
|
id: IdMap.getId("store"),
|
||||||
name: "Medusa Commerce",
|
name: "Medusa Commerce",
|
||||||
|
default_currency_code: "usd",
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
it("fails if currency not ok", async () => {
|
it("fails if currency not ok", async () => {
|
||||||
await expect(
|
await expect(
|
||||||
storeService.update({
|
storeService.update({
|
||||||
currencies: ["1cd"],
|
currencies: ["1cd", "usd"],
|
||||||
})
|
})
|
||||||
).rejects.toThrow("Invalid currency 1cd")
|
).rejects.toThrow("Invalid currency 1cd")
|
||||||
|
|
||||||
|
|||||||
@@ -112,7 +112,7 @@ class StoreService extends BaseService {
|
|||||||
this.currencyRepository_
|
this.currencyRepository_
|
||||||
)
|
)
|
||||||
|
|
||||||
const store = await this.retrieve()
|
const store = await this.retrieve(["currencies"])
|
||||||
|
|
||||||
const {
|
const {
|
||||||
metadata,
|
metadata,
|
||||||
@@ -125,23 +125,20 @@ class StoreService extends BaseService {
|
|||||||
store.metadata = this.setMetadata_(store.id, metadata)
|
store.metadata = this.setMetadata_(store.id, metadata)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (default_currency_code) {
|
if (storeCurrencies) {
|
||||||
const curr = await currencyRepository.findOne({
|
const defaultCurr = default_currency_code ?? store.default_currency_code
|
||||||
code: default_currency_code.toLowerCase(),
|
const hasDefCurrency = storeCurrencies.find(
|
||||||
})
|
(c) => c.toLowerCase() === defaultCurr.toLowerCase()
|
||||||
|
)
|
||||||
|
|
||||||
if (!curr) {
|
// throw if we are trying to remove a currency from store currently used as default
|
||||||
|
if (!hasDefCurrency) {
|
||||||
throw new MedusaError(
|
throw new MedusaError(
|
||||||
MedusaError.Types.INVALID_DATA,
|
MedusaError.Types.INVALID_DATA,
|
||||||
`Currency ${default_currency_code} not found`
|
`You are not allowed to remove default currency from store currencies without replacing it as well`
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
store.default_currency = curr
|
|
||||||
store.default_currency_code = curr.code
|
|
||||||
}
|
|
||||||
|
|
||||||
if (storeCurrencies) {
|
|
||||||
store.currencies = await Promise.all(
|
store.currencies = await Promise.all(
|
||||||
storeCurrencies.map(async (curr) => {
|
storeCurrencies.map(async (curr) => {
|
||||||
const currency = await currencyRepository.findOne({
|
const currency = await currencyRepository.findOne({
|
||||||
@@ -160,6 +157,35 @@ class StoreService extends BaseService {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (default_currency_code) {
|
||||||
|
const storeCurrCodes = store.currencies.map((c) => c.code)
|
||||||
|
const hasDefCurrency = storeCurrCodes.find(
|
||||||
|
(c) => c === default_currency_code.toLowerCase()
|
||||||
|
)
|
||||||
|
|
||||||
|
// throw if store currencies does not have default currency
|
||||||
|
if (!hasDefCurrency) {
|
||||||
|
throw new MedusaError(
|
||||||
|
MedusaError.Types.INVALID_DATA,
|
||||||
|
`Store does not have currency: ${default_currency_code}`
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const curr = await currencyRepository.findOne({
|
||||||
|
code: default_currency_code.toLowerCase(),
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!curr) {
|
||||||
|
throw new MedusaError(
|
||||||
|
MedusaError.Types.INVALID_DATA,
|
||||||
|
`Currency ${default_currency_code} not found`
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
store.default_currency = curr
|
||||||
|
store.default_currency_code = curr.code
|
||||||
|
}
|
||||||
|
|
||||||
for (const [key, value] of Object.entries(rest)) {
|
for (const [key, value] of Object.entries(rest)) {
|
||||||
store[key] = value
|
store[key] = value
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user