mirror of
https://github.com/gristlabs/grist-core.git
synced 2024-10-27 20:44:07 +00:00
use fresher node and debian version (#1255)
This moves to node 22 and debian bookworm, since the versions we've been building and testing with are getting old. There is some old material kept around for (speaks very quietly) Python 2 (looks around hoping no-one heard) which we continue to support for some long-time users but really really should drop soon. The changes for the node upgrade were all test related. I did them in a way that shouldn't impair running on older versions of node, and did spot checks for this. This is to give some breathing room for upgrading Grist Lab's grist-saas as follow up work.
This commit is contained in:
parent
1a527d74a0
commit
aa69652a33
2
.github/workflows/docker_latest.yml
vendored
2
.github/workflows/docker_latest.yml
vendored
@ -50,7 +50,7 @@ jobs:
|
|||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
python-version: [3.11]
|
python-version: [3.11]
|
||||||
node-version: [18.x]
|
node-version: [22.x]
|
||||||
image:
|
image:
|
||||||
# We build two images, `grist-oss` and `grist`.
|
# We build two images, `grist-oss` and `grist`.
|
||||||
# See https://github.com/gristlabs/grist-core?tab=readme-ov-file#available-docker-images
|
# See https://github.com/gristlabs/grist-core?tab=readme-ov-file#available-docker-images
|
||||||
|
15
.github/workflows/main.yml
vendored
15
.github/workflows/main.yml
vendored
@ -18,19 +18,19 @@ jobs:
|
|||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
python-version: [3.11]
|
python-version: [3.11]
|
||||||
node-version: [18.x]
|
node-version: [22.x]
|
||||||
tests:
|
tests:
|
||||||
- ':lint:python:client:common:smoke:stubs:'
|
- ':lint:python:client:common:smoke:stubs:'
|
||||||
- ':server-1-of-2:'
|
- ':server-1-of-2:'
|
||||||
- ':server-2-of-2:'
|
- ':server-2-of-2:'
|
||||||
- ':nbrowser-^[A-G]:'
|
- ':nbrowser-^[A-D]:'
|
||||||
- ':nbrowser-^[H-L]:'
|
- ':nbrowser-^[E-L]:'
|
||||||
- ':nbrowser-^[M-O]:'
|
- ':nbrowser-^[M-N]:'
|
||||||
- ':nbrowser-^[P-S]:'
|
- ':nbrowser-^[O-R]:'
|
||||||
- ':nbrowser-^[^A-S]:'
|
- ':nbrowser-^[^A-R]:'
|
||||||
include:
|
include:
|
||||||
- tests: ':lint:python:client:common:smoke:'
|
- tests: ':lint:python:client:common:smoke:'
|
||||||
node-version: 18.x
|
node-version: 22.x
|
||||||
python-version: '3.10'
|
python-version: '3.10'
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v3
|
||||||
@ -125,6 +125,7 @@ jobs:
|
|||||||
ARTIFACT_NAME=logs-$(echo $TESTS | sed 's/[^-a-zA-Z0-9]/_/g')
|
ARTIFACT_NAME=logs-$(echo $TESTS | sed 's/[^-a-zA-Z0-9]/_/g')
|
||||||
echo "Artifact name is '$ARTIFACT_NAME'"
|
echo "Artifact name is '$ARTIFACT_NAME'"
|
||||||
echo "ARTIFACT_NAME=$ARTIFACT_NAME" >> $GITHUB_ENV
|
echo "ARTIFACT_NAME=$ARTIFACT_NAME" >> $GITHUB_ENV
|
||||||
|
mkdir -p $TESTDIR
|
||||||
find $TESTDIR -iname "*.socket" -exec rm {} \;
|
find $TESTDIR -iname "*.socket" -exec rm {} \;
|
||||||
env:
|
env:
|
||||||
TESTS: ${{ matrix.tests }}
|
TESTS: ${{ matrix.tests }}
|
||||||
|
53
Dockerfile
53
Dockerfile
@ -10,7 +10,7 @@ FROM scratch AS ext
|
|||||||
## Javascript build stage
|
## Javascript build stage
|
||||||
################################################################################
|
################################################################################
|
||||||
|
|
||||||
FROM node:18-buster AS builder
|
FROM node:22-bookworm AS builder
|
||||||
|
|
||||||
# Install all node dependencies.
|
# Install all node dependencies.
|
||||||
WORKDIR /grist
|
WORKDIR /grist
|
||||||
@ -45,19 +45,30 @@ RUN \
|
|||||||
## Python collection stage
|
## Python collection stage
|
||||||
################################################################################
|
################################################################################
|
||||||
|
|
||||||
# Fetch python3.11 and python2.7
|
# Fetch python3.11
|
||||||
FROM python:3.11-slim-buster AS collector
|
FROM python:3.11-slim-bookworm AS collector-py3
|
||||||
|
|
||||||
# Install all python dependencies.
|
|
||||||
ADD sandbox/requirements.txt requirements.txt
|
|
||||||
ADD sandbox/requirements3.txt requirements3.txt
|
ADD sandbox/requirements3.txt requirements3.txt
|
||||||
|
RUN \
|
||||||
|
pip3 install -r requirements3.txt
|
||||||
|
|
||||||
|
# Fetch <shame>python2.7</shame>
|
||||||
|
# This is to support users with old documents.
|
||||||
|
# If you have documents with python2.7 formulas, try switching
|
||||||
|
# to python3 in the document settings. It'll probably work fine!
|
||||||
|
# And we'll be forced to turn off python2 support eventually,
|
||||||
|
# the workarounds needed to keep it are getting silly.
|
||||||
|
# It doesn't exist in recent Debian, so we need to reach back
|
||||||
|
# to buster.
|
||||||
|
FROM python:2.7-slim-buster AS collector-py2
|
||||||
|
ADD sandbox/requirements.txt requirements.txt
|
||||||
RUN \
|
RUN \
|
||||||
apt update && \
|
apt update && \
|
||||||
apt install -y --no-install-recommends python2 python-pip python-setuptools \
|
apt install -y --no-install-recommends python2 python-pip python-setuptools \
|
||||||
build-essential libxml2-dev libxslt-dev python-dev zlib1g-dev && \
|
build-essential libxml2-dev libxslt-dev python-dev zlib1g-dev && \
|
||||||
pip2 install wheel && \
|
pip2 install wheel && \
|
||||||
pip2 install -r requirements.txt && \
|
pip2 install -r requirements.txt && \
|
||||||
pip3 install -r requirements3.txt
|
pip2 install six && \
|
||||||
|
find /usr/lib -iname "libffi.so.6*" -exec cp {} /usr/local/lib \;
|
||||||
|
|
||||||
################################################################################
|
################################################################################
|
||||||
## Sandbox collection stage
|
## Sandbox collection stage
|
||||||
@ -66,6 +77,8 @@ RUN \
|
|||||||
# Fetch gvisor-based sandbox. Note, to enable it to run within default
|
# Fetch gvisor-based sandbox. Note, to enable it to run within default
|
||||||
# unprivileged docker, layers of protection that require privilege have
|
# unprivileged docker, layers of protection that require privilege have
|
||||||
# been stripped away, see https://github.com/google/gvisor/issues/4371
|
# been stripped away, see https://github.com/google/gvisor/issues/4371
|
||||||
|
# The sandbox binary is built on buster, but remains compatible with recent
|
||||||
|
# Debian.
|
||||||
FROM docker.io/gristlabs/gvisor-unprivileged:buster AS sandbox
|
FROM docker.io/gristlabs/gvisor-unprivileged:buster AS sandbox
|
||||||
|
|
||||||
################################################################################
|
################################################################################
|
||||||
@ -73,7 +86,7 @@ FROM docker.io/gristlabs/gvisor-unprivileged:buster AS sandbox
|
|||||||
################################################################################
|
################################################################################
|
||||||
|
|
||||||
# Now, start preparing final image.
|
# Now, start preparing final image.
|
||||||
FROM node:18-buster-slim
|
FROM node:22-bookworm-slim
|
||||||
|
|
||||||
# Install libexpat1, libsqlite3-0 for python3 library binary dependencies.
|
# Install libexpat1, libsqlite3-0 for python3 library binary dependencies.
|
||||||
# Install pgrep for managing gvisor processes.
|
# Install pgrep for managing gvisor processes.
|
||||||
@ -91,13 +104,23 @@ COPY --from=builder /grist/node_modules /grist/node_modules
|
|||||||
COPY --from=builder /grist/_build /grist/_build
|
COPY --from=builder /grist/_build /grist/_build
|
||||||
COPY --from=builder /grist/static /grist/static-built
|
COPY --from=builder /grist/static /grist/static-built
|
||||||
|
|
||||||
# Copy python files.
|
# Copy python2 files.
|
||||||
COPY --from=collector /usr/bin/python2.7 /usr/bin/python2.7
|
COPY --from=collector-py2 /usr/bin/python2.7 /usr/bin/python2.7
|
||||||
COPY --from=collector /usr/lib/python2.7 /usr/lib/python2.7
|
COPY --from=collector-py2 /usr/lib/python2.7 /usr/lib/python2.7
|
||||||
COPY --from=collector /usr/local/lib/python2.7 /usr/local/lib/python2.7
|
COPY --from=collector-py2 /usr/local/lib/python2.7 /usr/local/lib/python2.7
|
||||||
COPY --from=collector /usr/local/bin/python3.11 /usr/bin/python3.11
|
# Make a small python2 tweak so that material in /usr/local/lib is found.
|
||||||
COPY --from=collector /usr/local/lib/python3.11 /usr/local/lib/python3.11
|
RUN \
|
||||||
COPY --from=collector /usr/local/lib/libpython3.11.* /usr/local/lib/
|
mkdir /etc/python2.7 && \
|
||||||
|
echo "import sys\nsys.path.append('/usr/local/lib/python2.7/site-packages')" > /etc/python2.7/sitecustomize.py
|
||||||
|
# Copy across an older libffi library binary needed by python2.
|
||||||
|
# We moved it a bit sleazily to a predictable location to avoid awkward
|
||||||
|
# architecture-dependent logic.
|
||||||
|
COPY --from=collector-py2 /usr/local/lib/libffi.so.6* /usr/local/lib
|
||||||
|
|
||||||
|
# Copy python3 files.
|
||||||
|
COPY --from=collector-py3 /usr/local/bin/python3.11 /usr/bin/python3.11
|
||||||
|
COPY --from=collector-py3 /usr/local/lib/python3.11 /usr/local/lib/python3.11
|
||||||
|
COPY --from=collector-py3 /usr/local/lib/libpython3.11.* /usr/local/lib/
|
||||||
# Set default to python3
|
# Set default to python3
|
||||||
RUN \
|
RUN \
|
||||||
ln -s /usr/bin/python3.11 /usr/bin/python && \
|
ln -s /usr/bin/python3.11 /usr/bin/python && \
|
||||||
|
@ -71,7 +71,13 @@ export class GristClientSocket {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private _createWSSocket() {
|
private _createWSSocket() {
|
||||||
if (typeof WebSocket !== 'undefined') {
|
// We used to check if WebSocket was defined here, and use it
|
||||||
|
// if so, secure in the fact that we were in the browser and
|
||||||
|
// the browser would pass along cookie information. But recent
|
||||||
|
// node defines WebSocket, so we narrow down this path to when
|
||||||
|
// a global document is defined (window doesn't work because
|
||||||
|
// some tests mock it).
|
||||||
|
if (typeof document !== 'undefined') {
|
||||||
this._wsSocket = new WebSocket(this._url);
|
this._wsSocket = new WebSocket(this._url);
|
||||||
} else {
|
} else {
|
||||||
this._wsSocket = new WS(this._url, undefined, this._options);
|
this._wsSocket = new WS(this._url, undefined, this._options);
|
||||||
|
@ -6,7 +6,17 @@ describe("NumberFormat", function() {
|
|||||||
locale: 'en-US'
|
locale: 'en-US'
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// useGrouping became more nuanced in recent node.
|
||||||
|
// Its old 'true' value may now be 'always' or 'auto'.
|
||||||
|
const useGroupingAlways = buildNumberFormat(
|
||||||
|
{numMode: 'decimal'},
|
||||||
|
defaultDocSettings
|
||||||
|
).resolvedOptions().useGrouping as boolean|string;
|
||||||
|
const useGroupingAuto = (useGroupingAlways === 'always') ? 'auto' : true;
|
||||||
|
|
||||||
it("should convert Grist options into Intr.NumberFormat", function() {
|
it("should convert Grist options into Intr.NumberFormat", function() {
|
||||||
|
assert.include([true, 'always'], String(useGroupingAlways));
|
||||||
|
|
||||||
assert.ownInclude(buildNumberFormat({}, defaultDocSettings).resolvedOptions(), {
|
assert.ownInclude(buildNumberFormat({}, defaultDocSettings).resolvedOptions(), {
|
||||||
minimumFractionDigits: 0,
|
minimumFractionDigits: 0,
|
||||||
maximumFractionDigits: 10,
|
maximumFractionDigits: 10,
|
||||||
@ -17,21 +27,21 @@ describe("NumberFormat", function() {
|
|||||||
minimumFractionDigits: 0,
|
minimumFractionDigits: 0,
|
||||||
maximumFractionDigits: 3,
|
maximumFractionDigits: 3,
|
||||||
style: 'decimal',
|
style: 'decimal',
|
||||||
useGrouping: true,
|
useGrouping: useGroupingAlways,
|
||||||
});
|
});
|
||||||
assert.ownInclude(buildNumberFormat({numMode: 'percent'}, defaultDocSettings).resolvedOptions(), {
|
assert.ownInclude(buildNumberFormat({numMode: 'percent'}, defaultDocSettings).resolvedOptions(), {
|
||||||
minimumFractionDigits: 0,
|
minimumFractionDigits: 0,
|
||||||
maximumFractionDigits: 0,
|
maximumFractionDigits: 0,
|
||||||
// style: 'percent', // In node v14.17.0 style is 'decimal' (unclear why)
|
// style: 'percent', // In node v14.17.0 style is 'decimal' (unclear why)
|
||||||
// so we check final formatting instead in this case.
|
// so we check final formatting instead in this case.
|
||||||
useGrouping: true,
|
useGrouping: useGroupingAuto,
|
||||||
});
|
});
|
||||||
assert.equal(buildNumberFormat({numMode: 'percent'}, defaultDocSettings).format(0.5), '50%');
|
assert.equal(buildNumberFormat({numMode: 'percent'}, defaultDocSettings).format(0.5), '50%');
|
||||||
assert.ownInclude(buildNumberFormat({numMode: 'currency'}, defaultDocSettings).resolvedOptions(), {
|
assert.ownInclude(buildNumberFormat({numMode: 'currency'}, defaultDocSettings).resolvedOptions(), {
|
||||||
minimumFractionDigits: 2,
|
minimumFractionDigits: 2,
|
||||||
maximumFractionDigits: 2,
|
maximumFractionDigits: 2,
|
||||||
style: 'currency',
|
style: 'currency',
|
||||||
useGrouping: true,
|
useGrouping: useGroupingAuto,
|
||||||
currency: 'USD',
|
currency: 'USD',
|
||||||
});
|
});
|
||||||
assert.ownInclude(buildNumberFormat({numMode: 'scientific'}, defaultDocSettings).resolvedOptions(), {
|
assert.ownInclude(buildNumberFormat({numMode: 'scientific'}, defaultDocSettings).resolvedOptions(), {
|
||||||
@ -73,42 +83,42 @@ describe("NumberFormat", function() {
|
|||||||
minimumFractionDigits: 2,
|
minimumFractionDigits: 2,
|
||||||
maximumFractionDigits: 2,
|
maximumFractionDigits: 2,
|
||||||
style: 'currency',
|
style: 'currency',
|
||||||
useGrouping: true,
|
useGrouping: useGroupingAuto,
|
||||||
currency: 'EUR',
|
currency: 'EUR',
|
||||||
});
|
});
|
||||||
assert.ownInclude(buildNumberFormat({numMode: 'currency'}, {locale: 'en-NZ'}).resolvedOptions(), {
|
assert.ownInclude(buildNumberFormat({numMode: 'currency'}, {locale: 'en-NZ'}).resolvedOptions(), {
|
||||||
minimumFractionDigits: 2,
|
minimumFractionDigits: 2,
|
||||||
maximumFractionDigits: 2,
|
maximumFractionDigits: 2,
|
||||||
style: 'currency',
|
style: 'currency',
|
||||||
useGrouping: true,
|
useGrouping: useGroupingAuto,
|
||||||
currency: 'NZD',
|
currency: 'NZD',
|
||||||
});
|
});
|
||||||
assert.ownInclude(buildNumberFormat({numMode: 'currency'}, {locale: 'de-CH'}).resolvedOptions(), {
|
assert.ownInclude(buildNumberFormat({numMode: 'currency'}, {locale: 'de-CH'}).resolvedOptions(), {
|
||||||
minimumFractionDigits: 2,
|
minimumFractionDigits: 2,
|
||||||
maximumFractionDigits: 2,
|
maximumFractionDigits: 2,
|
||||||
style: 'currency',
|
style: 'currency',
|
||||||
useGrouping: true,
|
useGrouping: useGroupingAuto,
|
||||||
currency: 'CHF',
|
currency: 'CHF',
|
||||||
});
|
});
|
||||||
assert.ownInclude(buildNumberFormat({numMode: 'currency'}, {locale: 'es-AR'}).resolvedOptions(), {
|
assert.ownInclude(buildNumberFormat({numMode: 'currency'}, {locale: 'es-AR'}).resolvedOptions(), {
|
||||||
minimumFractionDigits: 2,
|
minimumFractionDigits: 2,
|
||||||
maximumFractionDigits: 2,
|
maximumFractionDigits: 2,
|
||||||
style: 'currency',
|
style: 'currency',
|
||||||
useGrouping: true,
|
useGrouping: useGroupingAuto,
|
||||||
currency: 'ARS',
|
currency: 'ARS',
|
||||||
});
|
});
|
||||||
assert.ownInclude(buildNumberFormat({numMode: 'currency'}, {locale: 'zh-TW'}).resolvedOptions(), {
|
assert.ownInclude(buildNumberFormat({numMode: 'currency'}, {locale: 'zh-TW'}).resolvedOptions(), {
|
||||||
minimumFractionDigits: 2,
|
minimumFractionDigits: 2,
|
||||||
maximumFractionDigits: 2,
|
maximumFractionDigits: 2,
|
||||||
style: 'currency',
|
style: 'currency',
|
||||||
useGrouping: true,
|
useGrouping: useGroupingAuto,
|
||||||
currency: 'TWD',
|
currency: 'TWD',
|
||||||
});
|
});
|
||||||
assert.ownInclude(buildNumberFormat({numMode: 'currency'}, {locale: 'en-AU'}).resolvedOptions(), {
|
assert.ownInclude(buildNumberFormat({numMode: 'currency'}, {locale: 'en-AU'}).resolvedOptions(), {
|
||||||
minimumFractionDigits: 2,
|
minimumFractionDigits: 2,
|
||||||
maximumFractionDigits: 2,
|
maximumFractionDigits: 2,
|
||||||
style: 'currency',
|
style: 'currency',
|
||||||
useGrouping: true,
|
useGrouping: useGroupingAuto,
|
||||||
currency: 'AUD',
|
currency: 'AUD',
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -19,6 +19,10 @@ describe('Importer', function() {
|
|||||||
// have tests go faster. Each successful test case should leave the document unchanged.
|
// have tests go faster. Each successful test case should leave the document unchanged.
|
||||||
if (!docUrl || !await gu.testCurrentUrl(docUrl)) {
|
if (!docUrl || !await gu.testCurrentUrl(docUrl)) {
|
||||||
const session = await gu.session().teamSite.login();
|
const session = await gu.session().teamSite.login();
|
||||||
|
// TODO: tests check colors literally, so need to be in
|
||||||
|
// light theme - but calling gu.setGristTheme results in
|
||||||
|
// some problems so right now if you are a dev you just
|
||||||
|
// need to run these tests in light mode, sorry.
|
||||||
await session.tempDoc(cleanup, 'Hello.grist');
|
await session.tempDoc(cleanup, 'Hello.grist');
|
||||||
docUrl = await driver.getCurrentUrl();
|
docUrl = await driver.getCurrentUrl();
|
||||||
}
|
}
|
||||||
@ -450,9 +454,9 @@ describe('Importer', function() {
|
|||||||
assert.equal(await driver.findWait('.test-importer-preview', 2000).isPresent(), true);
|
assert.equal(await driver.findWait('.test-importer-preview', 2000).isPresent(), true);
|
||||||
|
|
||||||
// Check that the merge field select button has a red outline.
|
// Check that the merge field select button has a red outline.
|
||||||
assert.equal(
|
assert.match(
|
||||||
await driver.find('.test-importer-merge-fields-select').getCssValue('border'),
|
await driver.find('.test-importer-merge-fields-select').getCssValue('border'),
|
||||||
'1px solid rgb(208, 2, 27)'
|
/solid rgb\(208, 2, 27\)/
|
||||||
);
|
);
|
||||||
|
|
||||||
// Select a merge field, and check that the red outline is gone.
|
// Select a merge field, and check that the red outline is gone.
|
||||||
@ -461,9 +465,9 @@ describe('Importer', function() {
|
|||||||
'.test-multi-select-menu .test-multi-select-menu-option',
|
'.test-multi-select-menu .test-multi-select-menu-option',
|
||||||
/Name/
|
/Name/
|
||||||
).click();
|
).click();
|
||||||
assert.equal(
|
assert.match(
|
||||||
await driver.find('.test-importer-merge-fields-select').getCssValue('border'),
|
await driver.find('.test-importer-merge-fields-select').getCssValue('border'),
|
||||||
'1px solid rgb(217, 217, 217)'
|
/solid rgb\(217, 217, 217\)/
|
||||||
);
|
);
|
||||||
// Hide dropdown
|
// Hide dropdown
|
||||||
await gu.sendKeys(Key.ESCAPE);
|
await gu.sendKeys(Key.ESCAPE);
|
||||||
@ -584,9 +588,9 @@ describe('Importer', function() {
|
|||||||
await driver.findContent('.test-importer-source', /UploadedData2Extended.csv/).click();
|
await driver.findContent('.test-importer-source', /UploadedData2Extended.csv/).click();
|
||||||
|
|
||||||
// Check that it failed, and that the merge fields select button is outlined in red.
|
// Check that it failed, and that the merge fields select button is outlined in red.
|
||||||
assert.equal(
|
assert.match(
|
||||||
await driver.find('.test-importer-merge-fields-select').getCssValue('border'),
|
await driver.find('.test-importer-merge-fields-select').getCssValue('border'),
|
||||||
'1px solid rgb(208, 2, 27)'
|
/solid rgb\(208, 2, 27\)/
|
||||||
);
|
);
|
||||||
assert.equal(
|
assert.equal(
|
||||||
await driver.find('.test-importer-source-selected .test-importer-from').getText(),
|
await driver.find('.test-importer-source-selected .test-importer-from').getText(),
|
||||||
|
@ -1464,6 +1464,7 @@ export function revertChanges(test: () => Promise<void>, invariant: () => any =
|
|||||||
export async function redo(optCount: number = 1, optTimeout?: number) {
|
export async function redo(optCount: number = 1, optTimeout?: number) {
|
||||||
for (let i = 0; i < optCount; ++i) {
|
for (let i = 0; i < optCount; ++i) {
|
||||||
await driver.find('.test-redo').doClick();
|
await driver.find('.test-redo').doClick();
|
||||||
|
await waitForServer(optTimeout);
|
||||||
}
|
}
|
||||||
await waitForServer(optTimeout);
|
await waitForServer(optTimeout);
|
||||||
}
|
}
|
||||||
@ -2781,11 +2782,11 @@ export function addSamplesForSuite(includeTutorial = false) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export async function openAccountMenu() {
|
export async function openAccountMenu() {
|
||||||
await driver.findWait('.test-dm-account', 1000).click();
|
await driver.findWait('.test-dm-account', 2000).click();
|
||||||
// Since the AccountWidget loads orgs and the user data asynchronously, the menu
|
// Since the AccountWidget loads orgs and the user data asynchronously, the menu
|
||||||
// can expand itself causing the click to land on a wrong button.
|
// can expand itself causing the click to land on a wrong button.
|
||||||
await waitForServer();
|
await waitForServer();
|
||||||
await driver.findWait('.test-site-switcher-org', 1000);
|
await driver.findWait('.test-site-switcher-org', 2000);
|
||||||
await driver.sleep(250); // There's still some jitter (scroll-bar? other user accounts?)
|
await driver.sleep(250); // There's still some jitter (scroll-bar? other user accounts?)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -35,7 +35,14 @@ export const getPreviewDiffCellValues = stackWrapFunc(async (cols: number[], row
|
|||||||
|
|
||||||
// Helper that waits for the diff preview to finish loading.
|
// Helper that waits for the diff preview to finish loading.
|
||||||
export const waitForDiffPreviewToLoad = stackWrapFunc(async (): Promise<void> => {
|
export const waitForDiffPreviewToLoad = stackWrapFunc(async (): Promise<void> => {
|
||||||
|
await gu.waitForServer();
|
||||||
await driver.wait(() => driver.find('.test-importer-preview').isPresent(), 5000);
|
await driver.wait(() => driver.find('.test-importer-preview').isPresent(), 5000);
|
||||||
|
await gu.waitToPass(async () => {
|
||||||
|
const preview = (await getPreviewDiffCellValues([0], [1]))[0];
|
||||||
|
if (preview[0] === undefined && preview[1] === undefined) {
|
||||||
|
throw new Error('sometimes data is a little slow to show up?');
|
||||||
|
}
|
||||||
|
}, 2000);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Helper that gets the list of visible column matching rows to the left of the preview.
|
// Helper that gets the list of visible column matching rows to the left of the preview.
|
||||||
|
@ -76,6 +76,8 @@ function makeConfig(username: string): AxiosRequestConfig {
|
|||||||
}
|
}
|
||||||
|
|
||||||
describe('DocApi', function () {
|
describe('DocApi', function () {
|
||||||
|
const webhooksTestPort = Number(process.env.WEBHOOK_TEST_PORT || 34365);
|
||||||
|
|
||||||
this.timeout(30000);
|
this.timeout(30000);
|
||||||
testUtils.setTmpLogLevel('error');
|
testUtils.setTmpLogLevel('error');
|
||||||
let oldEnv: testUtils.EnvironmentSnapshot;
|
let oldEnv: testUtils.EnvironmentSnapshot;
|
||||||
@ -121,7 +123,7 @@ describe('DocApi', function () {
|
|||||||
homeUrl = serverUrl = home.serverUrl;
|
homeUrl = serverUrl = home.serverUrl;
|
||||||
hasHomeApi = true;
|
hasHomeApi = true;
|
||||||
});
|
});
|
||||||
testDocApi();
|
testDocApi({webhooksTestPort});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('With GRIST_ANON_PLAYGROUND disabled', async () => {
|
describe('With GRIST_ANON_PLAYGROUND disabled', async () => {
|
||||||
@ -157,7 +159,7 @@ describe('DocApi', function () {
|
|||||||
homeUrl = serverUrl = home.serverUrl;
|
homeUrl = serverUrl = home.serverUrl;
|
||||||
hasHomeApi = true;
|
hasHomeApi = true;
|
||||||
});
|
});
|
||||||
testDocApi();
|
testDocApi({webhooksTestPort});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('behind a reverse-proxy', function () {
|
describe('behind a reverse-proxy', function () {
|
||||||
@ -206,7 +208,7 @@ describe('DocApi', function () {
|
|||||||
|
|
||||||
after(() => tearDown(proxy, [home, docs]));
|
after(() => tearDown(proxy, [home, docs]));
|
||||||
|
|
||||||
testDocApi();
|
testDocApi({webhooksTestPort});
|
||||||
});
|
});
|
||||||
|
|
||||||
async function testCompareDocs(proxy: TestServerReverseProxy, home: TestServer) {
|
async function testCompareDocs(proxy: TestServerReverseProxy, home: TestServer) {
|
||||||
@ -261,7 +263,7 @@ describe('DocApi', function () {
|
|||||||
serverUrl = docs.serverUrl;
|
serverUrl = docs.serverUrl;
|
||||||
hasHomeApi = false;
|
hasHomeApi = false;
|
||||||
});
|
});
|
||||||
testDocApi();
|
testDocApi({webhooksTestPort});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -323,7 +325,10 @@ describe('DocApi', function () {
|
|||||||
});
|
});
|
||||||
|
|
||||||
// Contains the tests. This is where you want to add more test.
|
// Contains the tests. This is where you want to add more test.
|
||||||
function testDocApi() {
|
function testDocApi(settings: {
|
||||||
|
webhooksTestPort: number,
|
||||||
|
}) {
|
||||||
|
const { webhooksTestPort } = settings;
|
||||||
let chimpy: AxiosRequestConfig, kiwi: AxiosRequestConfig,
|
let chimpy: AxiosRequestConfig, kiwi: AxiosRequestConfig,
|
||||||
charon: AxiosRequestConfig, nobody: AxiosRequestConfig, support: AxiosRequestConfig;
|
charon: AxiosRequestConfig, nobody: AxiosRequestConfig, support: AxiosRequestConfig;
|
||||||
|
|
||||||
@ -3478,13 +3483,20 @@ function testDocApi() {
|
|||||||
});
|
});
|
||||||
|
|
||||||
describe('webhooks related endpoints', async function () {
|
describe('webhooks related endpoints', async function () {
|
||||||
const serving: Serving = await serveSomething(app => {
|
let serving: Serving;
|
||||||
|
before(async function () {
|
||||||
|
serving = await serveSomething(app => {
|
||||||
app.use(express.json());
|
app.use(express.json());
|
||||||
app.post('/200', ({body}, res) => {
|
app.post('/200', ({body}, res) => {
|
||||||
res.sendStatus(200);
|
res.sendStatus(200);
|
||||||
res.end();
|
res.end();
|
||||||
});
|
});
|
||||||
}, webhooksTestPort);
|
}, webhooksTestPort);
|
||||||
|
});
|
||||||
|
|
||||||
|
after(async function () {
|
||||||
|
await serving.shutdown();
|
||||||
|
});
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Regression test for old _subscribe endpoint. /docs/{did}/webhooks should be used instead to subscribe
|
Regression test for old _subscribe endpoint. /docs/{did}/webhooks should be used instead to subscribe
|
||||||
@ -3577,7 +3589,8 @@ function testDocApi() {
|
|||||||
}
|
}
|
||||||
}]
|
}]
|
||||||
},
|
},
|
||||||
403, /Column not found notExisting/);
|
// this check was previously just wrong, was the test not running somehow??
|
||||||
|
404, /Column not found "notExisting"/);
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -5385,8 +5398,6 @@ async function getWorkspaceId(api: UserAPIImpl, name: string) {
|
|||||||
return workspaces.find((w) => w.name === name)!.id;
|
return workspaces.find((w) => w.name === name)!.id;
|
||||||
}
|
}
|
||||||
|
|
||||||
const webhooksTestPort = Number(process.env.WEBHOOK_TEST_PORT || 34365);
|
|
||||||
|
|
||||||
async function setupDataDir(dir: string) {
|
async function setupDataDir(dir: string) {
|
||||||
// we'll be serving Hello.grist content for various document ids, so let's make copies of it in
|
// we'll be serving Hello.grist content for various document ids, so let's make copies of it in
|
||||||
// tmpDir
|
// tmpDir
|
||||||
|
@ -84,15 +84,17 @@ describe("ProxyAgent", function () {
|
|||||||
it("should report error when proxy fails", async function() {
|
it("should report error when proxy fails", async function() {
|
||||||
// if the proxy isn't listening, fetches produces error messages.
|
// if the proxy isn't listening, fetches produces error messages.
|
||||||
await testProxyServer.dispose();
|
await testProxyServer.dispose();
|
||||||
|
// Error message depends a little on node version.
|
||||||
const logMessages2 = await captureLog('warn', async () => {
|
const logMessages2 = await captureLog('warn', async () => {
|
||||||
await assert.isRejected(testFetch('/200'), /ECONNREFUSED/);
|
await assert.isRejected(testFetch('/200'), /(request.*failed)|(ECONNREFUSED)/);
|
||||||
await assert.isRejected(testFetch('/404'), /ECONNREFUSED/);
|
await assert.isRejected(testFetch('/404'), /(request.*failed)|(ECONNREFUSED)/);
|
||||||
});
|
});
|
||||||
|
|
||||||
// We rely on "ProxyAgent error" message to detect issues with the proxy server.
|
// We rely on "ProxyAgent error" message to detect issues with the proxy server.
|
||||||
|
// Error message depends a little on node version.
|
||||||
assertMatchArray(logMessages2, [
|
assertMatchArray(logMessages2, [
|
||||||
/warn: ProxyAgent error.*ECONNREFUSED/,
|
/warn: ProxyAgent error.*((request.*failed)|(ECONNREFUSED)|(AggregateError))/,
|
||||||
/warn: ProxyAgent error.*ECONNREFUSED/,
|
/warn: ProxyAgent error.*((request.*failed)|(ECONNREFUSED)|(AggregateError))/,
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -51,7 +51,8 @@ describe('UnhandledErrors', function() {
|
|||||||
}, 1000, 100);
|
}, 1000, 100);
|
||||||
|
|
||||||
// We expect the server to be dead now.
|
// We expect the server to be dead now.
|
||||||
await assert.isRejected(fetch(`${server.serverUrl}/status`), /failed.*ECONNREFUSED/);
|
// Error message depends a little on node version.
|
||||||
|
await assert.isRejected(fetch(`${server.serverUrl}/status`), /(request.*failed)|(ECONNREFUSED)/);
|
||||||
|
|
||||||
} finally {
|
} finally {
|
||||||
await server.stop();
|
await server.stop();
|
||||||
|
Loading…
Reference in New Issue
Block a user