mirror of
https://github.com/neondatabase/neon.git
synced 2026-06-28 09:40:36 +00:00
Compare commits
63 Commits
copy_data
...
extension_
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
a4ddad92c1 | ||
|
|
fd3dfe9d52 | ||
|
|
384e3ab1a8 | ||
|
|
4259464f72 | ||
|
|
152206211b | ||
|
|
9c35c06c58 | ||
|
|
245b4c9d72 | ||
|
|
3d0f74fc0c | ||
|
|
ce55f70cac | ||
|
|
053d592ddb | ||
|
|
b85416b58d | ||
|
|
195838436c | ||
|
|
9313045de6 | ||
|
|
44ac7a45be | ||
|
|
e35e8a7dcb | ||
|
|
a79b0d69c4 | ||
|
|
d475e901e5 | ||
|
|
bf3b83b504 | ||
|
|
94781e8710 | ||
|
|
4b83a206bf | ||
|
|
f984f9e7d3 | ||
|
|
6b42464c23 | ||
|
|
605c30e5c5 | ||
|
|
0b11d8e836 | ||
|
|
7602483af9 | ||
|
|
5e1e859ab8 | ||
|
|
85a7511700 | ||
|
|
89b8ea132e | ||
|
|
bfbae98f24 | ||
|
|
02a1d4d8c1 | ||
|
|
4a35f29301 | ||
|
|
559e318328 | ||
|
|
a4d236b02f | ||
|
|
8b9f72e117 | ||
|
|
bb414e5a0a | ||
|
|
32c03bc784 | ||
|
|
c99e203094 | ||
|
|
e7b9259675 | ||
|
|
356f7d3a7e | ||
|
|
0f6b05337e | ||
|
|
2e81d280c8 | ||
|
|
f9700c8bb9 | ||
|
|
e6137d45d2 | ||
|
|
ab1d903600 | ||
|
|
bfd670b9a7 | ||
|
|
5e96ab43ea | ||
|
|
890061d371 | ||
|
|
6b74d1a76a | ||
|
|
a936b8a92b | ||
|
|
c7bea52849 | ||
|
|
1b7ab6d468 | ||
|
|
e07d5d00e9 | ||
|
|
15d3d007eb | ||
|
|
77157c7741 | ||
|
|
b9b1b3596c | ||
|
|
91a809332f | ||
|
|
214ecacfc4 | ||
|
|
7465c644b9 | ||
|
|
bb931f2ce0 | ||
|
|
8013f9630d | ||
|
|
34f22e9b12 | ||
|
|
4f3f817384 | ||
|
|
b50475b567 |
9
.github/workflows/benchmarking.yml
vendored
9
.github/workflows/benchmarking.yml
vendored
@@ -180,8 +180,7 @@ jobs:
|
|||||||
image: 369495373322.dkr.ecr.eu-central-1.amazonaws.com/rust:pinned
|
image: 369495373322.dkr.ecr.eu-central-1.amazonaws.com/rust:pinned
|
||||||
options: --init
|
options: --init
|
||||||
|
|
||||||
# Increase timeout to 8h, default timeout is 6h
|
timeout-minutes: 360 # 6h
|
||||||
timeout-minutes: 480
|
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v3
|
||||||
@@ -322,6 +321,8 @@ jobs:
|
|||||||
image: 369495373322.dkr.ecr.eu-central-1.amazonaws.com/rust:pinned
|
image: 369495373322.dkr.ecr.eu-central-1.amazonaws.com/rust:pinned
|
||||||
options: --init
|
options: --init
|
||||||
|
|
||||||
|
timeout-minutes: 360 # 6h
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v3
|
||||||
|
|
||||||
@@ -413,6 +414,8 @@ jobs:
|
|||||||
image: 369495373322.dkr.ecr.eu-central-1.amazonaws.com/rust:pinned
|
image: 369495373322.dkr.ecr.eu-central-1.amazonaws.com/rust:pinned
|
||||||
options: --init
|
options: --init
|
||||||
|
|
||||||
|
timeout-minutes: 360 # 6h
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v3
|
||||||
|
|
||||||
@@ -498,6 +501,8 @@ jobs:
|
|||||||
image: 369495373322.dkr.ecr.eu-central-1.amazonaws.com/rust:pinned
|
image: 369495373322.dkr.ecr.eu-central-1.amazonaws.com/rust:pinned
|
||||||
options: --init
|
options: --init
|
||||||
|
|
||||||
|
timeout-minutes: 360 # 6h
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v3
|
||||||
|
|
||||||
|
|||||||
118
.github/workflows/build_and_test.yml
vendored
118
.github/workflows/build_and_test.yml
vendored
@@ -722,35 +722,6 @@ jobs:
|
|||||||
--dockerfile Dockerfile.compute-node
|
--dockerfile Dockerfile.compute-node
|
||||||
--destination 369495373322.dkr.ecr.eu-central-1.amazonaws.com/compute-node-${{ matrix.version }}:${{needs.tag.outputs.build-tag}}
|
--destination 369495373322.dkr.ecr.eu-central-1.amazonaws.com/compute-node-${{ matrix.version }}:${{needs.tag.outputs.build-tag}}
|
||||||
--destination neondatabase/compute-node-${{ matrix.version }}:${{needs.tag.outputs.build-tag}}
|
--destination neondatabase/compute-node-${{ matrix.version }}:${{needs.tag.outputs.build-tag}}
|
||||||
--cleanup
|
|
||||||
|
|
||||||
# Due to a kaniko bug, we can't use cache for extensions image, thus it takes about the same amount of time as compute-node image to build (~10 min)
|
|
||||||
# During the transition period we need to have extensions in both places (in S3 and in compute-node image),
|
|
||||||
# so we won't build extension twice, but extract them from compute-node.
|
|
||||||
#
|
|
||||||
# For now we use extensions image only for new custom extensitons
|
|
||||||
- name: Kaniko build extensions only
|
|
||||||
run: |
|
|
||||||
# Kaniko is suposed to clean up after itself if --cleanup flag is set, but it doesn't.
|
|
||||||
# Despite some fixes were made in https://github.com/GoogleContainerTools/kaniko/pull/2504 (in kaniko v1.11.0),
|
|
||||||
# it still fails with error:
|
|
||||||
# error building image: could not save file: copying file: symlink postgres /kaniko/1/usr/local/pgsql/bin/postmaster: file exists
|
|
||||||
#
|
|
||||||
# Ref https://github.com/GoogleContainerTools/kaniko/issues/1406
|
|
||||||
find /kaniko -maxdepth 1 -mindepth 1 -type d -regex "/kaniko/[0-9]*" -exec rm -rv {} \;
|
|
||||||
|
|
||||||
/kaniko/executor --reproducible --snapshot-mode=redo --skip-unused-stages --cache=true \
|
|
||||||
--cache-repo 369495373322.dkr.ecr.eu-central-1.amazonaws.com/cache \
|
|
||||||
--context . \
|
|
||||||
--build-arg GIT_VERSION=${{ github.sha }} \
|
|
||||||
--build-arg PG_VERSION=${{ matrix.version }} \
|
|
||||||
--build-arg BUILD_TAG=${{needs.tag.outputs.build-tag}} \
|
|
||||||
--build-arg REPOSITORY=369495373322.dkr.ecr.eu-central-1.amazonaws.com \
|
|
||||||
--dockerfile Dockerfile.compute-node \
|
|
||||||
--destination 369495373322.dkr.ecr.eu-central-1.amazonaws.com/extensions-${{ matrix.version }}:${{needs.tag.outputs.build-tag}} \
|
|
||||||
--destination neondatabase/extensions-${{ matrix.version }}:${{needs.tag.outputs.build-tag}} \
|
|
||||||
--cleanup \
|
|
||||||
--target postgres-extensions
|
|
||||||
|
|
||||||
# Cleanup script fails otherwise - rm: cannot remove '/nvme/actions-runner/_work/_temp/_github_home/.ecr': Permission denied
|
# Cleanup script fails otherwise - rm: cannot remove '/nvme/actions-runner/_work/_temp/_github_home/.ecr': Permission denied
|
||||||
- name: Cleanup ECR folder
|
- name: Cleanup ECR folder
|
||||||
@@ -767,7 +738,7 @@ jobs:
|
|||||||
run:
|
run:
|
||||||
shell: sh -eu {0}
|
shell: sh -eu {0}
|
||||||
env:
|
env:
|
||||||
VM_BUILDER_VERSION: v0.11.1
|
VM_BUILDER_VERSION: v0.8.0
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
@@ -869,10 +840,8 @@ jobs:
|
|||||||
crane tag 369495373322.dkr.ecr.eu-central-1.amazonaws.com/compute-tools:${{needs.tag.outputs.build-tag}} latest
|
crane tag 369495373322.dkr.ecr.eu-central-1.amazonaws.com/compute-tools:${{needs.tag.outputs.build-tag}} latest
|
||||||
crane tag 369495373322.dkr.ecr.eu-central-1.amazonaws.com/compute-node-v14:${{needs.tag.outputs.build-tag}} latest
|
crane tag 369495373322.dkr.ecr.eu-central-1.amazonaws.com/compute-node-v14:${{needs.tag.outputs.build-tag}} latest
|
||||||
crane tag 369495373322.dkr.ecr.eu-central-1.amazonaws.com/vm-compute-node-v14:${{needs.tag.outputs.build-tag}} latest
|
crane tag 369495373322.dkr.ecr.eu-central-1.amazonaws.com/vm-compute-node-v14:${{needs.tag.outputs.build-tag}} latest
|
||||||
crane tag 369495373322.dkr.ecr.eu-central-1.amazonaws.com/extensions-v14:${{needs.tag.outputs.build-tag}} latest
|
|
||||||
crane tag 369495373322.dkr.ecr.eu-central-1.amazonaws.com/compute-node-v15:${{needs.tag.outputs.build-tag}} latest
|
crane tag 369495373322.dkr.ecr.eu-central-1.amazonaws.com/compute-node-v15:${{needs.tag.outputs.build-tag}} latest
|
||||||
crane tag 369495373322.dkr.ecr.eu-central-1.amazonaws.com/vm-compute-node-v15:${{needs.tag.outputs.build-tag}} latest
|
crane tag 369495373322.dkr.ecr.eu-central-1.amazonaws.com/vm-compute-node-v15:${{needs.tag.outputs.build-tag}} latest
|
||||||
crane tag 369495373322.dkr.ecr.eu-central-1.amazonaws.com/extensions-v15:${{needs.tag.outputs.build-tag}} latest
|
|
||||||
|
|
||||||
- name: Push images to production ECR
|
- name: Push images to production ECR
|
||||||
if: |
|
if: |
|
||||||
@@ -883,10 +852,8 @@ jobs:
|
|||||||
crane copy 369495373322.dkr.ecr.eu-central-1.amazonaws.com/compute-tools:${{needs.tag.outputs.build-tag}} 093970136003.dkr.ecr.eu-central-1.amazonaws.com/compute-tools:latest
|
crane copy 369495373322.dkr.ecr.eu-central-1.amazonaws.com/compute-tools:${{needs.tag.outputs.build-tag}} 093970136003.dkr.ecr.eu-central-1.amazonaws.com/compute-tools:latest
|
||||||
crane copy 369495373322.dkr.ecr.eu-central-1.amazonaws.com/compute-node-v14:${{needs.tag.outputs.build-tag}} 093970136003.dkr.ecr.eu-central-1.amazonaws.com/compute-node-v14:latest
|
crane copy 369495373322.dkr.ecr.eu-central-1.amazonaws.com/compute-node-v14:${{needs.tag.outputs.build-tag}} 093970136003.dkr.ecr.eu-central-1.amazonaws.com/compute-node-v14:latest
|
||||||
crane copy 369495373322.dkr.ecr.eu-central-1.amazonaws.com/vm-compute-node-v14:${{needs.tag.outputs.build-tag}} 093970136003.dkr.ecr.eu-central-1.amazonaws.com/vm-compute-node-v14:latest
|
crane copy 369495373322.dkr.ecr.eu-central-1.amazonaws.com/vm-compute-node-v14:${{needs.tag.outputs.build-tag}} 093970136003.dkr.ecr.eu-central-1.amazonaws.com/vm-compute-node-v14:latest
|
||||||
crane copy 369495373322.dkr.ecr.eu-central-1.amazonaws.com/extensions-v14:${{needs.tag.outputs.build-tag}} 093970136003.dkr.ecr.eu-central-1.amazonaws.com/extensions-v14:latest
|
|
||||||
crane copy 369495373322.dkr.ecr.eu-central-1.amazonaws.com/compute-node-v15:${{needs.tag.outputs.build-tag}} 093970136003.dkr.ecr.eu-central-1.amazonaws.com/compute-node-v15:latest
|
crane copy 369495373322.dkr.ecr.eu-central-1.amazonaws.com/compute-node-v15:${{needs.tag.outputs.build-tag}} 093970136003.dkr.ecr.eu-central-1.amazonaws.com/compute-node-v15:latest
|
||||||
crane copy 369495373322.dkr.ecr.eu-central-1.amazonaws.com/vm-compute-node-v15:${{needs.tag.outputs.build-tag}} 093970136003.dkr.ecr.eu-central-1.amazonaws.com/vm-compute-node-v15:latest
|
crane copy 369495373322.dkr.ecr.eu-central-1.amazonaws.com/vm-compute-node-v15:${{needs.tag.outputs.build-tag}} 093970136003.dkr.ecr.eu-central-1.amazonaws.com/vm-compute-node-v15:latest
|
||||||
crane copy 369495373322.dkr.ecr.eu-central-1.amazonaws.com/extensions-v15:${{needs.tag.outputs.build-tag}} 093970136003.dkr.ecr.eu-central-1.amazonaws.com/extensions-v15:latest
|
|
||||||
|
|
||||||
- name: Configure Docker Hub login
|
- name: Configure Docker Hub login
|
||||||
run: |
|
run: |
|
||||||
@@ -908,93 +875,16 @@ jobs:
|
|||||||
crane tag neondatabase/compute-tools:${{needs.tag.outputs.build-tag}} latest
|
crane tag neondatabase/compute-tools:${{needs.tag.outputs.build-tag}} latest
|
||||||
crane tag neondatabase/compute-node-v14:${{needs.tag.outputs.build-tag}} latest
|
crane tag neondatabase/compute-node-v14:${{needs.tag.outputs.build-tag}} latest
|
||||||
crane tag neondatabase/vm-compute-node-v14:${{needs.tag.outputs.build-tag}} latest
|
crane tag neondatabase/vm-compute-node-v14:${{needs.tag.outputs.build-tag}} latest
|
||||||
crane tag neondatabase/extensions-v14:${{needs.tag.outputs.build-tag}} latest
|
|
||||||
crane tag neondatabase/compute-node-v15:${{needs.tag.outputs.build-tag}} latest
|
crane tag neondatabase/compute-node-v15:${{needs.tag.outputs.build-tag}} latest
|
||||||
crane tag neondatabase/vm-compute-node-v15:${{needs.tag.outputs.build-tag}} latest
|
crane tag neondatabase/vm-compute-node-v15:${{needs.tag.outputs.build-tag}} latest
|
||||||
crane tag neondatabase/extensions-v15:${{needs.tag.outputs.build-tag}} latest
|
|
||||||
|
|
||||||
- name: Cleanup ECR folder
|
- name: Cleanup ECR folder
|
||||||
run: rm -rf ~/.ecr
|
run: rm -rf ~/.ecr
|
||||||
|
|
||||||
upload-postgres-extensions-to-s3:
|
|
||||||
if: |
|
|
||||||
(github.ref_name == 'main' || github.ref_name == 'release') &&
|
|
||||||
github.event_name != 'workflow_dispatch'
|
|
||||||
runs-on: ${{ github.ref_name == 'release' && fromJSON('["self-hosted", "prod", "x64"]') || fromJSON('["self-hosted", "gen3", "small"]') }}
|
|
||||||
needs: [ tag, promote-images ]
|
|
||||||
strategy:
|
|
||||||
fail-fast: false
|
|
||||||
matrix:
|
|
||||||
version: [ v14, v15 ]
|
|
||||||
|
|
||||||
env:
|
|
||||||
# While on transition period we extract public extensions from compute-node image and custom extensions from extensions image.
|
|
||||||
# Later all the extensions will be moved to extensions image.
|
|
||||||
EXTENSIONS_IMAGE: ${{ github.ref_name == 'release' && '093970136003' || '369495373322'}}.dkr.ecr.eu-central-1.amazonaws.com/extensions-${{ matrix.version }}:latest
|
|
||||||
COMPUTE_NODE_IMAGE: ${{ github.ref_name == 'release' && '093970136003' || '369495373322'}}.dkr.ecr.eu-central-1.amazonaws.com/compute-node-${{ matrix.version }}:latest
|
|
||||||
AWS_ACCESS_KEY_ID: ${{ github.ref_name == 'release' && secrets.AWS_ACCESS_KEY_PROD || secrets.AWS_ACCESS_KEY_DEV }}
|
|
||||||
AWS_SECRET_ACCESS_KEY: ${{ github.ref_name == 'release' && secrets.AWS_SECRET_KEY_PROD || secrets.AWS_SECRET_KEY_DEV }}
|
|
||||||
S3_BUCKETS: |
|
|
||||||
${{ github.ref_name == 'release' &&
|
|
||||||
'neon-prod-extensions-ap-southeast-1 neon-prod-extensions-eu-central-1 neon-prod-extensions-us-east-1 neon-prod-extensions-us-east-2 neon-prod-extensions-us-west-2' ||
|
|
||||||
'neon-dev-extensions-eu-central-1 neon-dev-extensions-eu-west-1 neon-dev-extensions-us-east-2' }}
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Pull postgres-extensions image
|
|
||||||
run: |
|
|
||||||
docker pull ${EXTENSIONS_IMAGE}
|
|
||||||
docker pull ${COMPUTE_NODE_IMAGE}
|
|
||||||
|
|
||||||
- name: Create postgres-extensions container
|
|
||||||
id: create-container
|
|
||||||
run: |
|
|
||||||
EID=$(docker create ${EXTENSIONS_IMAGE} true)
|
|
||||||
echo "EID=${EID}" >> $GITHUB_OUTPUT
|
|
||||||
|
|
||||||
CID=$(docker create ${COMPUTE_NODE_IMAGE} true)
|
|
||||||
echo "CID=${CID}" >> $GITHUB_OUTPUT
|
|
||||||
|
|
||||||
- name: Extract postgres-extensions from container
|
|
||||||
run: |
|
|
||||||
rm -rf ./extensions-to-upload ./custom-extensions # Just in case
|
|
||||||
|
|
||||||
# In compute image we have a bit different directory layout
|
|
||||||
mkdir -p extensions-to-upload/share
|
|
||||||
docker cp ${{ steps.create-container.outputs.CID }}:/usr/local/share/extension ./extensions-to-upload/share/extension
|
|
||||||
docker cp ${{ steps.create-container.outputs.CID }}:/usr/local/lib ./extensions-to-upload/lib
|
|
||||||
|
|
||||||
# Delete Neon extensitons (they always present on compute-node image)
|
|
||||||
rm -rf ./extensions-to-upload/share/extension/neon*
|
|
||||||
rm -rf ./extensions-to-upload/lib/neon*
|
|
||||||
|
|
||||||
# Delete leftovers from the extension build step
|
|
||||||
rm -rf ./extensions-to-upload/lib/pgxs
|
|
||||||
rm -rf ./extensions-to-upload/lib/pkgconfig
|
|
||||||
|
|
||||||
docker cp ${{ steps.create-container.outputs.EID }}:/extensions ./custom-extensions
|
|
||||||
for EXT_NAME in $(ls ./custom-extensions); do
|
|
||||||
mkdir -p ./extensions-to-upload/${EXT_NAME}/share
|
|
||||||
|
|
||||||
mv ./custom-extensions/${EXT_NAME}/share/extension ./extensions-to-upload/${EXT_NAME}/share/extension
|
|
||||||
mv ./custom-extensions/${EXT_NAME}/lib ./extensions-to-upload/${EXT_NAME}/lib
|
|
||||||
done
|
|
||||||
|
|
||||||
- name: Upload postgres-extensions to S3
|
|
||||||
run: |
|
|
||||||
for BUCKET in $(echo ${S3_BUCKETS}); do
|
|
||||||
aws s3 cp --recursive --only-show-errors ./extensions-to-upload s3://${BUCKET}/${{ needs.tag.outputs.build-tag }}/${{ matrix.version }}
|
|
||||||
done
|
|
||||||
|
|
||||||
- name: Cleanup
|
|
||||||
if: ${{ always() && (steps.create-container.outputs.CID || steps.create-container.outputs.EID) }}
|
|
||||||
run: |
|
|
||||||
docker rm ${{ steps.create-container.outputs.CID }} || true
|
|
||||||
docker rm ${{ steps.create-container.outputs.EID }} || true
|
|
||||||
|
|
||||||
deploy:
|
deploy:
|
||||||
runs-on: [ self-hosted, gen3, small ]
|
runs-on: [ self-hosted, gen3, small ]
|
||||||
container: 369495373322.dkr.ecr.eu-central-1.amazonaws.com/ansible:latest
|
container: 369495373322.dkr.ecr.eu-central-1.amazonaws.com/ansible:latest
|
||||||
needs: [ upload-postgres-extensions-to-s3, promote-images, tag, regress-tests ]
|
needs: [ promote-images, tag, regress-tests ]
|
||||||
if: ( github.ref_name == 'main' || github.ref_name == 'release' ) && github.event_name != 'workflow_dispatch'
|
if: ( github.ref_name == 'main' || github.ref_name == 'release' ) && github.event_name != 'workflow_dispatch'
|
||||||
steps:
|
steps:
|
||||||
- name: Fix git ownership
|
- name: Fix git ownership
|
||||||
@@ -1026,7 +916,7 @@ jobs:
|
|||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
- name: Create git tag
|
- name: Create tag "release-${{ needs.tag.outputs.build-tag }}"
|
||||||
if: github.ref_name == 'release'
|
if: github.ref_name == 'release'
|
||||||
uses: actions/github-script@v6
|
uses: actions/github-script@v6
|
||||||
with:
|
with:
|
||||||
@@ -1036,7 +926,7 @@ jobs:
|
|||||||
github.rest.git.createRef({
|
github.rest.git.createRef({
|
||||||
owner: context.repo.owner,
|
owner: context.repo.owner,
|
||||||
repo: context.repo.repo,
|
repo: context.repo.repo,
|
||||||
ref: "refs/tags/${{ needs.tag.outputs.build-tag }}",
|
ref: "refs/tags/release-${{ needs.tag.outputs.build-tag }}",
|
||||||
sha: context.sha,
|
sha: context.sha,
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
212
Cargo.lock
generated
212
Cargo.lock
generated
@@ -200,6 +200,17 @@ dependencies = [
|
|||||||
"critical-section",
|
"critical-section",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "atty"
|
||||||
|
version = "0.2.14"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "d9b39be18770d11421cdb1b9947a45dd3f37e93092cbf377614828a319d5fee8"
|
||||||
|
dependencies = [
|
||||||
|
"hermit-abi 0.1.19",
|
||||||
|
"libc",
|
||||||
|
"winapi",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "autocfg"
|
name = "autocfg"
|
||||||
version = "1.1.0"
|
version = "1.1.0"
|
||||||
@@ -794,6 +805,18 @@ dependencies = [
|
|||||||
"libloading",
|
"libloading",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "clap"
|
||||||
|
version = "3.2.25"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "4ea181bf566f71cb9a5d17a59e1871af638180a18fb0035c92ae62b705207123"
|
||||||
|
dependencies = [
|
||||||
|
"bitflags",
|
||||||
|
"clap_lex 0.2.4",
|
||||||
|
"indexmap",
|
||||||
|
"textwrap",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "clap"
|
name = "clap"
|
||||||
version = "4.3.0"
|
version = "4.3.0"
|
||||||
@@ -814,7 +837,7 @@ dependencies = [
|
|||||||
"anstream",
|
"anstream",
|
||||||
"anstyle",
|
"anstyle",
|
||||||
"bitflags",
|
"bitflags",
|
||||||
"clap_lex",
|
"clap_lex 0.5.0",
|
||||||
"strsim",
|
"strsim",
|
||||||
]
|
]
|
||||||
|
|
||||||
@@ -830,6 +853,15 @@ dependencies = [
|
|||||||
"syn 2.0.16",
|
"syn 2.0.16",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "clap_lex"
|
||||||
|
version = "0.2.4"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "2850f2f5a82cbf437dd5af4d49848fbdfc27c157c3d010345776f952765261c5"
|
||||||
|
dependencies = [
|
||||||
|
"os_str_bytes",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "clap_lex"
|
name = "clap_lex"
|
||||||
version = "0.5.0"
|
version = "0.5.0"
|
||||||
@@ -883,7 +915,7 @@ version = "0.1.0"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"chrono",
|
"chrono",
|
||||||
"clap",
|
"clap 4.3.0",
|
||||||
"compute_api",
|
"compute_api",
|
||||||
"futures",
|
"futures",
|
||||||
"hyper",
|
"hyper",
|
||||||
@@ -892,12 +924,14 @@ dependencies = [
|
|||||||
"opentelemetry",
|
"opentelemetry",
|
||||||
"postgres",
|
"postgres",
|
||||||
"regex",
|
"regex",
|
||||||
|
"remote_storage",
|
||||||
"reqwest",
|
"reqwest",
|
||||||
"serde",
|
"serde",
|
||||||
"serde_json",
|
"serde_json",
|
||||||
"tar",
|
"tar",
|
||||||
"tokio",
|
"tokio",
|
||||||
"tokio-postgres",
|
"tokio-postgres",
|
||||||
|
"toml_edit",
|
||||||
"tracing",
|
"tracing",
|
||||||
"tracing-opentelemetry",
|
"tracing-opentelemetry",
|
||||||
"tracing-subscriber",
|
"tracing-subscriber",
|
||||||
@@ -945,7 +979,7 @@ name = "control_plane"
|
|||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"clap",
|
"clap 4.3.0",
|
||||||
"comfy-table",
|
"comfy-table",
|
||||||
"compute_api",
|
"compute_api",
|
||||||
"git-version",
|
"git-version",
|
||||||
@@ -965,6 +999,7 @@ dependencies = [
|
|||||||
"tar",
|
"tar",
|
||||||
"thiserror",
|
"thiserror",
|
||||||
"toml",
|
"toml",
|
||||||
|
"tracing",
|
||||||
"url",
|
"url",
|
||||||
"utils",
|
"utils",
|
||||||
"workspace_hack",
|
"workspace_hack",
|
||||||
@@ -1015,19 +1050,19 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "criterion"
|
name = "criterion"
|
||||||
version = "0.5.1"
|
version = "0.4.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "f2b12d017a929603d80db1831cd3a24082f8137ce19c69e6447f54f5fc8d692f"
|
checksum = "e7c76e09c1aae2bc52b3d2f29e13c6572553b30c4aa1b8a49fd70de6412654cb"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anes",
|
"anes",
|
||||||
|
"atty",
|
||||||
"cast",
|
"cast",
|
||||||
"ciborium",
|
"ciborium",
|
||||||
"clap",
|
"clap 3.2.25",
|
||||||
"criterion-plot",
|
"criterion-plot",
|
||||||
"is-terminal",
|
|
||||||
"itertools",
|
"itertools",
|
||||||
|
"lazy_static",
|
||||||
"num-traits",
|
"num-traits",
|
||||||
"once_cell",
|
|
||||||
"oorandom",
|
"oorandom",
|
||||||
"plotters",
|
"plotters",
|
||||||
"rayon",
|
"rayon",
|
||||||
@@ -1108,7 +1143,7 @@ dependencies = [
|
|||||||
"crossterm_winapi",
|
"crossterm_winapi",
|
||||||
"libc",
|
"libc",
|
||||||
"mio",
|
"mio",
|
||||||
"parking_lot 0.12.1",
|
"parking_lot",
|
||||||
"signal-hook",
|
"signal-hook",
|
||||||
"signal-hook-mio",
|
"signal-hook-mio",
|
||||||
"winapi",
|
"winapi",
|
||||||
@@ -1178,7 +1213,7 @@ dependencies = [
|
|||||||
"hashbrown 0.12.3",
|
"hashbrown 0.12.3",
|
||||||
"lock_api",
|
"lock_api",
|
||||||
"once_cell",
|
"once_cell",
|
||||||
"parking_lot_core 0.9.7",
|
"parking_lot_core",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -1644,6 +1679,15 @@ version = "0.4.1"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "95505c38b4572b2d910cecb0281560f54b440a19336cbbcb27bf6ce6adc6f5a8"
|
checksum = "95505c38b4572b2d910cecb0281560f54b440a19336cbbcb27bf6ce6adc6f5a8"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "hermit-abi"
|
||||||
|
version = "0.1.19"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "62b467343b94ba476dcb2500d242dadbb39557df889310ac77c5d99100aaac33"
|
||||||
|
dependencies = [
|
||||||
|
"libc",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "hermit-abi"
|
name = "hermit-abi"
|
||||||
version = "0.2.6"
|
version = "0.2.6"
|
||||||
@@ -1898,9 +1942,6 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
|||||||
checksum = "7a5bbe824c507c5da5956355e86a746d82e0e1464f65d862cc5e71da70e94b2c"
|
checksum = "7a5bbe824c507c5da5956355e86a746d82e0e1464f65d862cc5e71da70e94b2c"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"cfg-if",
|
"cfg-if",
|
||||||
"js-sys",
|
|
||||||
"wasm-bindgen",
|
|
||||||
"web-sys",
|
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -2229,6 +2270,16 @@ dependencies = [
|
|||||||
"windows-sys 0.45.0",
|
"windows-sys 0.45.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "nu-ansi-term"
|
||||||
|
version = "0.46.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "77a8165726e8236064dbb45459242600304b42a5ea24ee2948e18e023bf7ba84"
|
||||||
|
dependencies = [
|
||||||
|
"overload",
|
||||||
|
"winapi",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "num-bigint"
|
name = "num-bigint"
|
||||||
version = "0.4.3"
|
version = "0.4.3"
|
||||||
@@ -2456,19 +2507,31 @@ dependencies = [
|
|||||||
"winapi",
|
"winapi",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "os_str_bytes"
|
||||||
|
version = "6.5.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "ceedf44fb00f2d1984b0bc98102627ce622e083e49a5bacdb3e514fa4238e267"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "outref"
|
name = "outref"
|
||||||
version = "0.5.1"
|
version = "0.5.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "4030760ffd992bef45b0ae3f10ce1aba99e33464c90d14dd7c039884963ddc7a"
|
checksum = "4030760ffd992bef45b0ae3f10ce1aba99e33464c90d14dd7c039884963ddc7a"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "overload"
|
||||||
|
version = "0.1.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "b15813163c1d831bf4a13c3610c05c0d03b39feb07f7e09fa234dac9b15aaf39"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "pagectl"
|
name = "pagectl"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"bytes",
|
"bytes",
|
||||||
"clap",
|
"clap 4.3.0",
|
||||||
"git-version",
|
"git-version",
|
||||||
"pageserver",
|
"pageserver",
|
||||||
"postgres_ffi",
|
"postgres_ffi",
|
||||||
@@ -2487,7 +2550,7 @@ dependencies = [
|
|||||||
"byteorder",
|
"byteorder",
|
||||||
"bytes",
|
"bytes",
|
||||||
"chrono",
|
"chrono",
|
||||||
"clap",
|
"clap 4.3.0",
|
||||||
"close_fds",
|
"close_fds",
|
||||||
"const_format",
|
"const_format",
|
||||||
"consumption_metrics",
|
"consumption_metrics",
|
||||||
@@ -2569,17 +2632,6 @@ dependencies = [
|
|||||||
"workspace_hack",
|
"workspace_hack",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "parking_lot"
|
|
||||||
version = "0.11.2"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "7d17b78036a60663b797adeaee46f5c9dfebb86948d1255007a1d6be0271ff99"
|
|
||||||
dependencies = [
|
|
||||||
"instant",
|
|
||||||
"lock_api",
|
|
||||||
"parking_lot_core 0.8.6",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "parking_lot"
|
name = "parking_lot"
|
||||||
version = "0.12.1"
|
version = "0.12.1"
|
||||||
@@ -2587,21 +2639,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
|||||||
checksum = "3742b2c103b9f06bc9fff0a37ff4912935851bee6d36f3c02bcc755bcfec228f"
|
checksum = "3742b2c103b9f06bc9fff0a37ff4912935851bee6d36f3c02bcc755bcfec228f"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"lock_api",
|
"lock_api",
|
||||||
"parking_lot_core 0.9.7",
|
"parking_lot_core",
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "parking_lot_core"
|
|
||||||
version = "0.8.6"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "60a2cfe6f0ad2bfc16aefa463b497d5c7a5ecd44a23efa72aa342d90177356dc"
|
|
||||||
dependencies = [
|
|
||||||
"cfg-if",
|
|
||||||
"instant",
|
|
||||||
"libc",
|
|
||||||
"redox_syscall 0.2.16",
|
|
||||||
"smallvec",
|
|
||||||
"winapi",
|
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -2617,16 +2655,6 @@ dependencies = [
|
|||||||
"windows-sys 0.45.0",
|
"windows-sys 0.45.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "pbkdf2"
|
|
||||||
version = "0.12.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "f0ca0b5a68607598bf3bad68f32227a8164f6254833f84eafaac409cd6746c31"
|
|
||||||
dependencies = [
|
|
||||||
"digest",
|
|
||||||
"hmac",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "peeking_take_while"
|
name = "peeking_take_while"
|
||||||
version = "0.1.2"
|
version = "0.1.2"
|
||||||
@@ -2932,7 +2960,7 @@ dependencies = [
|
|||||||
"lazy_static",
|
"lazy_static",
|
||||||
"libc",
|
"libc",
|
||||||
"memchr",
|
"memchr",
|
||||||
"parking_lot 0.12.1",
|
"parking_lot",
|
||||||
"procfs",
|
"procfs",
|
||||||
"thiserror",
|
"thiserror",
|
||||||
]
|
]
|
||||||
@@ -2997,11 +3025,12 @@ version = "0.1.0"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"async-trait",
|
"async-trait",
|
||||||
|
"atty",
|
||||||
"base64 0.13.1",
|
"base64 0.13.1",
|
||||||
"bstr",
|
"bstr",
|
||||||
"bytes",
|
"bytes",
|
||||||
"chrono",
|
"chrono",
|
||||||
"clap",
|
"clap 4.3.0",
|
||||||
"consumption_metrics",
|
"consumption_metrics",
|
||||||
"futures",
|
"futures",
|
||||||
"git-version",
|
"git-version",
|
||||||
@@ -3019,8 +3048,7 @@ dependencies = [
|
|||||||
"native-tls",
|
"native-tls",
|
||||||
"once_cell",
|
"once_cell",
|
||||||
"opentelemetry",
|
"opentelemetry",
|
||||||
"parking_lot 0.12.1",
|
"parking_lot",
|
||||||
"pbkdf2",
|
|
||||||
"pin-project-lite",
|
"pin-project-lite",
|
||||||
"postgres-native-tls",
|
"postgres-native-tls",
|
||||||
"postgres_backend",
|
"postgres_backend",
|
||||||
@@ -3031,7 +3059,6 @@ dependencies = [
|
|||||||
"regex",
|
"regex",
|
||||||
"reqwest",
|
"reqwest",
|
||||||
"reqwest-middleware",
|
"reqwest-middleware",
|
||||||
"reqwest-retry",
|
|
||||||
"reqwest-tracing",
|
"reqwest-tracing",
|
||||||
"routerify",
|
"routerify",
|
||||||
"rstest",
|
"rstest",
|
||||||
@@ -3267,29 +3294,6 @@ dependencies = [
|
|||||||
"thiserror",
|
"thiserror",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "reqwest-retry"
|
|
||||||
version = "0.2.2"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "48d0fd6ef4c6d23790399fe15efc8d12cd9f3d4133958f9bd7801ee5cbaec6c4"
|
|
||||||
dependencies = [
|
|
||||||
"anyhow",
|
|
||||||
"async-trait",
|
|
||||||
"chrono",
|
|
||||||
"futures",
|
|
||||||
"getrandom",
|
|
||||||
"http",
|
|
||||||
"hyper",
|
|
||||||
"parking_lot 0.11.2",
|
|
||||||
"reqwest",
|
|
||||||
"reqwest-middleware",
|
|
||||||
"retry-policies",
|
|
||||||
"task-local-extensions",
|
|
||||||
"tokio",
|
|
||||||
"tracing",
|
|
||||||
"wasm-timer",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "reqwest-tracing"
|
name = "reqwest-tracing"
|
||||||
version = "0.4.4"
|
version = "0.4.4"
|
||||||
@@ -3308,17 +3312,6 @@ dependencies = [
|
|||||||
"tracing-opentelemetry",
|
"tracing-opentelemetry",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "retry-policies"
|
|
||||||
version = "0.1.2"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "e09bbcb5003282bcb688f0bae741b278e9c7e8f378f561522c9806c58e075d9b"
|
|
||||||
dependencies = [
|
|
||||||
"anyhow",
|
|
||||||
"chrono",
|
|
||||||
"rand",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ring"
|
name = "ring"
|
||||||
version = "0.16.20"
|
version = "0.16.20"
|
||||||
@@ -3517,7 +3510,7 @@ dependencies = [
|
|||||||
"byteorder",
|
"byteorder",
|
||||||
"bytes",
|
"bytes",
|
||||||
"chrono",
|
"chrono",
|
||||||
"clap",
|
"clap 4.3.0",
|
||||||
"const_format",
|
"const_format",
|
||||||
"crc32c",
|
"crc32c",
|
||||||
"fs2",
|
"fs2",
|
||||||
@@ -3528,7 +3521,7 @@ dependencies = [
|
|||||||
"hyper",
|
"hyper",
|
||||||
"metrics",
|
"metrics",
|
||||||
"once_cell",
|
"once_cell",
|
||||||
"parking_lot 0.12.1",
|
"parking_lot",
|
||||||
"postgres",
|
"postgres",
|
||||||
"postgres-protocol",
|
"postgres-protocol",
|
||||||
"postgres_backend",
|
"postgres_backend",
|
||||||
@@ -3947,7 +3940,7 @@ dependencies = [
|
|||||||
"anyhow",
|
"anyhow",
|
||||||
"async-stream",
|
"async-stream",
|
||||||
"bytes",
|
"bytes",
|
||||||
"clap",
|
"clap 4.3.0",
|
||||||
"const_format",
|
"const_format",
|
||||||
"futures",
|
"futures",
|
||||||
"futures-core",
|
"futures-core",
|
||||||
@@ -3957,7 +3950,7 @@ dependencies = [
|
|||||||
"hyper",
|
"hyper",
|
||||||
"metrics",
|
"metrics",
|
||||||
"once_cell",
|
"once_cell",
|
||||||
"parking_lot 0.12.1",
|
"parking_lot",
|
||||||
"prost",
|
"prost",
|
||||||
"tokio",
|
"tokio",
|
||||||
"tokio-stream",
|
"tokio-stream",
|
||||||
@@ -4128,6 +4121,12 @@ dependencies = [
|
|||||||
"syn 1.0.109",
|
"syn 1.0.109",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "textwrap"
|
||||||
|
version = "0.16.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "222a222a5bfe1bba4a77b45ec488a741b3cb8872e5e499451fd7d0129c9c7c3d"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "thiserror"
|
name = "thiserror"
|
||||||
version = "1.0.40"
|
version = "1.0.40"
|
||||||
@@ -4285,7 +4284,7 @@ dependencies = [
|
|||||||
"futures-channel",
|
"futures-channel",
|
||||||
"futures-util",
|
"futures-util",
|
||||||
"log",
|
"log",
|
||||||
"parking_lot 0.12.1",
|
"parking_lot",
|
||||||
"percent-encoding",
|
"percent-encoding",
|
||||||
"phf",
|
"phf",
|
||||||
"pin-project-lite",
|
"pin-project-lite",
|
||||||
@@ -4543,7 +4542,7 @@ name = "trace"
|
|||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"clap",
|
"clap 4.3.0",
|
||||||
"pageserver_api",
|
"pageserver_api",
|
||||||
"utils",
|
"utils",
|
||||||
"workspace_hack",
|
"workspace_hack",
|
||||||
@@ -4645,6 +4644,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
|||||||
checksum = "30a651bc37f915e81f087d86e62a18eec5f79550c7faff886f7090b4ea757c77"
|
checksum = "30a651bc37f915e81f087d86e62a18eec5f79550c7faff886f7090b4ea757c77"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"matchers",
|
"matchers",
|
||||||
|
"nu-ansi-term",
|
||||||
"once_cell",
|
"once_cell",
|
||||||
"regex",
|
"regex",
|
||||||
"serde",
|
"serde",
|
||||||
@@ -4813,6 +4813,7 @@ version = "0.1.0"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"async-trait",
|
"async-trait",
|
||||||
|
"atty",
|
||||||
"bincode",
|
"bincode",
|
||||||
"byteorder",
|
"byteorder",
|
||||||
"bytes",
|
"bytes",
|
||||||
@@ -4889,7 +4890,7 @@ name = "wal_craft"
|
|||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"clap",
|
"clap 4.3.0",
|
||||||
"env_logger",
|
"env_logger",
|
||||||
"log",
|
"log",
|
||||||
"once_cell",
|
"once_cell",
|
||||||
@@ -4993,21 +4994,6 @@ version = "0.2.86"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "ed9d5b4305409d1fc9482fee2d7f9bcbf24b3972bf59817ef757e23982242a93"
|
checksum = "ed9d5b4305409d1fc9482fee2d7f9bcbf24b3972bf59817ef757e23982242a93"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "wasm-timer"
|
|
||||||
version = "0.2.5"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "be0ecb0db480561e9a7642b5d3e4187c128914e58aa84330b9493e3eb68c5e7f"
|
|
||||||
dependencies = [
|
|
||||||
"futures",
|
|
||||||
"js-sys",
|
|
||||||
"parking_lot 0.11.2",
|
|
||||||
"pin-utils",
|
|
||||||
"wasm-bindgen",
|
|
||||||
"wasm-bindgen-futures",
|
|
||||||
"web-sys",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "web-sys"
|
name = "web-sys"
|
||||||
version = "0.3.63"
|
version = "0.3.63"
|
||||||
@@ -5269,7 +5255,7 @@ dependencies = [
|
|||||||
"anyhow",
|
"anyhow",
|
||||||
"bytes",
|
"bytes",
|
||||||
"chrono",
|
"chrono",
|
||||||
"clap",
|
"clap 4.3.0",
|
||||||
"clap_builder",
|
"clap_builder",
|
||||||
"crossbeam-utils",
|
"crossbeam-utils",
|
||||||
"either",
|
"either",
|
||||||
|
|||||||
@@ -34,6 +34,7 @@ license = "Apache-2.0"
|
|||||||
anyhow = { version = "1.0", features = ["backtrace"] }
|
anyhow = { version = "1.0", features = ["backtrace"] }
|
||||||
async-stream = "0.3"
|
async-stream = "0.3"
|
||||||
async-trait = "0.1"
|
async-trait = "0.1"
|
||||||
|
atty = "0.2.14"
|
||||||
aws-config = { version = "0.55", default-features = false, features=["rustls"] }
|
aws-config = { version = "0.55", default-features = false, features=["rustls"] }
|
||||||
aws-sdk-s3 = "0.27"
|
aws-sdk-s3 = "0.27"
|
||||||
aws-smithy-http = "0.55"
|
aws-smithy-http = "0.55"
|
||||||
@@ -86,7 +87,6 @@ opentelemetry = "0.18.0"
|
|||||||
opentelemetry-otlp = { version = "0.11.0", default_features=false, features = ["http-proto", "trace", "http", "reqwest-client"] }
|
opentelemetry-otlp = { version = "0.11.0", default_features=false, features = ["http-proto", "trace", "http", "reqwest-client"] }
|
||||||
opentelemetry-semantic-conventions = "0.10.0"
|
opentelemetry-semantic-conventions = "0.10.0"
|
||||||
parking_lot = "0.12"
|
parking_lot = "0.12"
|
||||||
pbkdf2 = "0.12.1"
|
|
||||||
pin-project-lite = "0.2"
|
pin-project-lite = "0.2"
|
||||||
prometheus = {version = "0.13", default_features=false, features = ["process"]} # removes protobuf dependency
|
prometheus = {version = "0.13", default_features=false, features = ["process"]} # removes protobuf dependency
|
||||||
prost = "0.11"
|
prost = "0.11"
|
||||||
@@ -95,7 +95,6 @@ regex = "1.4"
|
|||||||
reqwest = { version = "0.11", default-features = false, features = ["rustls-tls"] }
|
reqwest = { version = "0.11", default-features = false, features = ["rustls-tls"] }
|
||||||
reqwest-tracing = { version = "0.4.0", features = ["opentelemetry_0_18"] }
|
reqwest-tracing = { version = "0.4.0", features = ["opentelemetry_0_18"] }
|
||||||
reqwest-middleware = "0.2.0"
|
reqwest-middleware = "0.2.0"
|
||||||
reqwest-retry = "0.2.2"
|
|
||||||
routerify = "3"
|
routerify = "3"
|
||||||
rpds = "0.13"
|
rpds = "0.13"
|
||||||
rustls = "0.20"
|
rustls = "0.20"
|
||||||
@@ -129,7 +128,7 @@ tonic = {version = "0.9", features = ["tls", "tls-roots"]}
|
|||||||
tracing = "0.1"
|
tracing = "0.1"
|
||||||
tracing-error = "0.2.0"
|
tracing-error = "0.2.0"
|
||||||
tracing-opentelemetry = "0.18.0"
|
tracing-opentelemetry = "0.18.0"
|
||||||
tracing-subscriber = { version = "0.3", default_features = false, features = ["smallvec", "fmt", "tracing-log", "std", "env-filter"] }
|
tracing-subscriber = { version = "0.3", features = ["env-filter"] }
|
||||||
url = "2.2"
|
url = "2.2"
|
||||||
uuid = { version = "1.2", features = ["v4", "serde"] }
|
uuid = { version = "1.2", features = ["v4", "serde"] }
|
||||||
walkdir = "2.3.2"
|
walkdir = "2.3.2"
|
||||||
@@ -171,7 +170,7 @@ utils = { version = "0.1", path = "./libs/utils/" }
|
|||||||
workspace_hack = { version = "0.1", path = "./workspace_hack/" }
|
workspace_hack = { version = "0.1", path = "./workspace_hack/" }
|
||||||
|
|
||||||
## Build dependencies
|
## Build dependencies
|
||||||
criterion = "0.5.1"
|
criterion = "0.4"
|
||||||
rcgen = "0.10"
|
rcgen = "0.10"
|
||||||
rstest = "0.17"
|
rstest = "0.17"
|
||||||
tempfile = "3.4"
|
tempfile = "3.4"
|
||||||
|
|||||||
@@ -189,8 +189,8 @@ RUN wget https://github.com/df7cb/postgresql-unit/archive/refs/tags/7.7.tar.gz -
|
|||||||
FROM build-deps AS vector-pg-build
|
FROM build-deps AS vector-pg-build
|
||||||
COPY --from=pg-build /usr/local/pgsql/ /usr/local/pgsql/
|
COPY --from=pg-build /usr/local/pgsql/ /usr/local/pgsql/
|
||||||
|
|
||||||
RUN wget https://github.com/pgvector/pgvector/archive/refs/tags/v0.4.4.tar.gz -O pgvector.tar.gz && \
|
RUN wget https://github.com/pgvector/pgvector/archive/refs/tags/v0.4.0.tar.gz -O pgvector.tar.gz && \
|
||||||
echo "1cb70a63f8928e396474796c22a20be9f7285a8a013009deb8152445b61b72e6 pgvector.tar.gz" | sha256sum --check && \
|
echo "b76cf84ddad452cc880a6c8c661d137ddd8679c000a16332f4f03ecf6e10bcc8 pgvector.tar.gz" | sha256sum --check && \
|
||||||
mkdir pgvector-src && cd pgvector-src && tar xvzf ../pgvector.tar.gz --strip-components=1 -C . && \
|
mkdir pgvector-src && cd pgvector-src && tar xvzf ../pgvector.tar.gz --strip-components=1 -C . && \
|
||||||
make -j $(getconf _NPROCESSORS_ONLN) PG_CONFIG=/usr/local/pgsql/bin/pg_config && \
|
make -j $(getconf _NPROCESSORS_ONLN) PG_CONFIG=/usr/local/pgsql/bin/pg_config && \
|
||||||
make -j $(getconf _NPROCESSORS_ONLN) install PG_CONFIG=/usr/local/pgsql/bin/pg_config && \
|
make -j $(getconf _NPROCESSORS_ONLN) install PG_CONFIG=/usr/local/pgsql/bin/pg_config && \
|
||||||
@@ -481,60 +481,6 @@ RUN wget https://github.com/rdkit/rdkit/archive/refs/tags/Release_2023_03_1.tar.
|
|||||||
make -j $(getconf _NPROCESSORS_ONLN) install && \
|
make -j $(getconf _NPROCESSORS_ONLN) install && \
|
||||||
echo 'trusted = true' >> /usr/local/pgsql/share/extension/rdkit.control
|
echo 'trusted = true' >> /usr/local/pgsql/share/extension/rdkit.control
|
||||||
|
|
||||||
#########################################################################################
|
|
||||||
#
|
|
||||||
# Layer "pg-uuidv7-pg-build"
|
|
||||||
# compile pg_uuidv7 extension
|
|
||||||
#
|
|
||||||
#########################################################################################
|
|
||||||
FROM build-deps AS pg-uuidv7-pg-build
|
|
||||||
COPY --from=pg-build /usr/local/pgsql/ /usr/local/pgsql/
|
|
||||||
|
|
||||||
ENV PATH "/usr/local/pgsql/bin/:$PATH"
|
|
||||||
RUN wget https://github.com/fboulnois/pg_uuidv7/archive/refs/tags/v1.0.1.tar.gz -O pg_uuidv7.tar.gz && \
|
|
||||||
echo "0d0759ab01b7fb23851ecffb0bce27822e1868a4a5819bfd276101c716637a7a pg_uuidv7.tar.gz" | sha256sum --check && \
|
|
||||||
mkdir pg_uuidv7-src && cd pg_uuidv7-src && tar xvzf ../pg_uuidv7.tar.gz --strip-components=1 -C . && \
|
|
||||||
make -j $(getconf _NPROCESSORS_ONLN) && \
|
|
||||||
make -j $(getconf _NPROCESSORS_ONLN) install && \
|
|
||||||
echo 'trusted = true' >> /usr/local/pgsql/share/extension/pg_uuidv7.control
|
|
||||||
|
|
||||||
#########################################################################################
|
|
||||||
#
|
|
||||||
# Layer "pg-roaringbitmap-pg-build"
|
|
||||||
# compile pg_roaringbitmap extension
|
|
||||||
#
|
|
||||||
#########################################################################################
|
|
||||||
FROM build-deps AS pg-roaringbitmap-pg-build
|
|
||||||
COPY --from=pg-build /usr/local/pgsql/ /usr/local/pgsql/
|
|
||||||
|
|
||||||
ENV PATH "/usr/local/pgsql/bin/:$PATH"
|
|
||||||
RUN wget https://github.com/ChenHuajun/pg_roaringbitmap/archive/refs/tags/v0.5.4.tar.gz -O pg_roaringbitmap.tar.gz && \
|
|
||||||
echo "b75201efcb1c2d1b014ec4ae6a22769cc7a224e6e406a587f5784a37b6b5a2aa pg_roaringbitmap.tar.gz" | sha256sum --check && \
|
|
||||||
mkdir pg_roaringbitmap-src && cd pg_roaringbitmap-src && tar xvzf ../pg_roaringbitmap.tar.gz --strip-components=1 -C . && \
|
|
||||||
make -j $(getconf _NPROCESSORS_ONLN) && \
|
|
||||||
make -j $(getconf _NPROCESSORS_ONLN) install && \
|
|
||||||
echo 'trusted = true' >> /usr/local/pgsql/share/extension/roaringbitmap.control
|
|
||||||
|
|
||||||
#########################################################################################
|
|
||||||
#
|
|
||||||
# Layer "pg-anon-pg-build"
|
|
||||||
# compile anon extension
|
|
||||||
#
|
|
||||||
#########################################################################################
|
|
||||||
FROM build-deps AS pg-anon-pg-build
|
|
||||||
COPY --from=pg-build /usr/local/pgsql/ /usr/local/pgsql/
|
|
||||||
|
|
||||||
# Kaniko doesn't allow to do `${from#/usr/local/pgsql/}`, so we use `${from:17}` instead
|
|
||||||
ENV PATH "/usr/local/pgsql/bin/:$PATH"
|
|
||||||
RUN wget https://gitlab.com/dalibo/postgresql_anonymizer/-/archive/1.1.0/postgresql_anonymizer-1.1.0.tar.gz -O pg_anon.tar.gz && \
|
|
||||||
echo "08b09d2ff9b962f96c60db7e6f8e79cf7253eb8772516998fc35ece08633d3ad pg_anon.tar.gz" | sha256sum --check && \
|
|
||||||
mkdir pg_anon-src && cd pg_anon-src && tar xvzf ../pg_anon.tar.gz --strip-components=1 -C . && \
|
|
||||||
find /usr/local/pgsql -type f | sort > /before.txt && \
|
|
||||||
make -j $(getconf _NPROCESSORS_ONLN) install PG_CONFIG=/usr/local/pgsql/bin/pg_config && \
|
|
||||||
echo 'trusted = true' >> /usr/local/pgsql/share/extension/anon.control && \
|
|
||||||
find /usr/local/pgsql -type f | sort > /after.txt && \
|
|
||||||
/bin/bash -c 'for from in $(comm -13 /before.txt /after.txt); do to=/extensions/anon/${from:17} && mkdir -p $(dirname ${to}) && cp -a ${from} ${to}; done'
|
|
||||||
|
|
||||||
#########################################################################################
|
#########################################################################################
|
||||||
#
|
#
|
||||||
# Layer "rust extensions"
|
# Layer "rust extensions"
|
||||||
@@ -643,7 +589,6 @@ RUN wget https://github.com/pksunkara/pgx_ulid/archive/refs/tags/v0.1.0.tar.gz -
|
|||||||
#
|
#
|
||||||
#########################################################################################
|
#########################################################################################
|
||||||
FROM build-deps AS neon-pg-ext-build
|
FROM build-deps AS neon-pg-ext-build
|
||||||
# Public extensions
|
|
||||||
COPY --from=postgis-build /usr/local/pgsql/ /usr/local/pgsql/
|
COPY --from=postgis-build /usr/local/pgsql/ /usr/local/pgsql/
|
||||||
COPY --from=postgis-build /sfcgal/* /
|
COPY --from=postgis-build /sfcgal/* /
|
||||||
COPY --from=plv8-build /usr/local/pgsql/ /usr/local/pgsql/
|
COPY --from=plv8-build /usr/local/pgsql/ /usr/local/pgsql/
|
||||||
@@ -669,8 +614,6 @@ COPY --from=kq-imcx-pg-build /usr/local/pgsql/ /usr/local/pgsql/
|
|||||||
COPY --from=pg-cron-pg-build /usr/local/pgsql/ /usr/local/pgsql/
|
COPY --from=pg-cron-pg-build /usr/local/pgsql/ /usr/local/pgsql/
|
||||||
COPY --from=pg-pgx-ulid-build /usr/local/pgsql/ /usr/local/pgsql/
|
COPY --from=pg-pgx-ulid-build /usr/local/pgsql/ /usr/local/pgsql/
|
||||||
COPY --from=rdkit-pg-build /usr/local/pgsql/ /usr/local/pgsql/
|
COPY --from=rdkit-pg-build /usr/local/pgsql/ /usr/local/pgsql/
|
||||||
COPY --from=pg-uuidv7-pg-build /usr/local/pgsql/ /usr/local/pgsql/
|
|
||||||
COPY --from=pg-roaringbitmap-pg-build /usr/local/pgsql/ /usr/local/pgsql/
|
|
||||||
COPY pgxn/ pgxn/
|
COPY pgxn/ pgxn/
|
||||||
|
|
||||||
RUN make -j $(getconf _NPROCESSORS_ONLN) \
|
RUN make -j $(getconf _NPROCESSORS_ONLN) \
|
||||||
@@ -719,22 +662,6 @@ RUN rm -r /usr/local/pgsql/include
|
|||||||
# if they were to be used by other libraries.
|
# if they were to be used by other libraries.
|
||||||
RUN rm /usr/local/pgsql/lib/lib*.a
|
RUN rm /usr/local/pgsql/lib/lib*.a
|
||||||
|
|
||||||
#########################################################################################
|
|
||||||
#
|
|
||||||
# Extenstion only
|
|
||||||
#
|
|
||||||
#########################################################################################
|
|
||||||
FROM scratch AS postgres-extensions
|
|
||||||
# After the transition this layer will include all extensitons.
|
|
||||||
# As for now, it's only for new custom ones
|
|
||||||
#
|
|
||||||
# # Default extensions
|
|
||||||
# COPY --from=postgres-cleanup-layer /usr/local/pgsql/share/extension /usr/local/pgsql/share/extension
|
|
||||||
# COPY --from=postgres-cleanup-layer /usr/local/pgsql/lib /usr/local/pgsql/lib
|
|
||||||
# Custom extensions
|
|
||||||
COPY --from=pg-anon-pg-build /extensions/anon/lib/ /extensions/anon/lib
|
|
||||||
COPY --from=pg-anon-pg-build /extensions/anon/share/extension /extensions/anon/share/extension
|
|
||||||
|
|
||||||
#########################################################################################
|
#########################################################################################
|
||||||
#
|
#
|
||||||
# Final layer
|
# Final layer
|
||||||
|
|||||||
16
README.md
16
README.md
@@ -132,13 +132,13 @@ Python (3.9 or higher), and install python3 packages using `./scripts/pysync` (r
|
|||||||
# Create repository in .neon with proper paths to binaries and data
|
# Create repository in .neon with proper paths to binaries and data
|
||||||
# Later that would be responsibility of a package install script
|
# Later that would be responsibility of a package install script
|
||||||
> cargo neon init
|
> cargo neon init
|
||||||
Initializing pageserver node 1 at '127.0.0.1:64000' in ".neon"
|
Starting pageserver at '127.0.0.1:64000' in '.neon'.
|
||||||
|
|
||||||
# start pageserver, safekeeper, and broker for their intercommunication
|
# start pageserver, safekeeper, and broker for their intercommunication
|
||||||
> cargo neon start
|
> cargo neon start
|
||||||
Starting neon broker at 127.0.0.1:50051.
|
Starting neon broker at 127.0.0.1:50051
|
||||||
storage_broker started, pid: 2918372
|
storage_broker started, pid: 2918372
|
||||||
Starting pageserver node 1 at '127.0.0.1:64000' in ".neon".
|
Starting pageserver at '127.0.0.1:64000' in '.neon'.
|
||||||
pageserver started, pid: 2918386
|
pageserver started, pid: 2918386
|
||||||
Starting safekeeper at '127.0.0.1:5454' in '.neon/safekeepers/sk1'.
|
Starting safekeeper at '127.0.0.1:5454' in '.neon/safekeepers/sk1'.
|
||||||
safekeeper 1 started, pid: 2918437
|
safekeeper 1 started, pid: 2918437
|
||||||
@@ -152,7 +152,8 @@ Setting tenant 9ef87a5bf0d92544f6fafeeb3239695c as a default one
|
|||||||
# start postgres compute node
|
# start postgres compute node
|
||||||
> cargo neon endpoint start main
|
> cargo neon endpoint start main
|
||||||
Starting new endpoint main (PostgreSQL v14) on timeline de200bd42b49cc1814412c7e592dd6e9 ...
|
Starting new endpoint main (PostgreSQL v14) on timeline de200bd42b49cc1814412c7e592dd6e9 ...
|
||||||
Starting postgres at 'postgresql://cloud_admin@127.0.0.1:55432/postgres'
|
Extracting base backup to create postgres instance: path=.neon/pgdatadirs/tenants/9ef87a5bf0d92544f6fafeeb3239695c/main port=55432
|
||||||
|
Starting postgres at 'host=127.0.0.1 port=55432 user=cloud_admin dbname=postgres'
|
||||||
|
|
||||||
# check list of running postgres instances
|
# check list of running postgres instances
|
||||||
> cargo neon endpoint list
|
> cargo neon endpoint list
|
||||||
@@ -188,17 +189,18 @@ Created timeline 'b3b863fa45fa9e57e615f9f2d944e601' at Lsn 0/16F9A00 for tenant:
|
|||||||
# start postgres on that branch
|
# start postgres on that branch
|
||||||
> cargo neon endpoint start migration_check --branch-name migration_check
|
> cargo neon endpoint start migration_check --branch-name migration_check
|
||||||
Starting new endpoint migration_check (PostgreSQL v14) on timeline b3b863fa45fa9e57e615f9f2d944e601 ...
|
Starting new endpoint migration_check (PostgreSQL v14) on timeline b3b863fa45fa9e57e615f9f2d944e601 ...
|
||||||
Starting postgres at 'postgresql://cloud_admin@127.0.0.1:55434/postgres'
|
Extracting base backup to create postgres instance: path=.neon/pgdatadirs/tenants/9ef87a5bf0d92544f6fafeeb3239695c/migration_check port=55433
|
||||||
|
Starting postgres at 'host=127.0.0.1 port=55433 user=cloud_admin dbname=postgres'
|
||||||
|
|
||||||
# check the new list of running postgres instances
|
# check the new list of running postgres instances
|
||||||
> cargo neon endpoint list
|
> cargo neon endpoint list
|
||||||
ENDPOINT ADDRESS TIMELINE BRANCH NAME LSN STATUS
|
ENDPOINT ADDRESS TIMELINE BRANCH NAME LSN STATUS
|
||||||
main 127.0.0.1:55432 de200bd42b49cc1814412c7e592dd6e9 main 0/16F9A38 running
|
main 127.0.0.1:55432 de200bd42b49cc1814412c7e592dd6e9 main 0/16F9A38 running
|
||||||
migration_check 127.0.0.1:55434 b3b863fa45fa9e57e615f9f2d944e601 migration_check 0/16F9A70 running
|
migration_check 127.0.0.1:55433 b3b863fa45fa9e57e615f9f2d944e601 migration_check 0/16F9A70 running
|
||||||
|
|
||||||
# this new postgres instance will have all the data from 'main' postgres,
|
# this new postgres instance will have all the data from 'main' postgres,
|
||||||
# but all modifications would not affect data in original postgres
|
# but all modifications would not affect data in original postgres
|
||||||
> psql -p55434 -h 127.0.0.1 -U cloud_admin postgres
|
> psql -p55433 -h 127.0.0.1 -U cloud_admin postgres
|
||||||
postgres=# select * from t;
|
postgres=# select * from t;
|
||||||
key | value
|
key | value
|
||||||
-----+-------
|
-----+-------
|
||||||
|
|||||||
@@ -30,3 +30,5 @@ url.workspace = true
|
|||||||
compute_api.workspace = true
|
compute_api.workspace = true
|
||||||
utils.workspace = true
|
utils.workspace = true
|
||||||
workspace_hack.workspace = true
|
workspace_hack.workspace = true
|
||||||
|
toml_edit.workspace = true
|
||||||
|
remote_storage = { version = "0.1", path = "../libs/remote_storage/" }
|
||||||
|
|||||||
@@ -27,7 +27,8 @@
|
|||||||
//! compute_ctl -D /var/db/postgres/compute \
|
//! compute_ctl -D /var/db/postgres/compute \
|
||||||
//! -C 'postgresql://cloud_admin@localhost/postgres' \
|
//! -C 'postgresql://cloud_admin@localhost/postgres' \
|
||||||
//! -S /var/db/postgres/specs/current.json \
|
//! -S /var/db/postgres/specs/current.json \
|
||||||
//! -b /usr/local/bin/postgres
|
//! -b /usr/local/bin/postgres \
|
||||||
|
//! -r {"bucket": "my-bucket", "region": "eu-central-1"}
|
||||||
//! ```
|
//! ```
|
||||||
//!
|
//!
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
@@ -48,12 +49,16 @@ use compute_api::responses::ComputeStatus;
|
|||||||
|
|
||||||
use compute_tools::compute::{ComputeNode, ComputeState, ParsedSpec};
|
use compute_tools::compute::{ComputeNode, ComputeState, ParsedSpec};
|
||||||
use compute_tools::configurator::launch_configurator;
|
use compute_tools::configurator::launch_configurator;
|
||||||
|
use compute_tools::extension_server::{
|
||||||
|
download_extension, get_availiable_extensions, init_remote_storage,
|
||||||
|
};
|
||||||
use compute_tools::http::api::launch_http_server;
|
use compute_tools::http::api::launch_http_server;
|
||||||
use compute_tools::logger::*;
|
use compute_tools::logger::*;
|
||||||
use compute_tools::monitor::launch_monitor;
|
use compute_tools::monitor::launch_monitor;
|
||||||
use compute_tools::params::*;
|
use compute_tools::params::*;
|
||||||
use compute_tools::spec::*;
|
use compute_tools::spec::*;
|
||||||
|
|
||||||
|
use tokio::runtime::Runtime;
|
||||||
const BUILD_TAG_DEFAULT: &str = "local";
|
const BUILD_TAG_DEFAULT: &str = "local";
|
||||||
|
|
||||||
fn main() -> Result<()> {
|
fn main() -> Result<()> {
|
||||||
@@ -64,6 +69,23 @@ fn main() -> Result<()> {
|
|||||||
info!("build_tag: {build_tag}");
|
info!("build_tag: {build_tag}");
|
||||||
|
|
||||||
let matches = cli().get_matches();
|
let matches = cli().get_matches();
|
||||||
|
let pgbin_default = String::from("postgres");
|
||||||
|
let pgbin = matches.get_one::<String>("pgbin").unwrap_or(&pgbin_default);
|
||||||
|
|
||||||
|
let remote_ext_config = matches.get_one::<String>("remote-ext-config");
|
||||||
|
let ext_remote_storage = match remote_ext_config {
|
||||||
|
Some(x) => Some(init_remote_storage(x)?),
|
||||||
|
None => None,
|
||||||
|
};
|
||||||
|
|
||||||
|
let rt = Runtime::new().unwrap();
|
||||||
|
let copy_remote_storage = ext_remote_storage.clone();
|
||||||
|
|
||||||
|
// rt.block_on(async move {
|
||||||
|
// download_extension(©_remote_storage, ExtensionType::Shared, pgbin)
|
||||||
|
// .await
|
||||||
|
// .expect("download extension should work");
|
||||||
|
// });
|
||||||
|
|
||||||
let http_port = *matches
|
let http_port = *matches
|
||||||
.get_one::<u16>("http-port")
|
.get_one::<u16>("http-port")
|
||||||
@@ -128,9 +150,6 @@ fn main() -> Result<()> {
|
|||||||
let compute_id = matches.get_one::<String>("compute-id");
|
let compute_id = matches.get_one::<String>("compute-id");
|
||||||
let control_plane_uri = matches.get_one::<String>("control-plane-uri");
|
let control_plane_uri = matches.get_one::<String>("control-plane-uri");
|
||||||
|
|
||||||
// Try to use just 'postgres' if no path is provided
|
|
||||||
let pgbin = matches.get_one::<String>("pgbin").unwrap();
|
|
||||||
|
|
||||||
let spec;
|
let spec;
|
||||||
let mut live_config_allowed = false;
|
let mut live_config_allowed = false;
|
||||||
match spec_json {
|
match spec_json {
|
||||||
@@ -182,6 +201,9 @@ fn main() -> Result<()> {
|
|||||||
live_config_allowed,
|
live_config_allowed,
|
||||||
state: Mutex::new(new_state),
|
state: Mutex::new(new_state),
|
||||||
state_changed: Condvar::new(),
|
state_changed: Condvar::new(),
|
||||||
|
ext_remote_storage,
|
||||||
|
availiable_extensions: Vec::new(),
|
||||||
|
availiable_libraries: Vec::new(),
|
||||||
};
|
};
|
||||||
let compute = Arc::new(compute_node);
|
let compute = Arc::new(compute_node);
|
||||||
|
|
||||||
@@ -190,6 +212,21 @@ fn main() -> Result<()> {
|
|||||||
let _http_handle =
|
let _http_handle =
|
||||||
launch_http_server(http_port, &compute).expect("cannot launch http endpoint thread");
|
launch_http_server(http_port, &compute).expect("cannot launch http endpoint thread");
|
||||||
|
|
||||||
|
let extension_server_port: u16 = http_port;
|
||||||
|
|
||||||
|
// exen before we have spec, we can get public availiable extensions
|
||||||
|
// TODO turn get_availiable_extensions() & other functions into ComputeNode method,
|
||||||
|
// we pass to many params from it anyways..
|
||||||
|
|
||||||
|
compute_node.availiable_extensions = get_availiable_extensions(
|
||||||
|
ext_remote_storage,
|
||||||
|
pg_version, //TODO
|
||||||
|
pgbin,
|
||||||
|
None,
|
||||||
|
);
|
||||||
|
|
||||||
|
// TODO same for libraries
|
||||||
|
|
||||||
if !spec_set {
|
if !spec_set {
|
||||||
// No spec provided, hang waiting for it.
|
// No spec provided, hang waiting for it.
|
||||||
info!("no compute spec provided, waiting");
|
info!("no compute spec provided, waiting");
|
||||||
@@ -227,10 +264,21 @@ fn main() -> Result<()> {
|
|||||||
let _configurator_handle =
|
let _configurator_handle =
|
||||||
launch_configurator(&compute).expect("cannot launch configurator thread");
|
launch_configurator(&compute).expect("cannot launch configurator thread");
|
||||||
|
|
||||||
|
// download private tenant extensions before postgres start
|
||||||
|
// TODO
|
||||||
|
// compute_node.availiable_extensions = get_availiable_extensions(ext_remote_storage,
|
||||||
|
// pg_version, //TODO
|
||||||
|
// pgbin,
|
||||||
|
// tenant_id); //TODO get tenant_id from spec
|
||||||
|
|
||||||
|
// download preload shared libraries before postgres start (if any)
|
||||||
|
// TODO
|
||||||
|
// download_library_file();
|
||||||
|
|
||||||
// Start Postgres
|
// Start Postgres
|
||||||
let mut delay_exit = false;
|
let mut delay_exit = false;
|
||||||
let mut exit_code = None;
|
let mut exit_code = None;
|
||||||
let pg = match compute.start_compute() {
|
let pg = match compute.start_compute(extension_server_port) {
|
||||||
Ok(pg) => Some(pg),
|
Ok(pg) => Some(pg),
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
error!("could not start the compute node: {:?}", err);
|
error!("could not start the compute node: {:?}", err);
|
||||||
@@ -256,16 +304,6 @@ fn main() -> Result<()> {
|
|||||||
exit_code = ecode.code()
|
exit_code = ecode.code()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Maybe sync safekeepers again, to speed up next startup
|
|
||||||
let compute_state = compute.state.lock().unwrap().clone();
|
|
||||||
let pspec = compute_state.pspec.as_ref().expect("spec must be set");
|
|
||||||
if matches!(pspec.spec.mode, compute_api::spec::ComputeMode::Primary) {
|
|
||||||
info!("syncing safekeepers on shutdown");
|
|
||||||
let storage_auth_token = pspec.storage_auth_token.clone();
|
|
||||||
let lsn = compute.sync_safekeepers(storage_auth_token)?;
|
|
||||||
info!("synced safekeepers at lsn {lsn}");
|
|
||||||
}
|
|
||||||
|
|
||||||
if let Err(err) = compute.check_for_core_dumps() {
|
if let Err(err) = compute.check_for_core_dumps() {
|
||||||
error!("error while checking for core dumps: {err:?}");
|
error!("error while checking for core dumps: {err:?}");
|
||||||
}
|
}
|
||||||
@@ -359,6 +397,12 @@ fn cli() -> clap::Command {
|
|||||||
.long("control-plane-uri")
|
.long("control-plane-uri")
|
||||||
.value_name("CONTROL_PLANE_API_BASE_URI"),
|
.value_name("CONTROL_PLANE_API_BASE_URI"),
|
||||||
)
|
)
|
||||||
|
.arg(
|
||||||
|
Arg::new("remote-ext-config")
|
||||||
|
.short('r')
|
||||||
|
.long("remote-ext-config")
|
||||||
|
.value_name("REMOTE_EXT_CONFIG"),
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
|||||||
@@ -16,6 +16,8 @@ use utils::lsn::Lsn;
|
|||||||
use compute_api::responses::{ComputeMetrics, ComputeStatus};
|
use compute_api::responses::{ComputeMetrics, ComputeStatus};
|
||||||
use compute_api::spec::{ComputeMode, ComputeSpec};
|
use compute_api::spec::{ComputeMode, ComputeSpec};
|
||||||
|
|
||||||
|
use remote_storage::{GenericRemoteStorage, RemotePath};
|
||||||
|
|
||||||
use crate::config;
|
use crate::config;
|
||||||
use crate::pg_helpers::*;
|
use crate::pg_helpers::*;
|
||||||
use crate::spec::*;
|
use crate::spec::*;
|
||||||
@@ -45,6 +47,10 @@ pub struct ComputeNode {
|
|||||||
pub state: Mutex<ComputeState>,
|
pub state: Mutex<ComputeState>,
|
||||||
/// `Condvar` to allow notifying waiters about state changes.
|
/// `Condvar` to allow notifying waiters about state changes.
|
||||||
pub state_changed: Condvar,
|
pub state_changed: Condvar,
|
||||||
|
/// S3 extensions configuration variables
|
||||||
|
pub ext_remote_storage: Option<GenericRemoteStorage>,
|
||||||
|
pub availiable_extensions: Vec<RemotePath>,
|
||||||
|
pub availiable_libraries: Vec<RemotePath>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
@@ -133,84 +139,6 @@ impl TryFrom<ComputeSpec> for ParsedSpec {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Create special neon_superuser role, that's a slightly nerfed version of a real superuser
|
|
||||||
/// that we give to customers
|
|
||||||
fn create_neon_superuser(spec: &ComputeSpec, client: &mut Client) -> Result<()> {
|
|
||||||
let roles = spec
|
|
||||||
.cluster
|
|
||||||
.roles
|
|
||||||
.iter()
|
|
||||||
.map(|r| format!("'{}'", escape_literal(&r.name)))
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
|
|
||||||
let dbs = spec
|
|
||||||
.cluster
|
|
||||||
.databases
|
|
||||||
.iter()
|
|
||||||
.map(|db| format!("'{}'", escape_literal(&db.name)))
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
|
|
||||||
let roles_decl = if roles.is_empty() {
|
|
||||||
String::from("roles text[] := NULL;")
|
|
||||||
} else {
|
|
||||||
format!(
|
|
||||||
r#"
|
|
||||||
roles text[] := ARRAY(SELECT rolname
|
|
||||||
FROM pg_catalog.pg_roles
|
|
||||||
WHERE rolname IN ({}));"#,
|
|
||||||
roles.join(", ")
|
|
||||||
)
|
|
||||||
};
|
|
||||||
|
|
||||||
let database_decl = if dbs.is_empty() {
|
|
||||||
String::from("dbs text[] := NULL;")
|
|
||||||
} else {
|
|
||||||
format!(
|
|
||||||
r#"
|
|
||||||
dbs text[] := ARRAY(SELECT datname
|
|
||||||
FROM pg_catalog.pg_database
|
|
||||||
WHERE datname IN ({}));"#,
|
|
||||||
dbs.join(", ")
|
|
||||||
)
|
|
||||||
};
|
|
||||||
|
|
||||||
// ALL PRIVILEGES grants CREATE, CONNECT, and TEMPORARY on all databases
|
|
||||||
// (see https://www.postgresql.org/docs/current/ddl-priv.html)
|
|
||||||
let query = format!(
|
|
||||||
r#"
|
|
||||||
DO $$
|
|
||||||
DECLARE
|
|
||||||
r text;
|
|
||||||
{}
|
|
||||||
{}
|
|
||||||
BEGIN
|
|
||||||
IF NOT EXISTS (
|
|
||||||
SELECT FROM pg_catalog.pg_roles WHERE rolname = 'neon_superuser')
|
|
||||||
THEN
|
|
||||||
CREATE ROLE neon_superuser CREATEDB CREATEROLE NOLOGIN IN ROLE pg_read_all_data, pg_write_all_data;
|
|
||||||
IF array_length(roles, 1) IS NOT NULL THEN
|
|
||||||
EXECUTE format('GRANT neon_superuser TO %s',
|
|
||||||
array_to_string(ARRAY(SELECT quote_ident(x) FROM unnest(roles) as x), ', '));
|
|
||||||
FOREACH r IN ARRAY roles LOOP
|
|
||||||
EXECUTE format('ALTER ROLE %s CREATEROLE CREATEDB', quote_ident(r));
|
|
||||||
END LOOP;
|
|
||||||
END IF;
|
|
||||||
IF array_length(dbs, 1) IS NOT NULL THEN
|
|
||||||
EXECUTE format('GRANT ALL PRIVILEGES ON DATABASE %s TO neon_superuser',
|
|
||||||
array_to_string(ARRAY(SELECT quote_ident(x) FROM unnest(dbs) as x), ', '));
|
|
||||||
END IF;
|
|
||||||
END IF;
|
|
||||||
END
|
|
||||||
$$;"#,
|
|
||||||
roles_decl, database_decl,
|
|
||||||
);
|
|
||||||
info!("Neon superuser created:\n{}", &query);
|
|
||||||
client
|
|
||||||
.simple_query(&query)
|
|
||||||
.map_err(|e| anyhow::anyhow!(e).context(query))?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ComputeNode {
|
impl ComputeNode {
|
||||||
pub fn set_status(&self, status: ComputeStatus) {
|
pub fn set_status(&self, status: ComputeStatus) {
|
||||||
let mut state = self.state.lock().unwrap();
|
let mut state = self.state.lock().unwrap();
|
||||||
@@ -235,7 +163,7 @@ impl ComputeNode {
|
|||||||
|
|
||||||
// Get basebackup from the libpq connection to pageserver using `connstr` and
|
// Get basebackup from the libpq connection to pageserver using `connstr` and
|
||||||
// unarchive it to `pgdata` directory overriding all its previous content.
|
// unarchive it to `pgdata` directory overriding all its previous content.
|
||||||
#[instrument(skip_all, fields(%lsn))]
|
#[instrument(skip(self, compute_state))]
|
||||||
fn get_basebackup(&self, compute_state: &ComputeState, lsn: Lsn) -> Result<()> {
|
fn get_basebackup(&self, compute_state: &ComputeState, lsn: Lsn) -> Result<()> {
|
||||||
let spec = compute_state.pspec.as_ref().expect("spec must be set");
|
let spec = compute_state.pspec.as_ref().expect("spec must be set");
|
||||||
let start_time = Utc::now();
|
let start_time = Utc::now();
|
||||||
@@ -277,8 +205,8 @@ impl ComputeNode {
|
|||||||
|
|
||||||
// Run `postgres` in a special mode with `--sync-safekeepers` argument
|
// Run `postgres` in a special mode with `--sync-safekeepers` argument
|
||||||
// and return the reported LSN back to the caller.
|
// and return the reported LSN back to the caller.
|
||||||
#[instrument(skip_all)]
|
#[instrument(skip(self, storage_auth_token))]
|
||||||
pub fn sync_safekeepers(&self, storage_auth_token: Option<String>) -> Result<Lsn> {
|
fn sync_safekeepers(&self, storage_auth_token: Option<String>) -> Result<Lsn> {
|
||||||
let start_time = Utc::now();
|
let start_time = Utc::now();
|
||||||
|
|
||||||
let sync_handle = Command::new(&self.pgbin)
|
let sync_handle = Command::new(&self.pgbin)
|
||||||
@@ -322,15 +250,23 @@ impl ComputeNode {
|
|||||||
|
|
||||||
/// Do all the preparations like PGDATA directory creation, configuration,
|
/// Do all the preparations like PGDATA directory creation, configuration,
|
||||||
/// safekeepers sync, basebackup, etc.
|
/// safekeepers sync, basebackup, etc.
|
||||||
#[instrument(skip_all)]
|
#[instrument(skip(self, compute_state))]
|
||||||
pub fn prepare_pgdata(&self, compute_state: &ComputeState) -> Result<()> {
|
pub fn prepare_pgdata(
|
||||||
|
&self,
|
||||||
|
compute_state: &ComputeState,
|
||||||
|
extension_server_port: u16,
|
||||||
|
) -> Result<()> {
|
||||||
let pspec = compute_state.pspec.as_ref().expect("spec must be set");
|
let pspec = compute_state.pspec.as_ref().expect("spec must be set");
|
||||||
let spec = &pspec.spec;
|
let spec = &pspec.spec;
|
||||||
let pgdata_path = Path::new(&self.pgdata);
|
let pgdata_path = Path::new(&self.pgdata);
|
||||||
|
|
||||||
// Remove/create an empty pgdata directory and put configuration there.
|
// Remove/create an empty pgdata directory and put configuration there.
|
||||||
self.create_pgdata()?;
|
self.create_pgdata()?;
|
||||||
config::write_postgres_conf(&pgdata_path.join("postgresql.conf"), &pspec.spec)?;
|
config::write_postgres_conf(
|
||||||
|
&pgdata_path.join("postgresql.conf"),
|
||||||
|
&pspec.spec,
|
||||||
|
Some(extension_server_port),
|
||||||
|
)?;
|
||||||
|
|
||||||
// Syncing safekeepers is only safe with primary nodes: if a primary
|
// Syncing safekeepers is only safe with primary nodes: if a primary
|
||||||
// is already connected it will be kicked out, so a secondary (standby)
|
// is already connected it will be kicked out, so a secondary (standby)
|
||||||
@@ -380,7 +316,7 @@ impl ComputeNode {
|
|||||||
|
|
||||||
/// Start Postgres as a child process and manage DBs/roles.
|
/// Start Postgres as a child process and manage DBs/roles.
|
||||||
/// After that this will hang waiting on the postmaster process to exit.
|
/// After that this will hang waiting on the postmaster process to exit.
|
||||||
#[instrument(skip_all)]
|
#[instrument(skip(self))]
|
||||||
pub fn start_postgres(
|
pub fn start_postgres(
|
||||||
&self,
|
&self,
|
||||||
storage_auth_token: Option<String>,
|
storage_auth_token: Option<String>,
|
||||||
@@ -404,7 +340,7 @@ impl ComputeNode {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Do initial configuration of the already started Postgres.
|
/// Do initial configuration of the already started Postgres.
|
||||||
#[instrument(skip_all)]
|
#[instrument(skip(self, compute_state))]
|
||||||
pub fn apply_config(&self, compute_state: &ComputeState) -> Result<()> {
|
pub fn apply_config(&self, compute_state: &ComputeState) -> Result<()> {
|
||||||
// If connection fails,
|
// If connection fails,
|
||||||
// it may be the old node with `zenith_admin` superuser.
|
// it may be the old node with `zenith_admin` superuser.
|
||||||
@@ -425,8 +361,6 @@ impl ComputeNode {
|
|||||||
.map_err(|_| anyhow::anyhow!("invalid connstr"))?;
|
.map_err(|_| anyhow::anyhow!("invalid connstr"))?;
|
||||||
|
|
||||||
let mut client = Client::connect(zenith_admin_connstr.as_str(), NoTls)?;
|
let mut client = Client::connect(zenith_admin_connstr.as_str(), NoTls)?;
|
||||||
// Disable forwarding so that users don't get a cloud_admin role
|
|
||||||
client.simple_query("SET neon.forward_ddl = false")?;
|
|
||||||
client.simple_query("CREATE USER cloud_admin WITH SUPERUSER")?;
|
client.simple_query("CREATE USER cloud_admin WITH SUPERUSER")?;
|
||||||
client.simple_query("GRANT zenith_admin TO cloud_admin")?;
|
client.simple_query("GRANT zenith_admin TO cloud_admin")?;
|
||||||
drop(client);
|
drop(client);
|
||||||
@@ -437,16 +371,14 @@ impl ComputeNode {
|
|||||||
Ok(client) => client,
|
Ok(client) => client,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Proceed with post-startup configuration. Note, that order of operations is important.
|
||||||
// Disable DDL forwarding because control plane already knows about these roles/databases.
|
// Disable DDL forwarding because control plane already knows about these roles/databases.
|
||||||
client.simple_query("SET neon.forward_ddl = false")?;
|
client.simple_query("SET neon.forward_ddl = false")?;
|
||||||
|
|
||||||
// Proceed with post-startup configuration. Note, that order of operations is important.
|
|
||||||
let spec = &compute_state.pspec.as_ref().expect("spec must be set").spec;
|
let spec = &compute_state.pspec.as_ref().expect("spec must be set").spec;
|
||||||
create_neon_superuser(spec, &mut client)?;
|
|
||||||
handle_roles(spec, &mut client)?;
|
handle_roles(spec, &mut client)?;
|
||||||
handle_databases(spec, &mut client)?;
|
handle_databases(spec, &mut client)?;
|
||||||
handle_role_deletions(spec, self.connstr.as_str(), &mut client)?;
|
handle_role_deletions(spec, self.connstr.as_str(), &mut client)?;
|
||||||
handle_grants(spec, self.connstr.as_str())?;
|
handle_grants(spec, self.connstr.as_str(), &mut client)?;
|
||||||
handle_extensions(spec, &mut client)?;
|
handle_extensions(spec, &mut client)?;
|
||||||
|
|
||||||
// 'Close' connection
|
// 'Close' connection
|
||||||
@@ -458,7 +390,7 @@ impl ComputeNode {
|
|||||||
// We could've wrapped this around `pg_ctl reload`, but right now we don't use
|
// We could've wrapped this around `pg_ctl reload`, but right now we don't use
|
||||||
// `pg_ctl` for start / stop, so this just seems much easier to do as we already
|
// `pg_ctl` for start / stop, so this just seems much easier to do as we already
|
||||||
// have opened connection to Postgres and superuser access.
|
// have opened connection to Postgres and superuser access.
|
||||||
#[instrument(skip_all)]
|
#[instrument(skip(self, client))]
|
||||||
fn pg_reload_conf(&self, client: &mut Client) -> Result<()> {
|
fn pg_reload_conf(&self, client: &mut Client) -> Result<()> {
|
||||||
client.simple_query("SELECT pg_reload_conf()")?;
|
client.simple_query("SELECT pg_reload_conf()")?;
|
||||||
Ok(())
|
Ok(())
|
||||||
@@ -466,13 +398,13 @@ impl ComputeNode {
|
|||||||
|
|
||||||
/// Similar to `apply_config()`, but does a bit different sequence of operations,
|
/// Similar to `apply_config()`, but does a bit different sequence of operations,
|
||||||
/// as it's used to reconfigure a previously started and configured Postgres node.
|
/// as it's used to reconfigure a previously started and configured Postgres node.
|
||||||
#[instrument(skip_all)]
|
#[instrument(skip(self))]
|
||||||
pub fn reconfigure(&self) -> Result<()> {
|
pub fn reconfigure(&self) -> Result<()> {
|
||||||
let spec = self.state.lock().unwrap().pspec.clone().unwrap().spec;
|
let spec = self.state.lock().unwrap().pspec.clone().unwrap().spec;
|
||||||
|
|
||||||
// Write new config
|
// Write new config
|
||||||
let pgdata_path = Path::new(&self.pgdata);
|
let pgdata_path = Path::new(&self.pgdata);
|
||||||
config::write_postgres_conf(&pgdata_path.join("postgresql.conf"), &spec)?;
|
config::write_postgres_conf(&pgdata_path.join("postgresql.conf"), &spec, None)?;
|
||||||
|
|
||||||
let mut client = Client::connect(self.connstr.as_str(), NoTls)?;
|
let mut client = Client::connect(self.connstr.as_str(), NoTls)?;
|
||||||
self.pg_reload_conf(&mut client)?;
|
self.pg_reload_conf(&mut client)?;
|
||||||
@@ -484,7 +416,7 @@ impl ComputeNode {
|
|||||||
handle_roles(&spec, &mut client)?;
|
handle_roles(&spec, &mut client)?;
|
||||||
handle_databases(&spec, &mut client)?;
|
handle_databases(&spec, &mut client)?;
|
||||||
handle_role_deletions(&spec, self.connstr.as_str(), &mut client)?;
|
handle_role_deletions(&spec, self.connstr.as_str(), &mut client)?;
|
||||||
handle_grants(&spec, self.connstr.as_str())?;
|
handle_grants(&spec, self.connstr.as_str(), &mut client)?;
|
||||||
handle_extensions(&spec, &mut client)?;
|
handle_extensions(&spec, &mut client)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -501,8 +433,8 @@ impl ComputeNode {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
#[instrument(skip_all)]
|
#[instrument(skip(self))]
|
||||||
pub fn start_compute(&self) -> Result<std::process::Child> {
|
pub fn start_compute(&self, extension_server_port: u16) -> Result<std::process::Child> {
|
||||||
let compute_state = self.state.lock().unwrap().clone();
|
let compute_state = self.state.lock().unwrap().clone();
|
||||||
let pspec = compute_state.pspec.as_ref().expect("spec must be set");
|
let pspec = compute_state.pspec.as_ref().expect("spec must be set");
|
||||||
info!(
|
info!(
|
||||||
@@ -513,12 +445,12 @@ impl ComputeNode {
|
|||||||
pspec.timeline_id,
|
pspec.timeline_id,
|
||||||
);
|
);
|
||||||
|
|
||||||
self.prepare_pgdata(&compute_state)?;
|
self.prepare_pgdata(&compute_state, extension_server_port)?;
|
||||||
|
|
||||||
let start_time = Utc::now();
|
let start_time = Utc::now();
|
||||||
|
|
||||||
let pg = self.start_postgres(pspec.storage_auth_token.clone())?;
|
let pg = self.start_postgres(pspec.storage_auth_token.clone())?;
|
||||||
|
|
||||||
let config_time = Utc::now();
|
|
||||||
if pspec.spec.mode == ComputeMode::Primary && !pspec.spec.skip_pg_catalog_updates {
|
if pspec.spec.mode == ComputeMode::Primary && !pspec.spec.skip_pg_catalog_updates {
|
||||||
self.apply_config(&compute_state)?;
|
self.apply_config(&compute_state)?;
|
||||||
}
|
}
|
||||||
@@ -526,13 +458,8 @@ impl ComputeNode {
|
|||||||
let startup_end_time = Utc::now();
|
let startup_end_time = Utc::now();
|
||||||
{
|
{
|
||||||
let mut state = self.state.lock().unwrap();
|
let mut state = self.state.lock().unwrap();
|
||||||
state.metrics.start_postgres_ms = config_time
|
|
||||||
.signed_duration_since(start_time)
|
|
||||||
.to_std()
|
|
||||||
.unwrap()
|
|
||||||
.as_millis() as u64;
|
|
||||||
state.metrics.config_ms = startup_end_time
|
state.metrics.config_ms = startup_end_time
|
||||||
.signed_duration_since(config_time)
|
.signed_duration_since(start_time)
|
||||||
.to_std()
|
.to_std()
|
||||||
.unwrap()
|
.unwrap()
|
||||||
.as_millis() as u64;
|
.as_millis() as u64;
|
||||||
|
|||||||
@@ -33,7 +33,11 @@ pub fn line_in_file(path: &Path, line: &str) -> Result<bool> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Create or completely rewrite configuration file specified by `path`
|
/// Create or completely rewrite configuration file specified by `path`
|
||||||
pub fn write_postgres_conf(path: &Path, spec: &ComputeSpec) -> Result<()> {
|
pub fn write_postgres_conf(
|
||||||
|
path: &Path,
|
||||||
|
spec: &ComputeSpec,
|
||||||
|
extension_server_port: Option<u16>,
|
||||||
|
) -> Result<()> {
|
||||||
// File::create() destroys the file content if it exists.
|
// File::create() destroys the file content if it exists.
|
||||||
let mut file = File::create(path)?;
|
let mut file = File::create(path)?;
|
||||||
|
|
||||||
@@ -95,5 +99,9 @@ pub fn write_postgres_conf(path: &Path, spec: &ComputeSpec) -> Result<()> {
|
|||||||
writeln!(file, "# Managed by compute_ctl: end")?;
|
writeln!(file, "# Managed by compute_ctl: end")?;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let Some(port) = extension_server_port {
|
||||||
|
writeln!(file, "neon.extension_server_port={}", port)?;
|
||||||
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ use compute_api::responses::ComputeStatus;
|
|||||||
|
|
||||||
use crate::compute::ComputeNode;
|
use crate::compute::ComputeNode;
|
||||||
|
|
||||||
#[instrument(skip_all)]
|
#[instrument(skip(compute))]
|
||||||
fn configurator_main_loop(compute: &Arc<ComputeNode>) {
|
fn configurator_main_loop(compute: &Arc<ComputeNode>) {
|
||||||
info!("waiting for reconfiguration requests");
|
info!("waiting for reconfiguration requests");
|
||||||
loop {
|
loop {
|
||||||
|
|||||||
182
compute_tools/src/extension_server.rs
Normal file
182
compute_tools/src/extension_server.rs
Normal file
@@ -0,0 +1,182 @@
|
|||||||
|
use anyhow::{self, bail, Result};
|
||||||
|
use remote_storage::*;
|
||||||
|
use serde_json::{self, Value};
|
||||||
|
use std::fs::File;
|
||||||
|
use std::io::{BufWriter, Write};
|
||||||
|
use std::num::{NonZeroU32, NonZeroUsize};
|
||||||
|
use std::path::{Path, PathBuf};
|
||||||
|
use std::str;
|
||||||
|
use tokio::io::AsyncReadExt;
|
||||||
|
use tracing::info;
|
||||||
|
use utils::id::TenantId;
|
||||||
|
|
||||||
|
fn get_pg_config(argument: &str, pgbin: &str) -> String {
|
||||||
|
// gives the result of `pg_config [argument]`
|
||||||
|
// where argument is a flag like `--version` or `--sharedir`
|
||||||
|
let pgconfig = pgbin.replace("postgres", "pg_config");
|
||||||
|
let config_output = std::process::Command::new(pgconfig)
|
||||||
|
.arg(argument)
|
||||||
|
.output()
|
||||||
|
.expect("pg_config error");
|
||||||
|
std::str::from_utf8(&config_output.stdout)
|
||||||
|
.expect("pg_config error")
|
||||||
|
.trim()
|
||||||
|
.to_string()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_pg_version(pgbin: &str) -> String {
|
||||||
|
// pg_config --version returns a (platform specific) human readable string
|
||||||
|
// such as "PostgreSQL 15.4". We parse this to v14/v15
|
||||||
|
let human_version = get_pg_config("--version", pgbin);
|
||||||
|
if human_version.contains("v15") {
|
||||||
|
return "v15".to_string();
|
||||||
|
}
|
||||||
|
"v14".to_string()
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn download_helper(
|
||||||
|
remote_storage: &GenericRemoteStorage,
|
||||||
|
remote_from_path: &RemotePath,
|
||||||
|
download_location: &Path,
|
||||||
|
) -> anyhow::Result<()> {
|
||||||
|
// downloads file at remote_from_path to download_location/[file_name]
|
||||||
|
let local_path = download_location.join(remote_from_path.object_name().expect("bad object"));
|
||||||
|
info!(
|
||||||
|
"Downloading {:?} to location {:?}",
|
||||||
|
&remote_from_path, &local_path
|
||||||
|
);
|
||||||
|
let mut download = remote_storage.download(remote_from_path).await?;
|
||||||
|
let mut write_data_buffer = Vec::new();
|
||||||
|
download
|
||||||
|
.download_stream
|
||||||
|
.read_to_end(&mut write_data_buffer)
|
||||||
|
.await?;
|
||||||
|
let mut output_file = BufWriter::new(File::create(local_path)?);
|
||||||
|
output_file.write_all(&write_data_buffer)?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
// download extension control files
|
||||||
|
//
|
||||||
|
// return list of all extension files to use it in the future searches
|
||||||
|
//
|
||||||
|
// if tenant_id is provided - search in a private per-tenant extension path,
|
||||||
|
// otherwise - in public extension path
|
||||||
|
//
|
||||||
|
pub async fn get_availiable_extensions(
|
||||||
|
remote_storage: &GenericRemoteStorage,
|
||||||
|
pg_version: &str,
|
||||||
|
pgbin: &str,
|
||||||
|
tenant_id: Option<TenantId>,
|
||||||
|
) -> anyhow::Result<Vec<RemotePath>> {
|
||||||
|
let local_sharedir = Path::new(&get_pg_config("--sharedir", pgbin)).join("extension");
|
||||||
|
|
||||||
|
let remote_sharedir = match tenant_id {
|
||||||
|
None => RemotePath::new(&Path::new(&pg_version).join("share/postgresql/extension"))?,
|
||||||
|
Some(tenant_id) => RemotePath::new(
|
||||||
|
&Path::new(&pg_version)
|
||||||
|
.join(&tenant_id.to_string())
|
||||||
|
.join("share/postgresql/extension"),
|
||||||
|
)?,
|
||||||
|
};
|
||||||
|
|
||||||
|
let from_paths = remote_storage.list_files(Some(&remote_sharedir)).await?;
|
||||||
|
|
||||||
|
// download all found control files
|
||||||
|
for remote_from_path in &from_paths {
|
||||||
|
if remote_from_path.extension() == Some("control") {
|
||||||
|
download_helper(remote_storage, &remote_from_path, &local_sharedir).await?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(from_paths)
|
||||||
|
}
|
||||||
|
|
||||||
|
// download all sql files for a given extension name
|
||||||
|
//
|
||||||
|
pub async fn download_extension_sql_files(
|
||||||
|
ext_name: &str,
|
||||||
|
availiable_extensions: Vec<RemotePath>,
|
||||||
|
remote_storage: &GenericRemoteStorage,
|
||||||
|
pgbin: &str,
|
||||||
|
) -> Result<()> {
|
||||||
|
let local_sharedir = Path::new(&get_pg_config("--sharedir", pgbin)).join("extension");
|
||||||
|
|
||||||
|
// check if extension files exist
|
||||||
|
let files_to_download: Vec<&RemotePath> = availiable_extensions
|
||||||
|
.iter()
|
||||||
|
.filter(|ext| {
|
||||||
|
ext.extension() == Some("sql") && ext.object_name().unwrap().starts_with(ext_name)
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
if files_to_download.is_empty() {
|
||||||
|
bail!("Files for extension {ext_name} are not found in the extension store");
|
||||||
|
}
|
||||||
|
|
||||||
|
for remote_from_path in files_to_download {
|
||||||
|
download_helper(remote_storage, &remote_from_path, &local_sharedir).await?;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
// download shared library file
|
||||||
|
pub async fn download_library_file(
|
||||||
|
lib_name: &str,
|
||||||
|
availiable_libraries: Vec<RemotePath>,
|
||||||
|
remote_storage: &GenericRemoteStorage,
|
||||||
|
pgbin: &str,
|
||||||
|
) -> Result<()> {
|
||||||
|
let local_libdir: PathBuf = Path::new(&get_pg_config("--libdir", pgbin)).into();
|
||||||
|
|
||||||
|
// check if the library file exists
|
||||||
|
let lib = availiable_libraries
|
||||||
|
.iter()
|
||||||
|
.find(|lib: &&RemotePath| lib.object_name().unwrap() == lib_name);
|
||||||
|
|
||||||
|
match lib {
|
||||||
|
None => bail!("Shared library file {lib_name} is not found in the extension store"),
|
||||||
|
Some(lib) => {
|
||||||
|
download_helper(remote_storage, &lib, &local_libdir).await?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn init_remote_storage(remote_ext_config: &str) -> anyhow::Result<GenericRemoteStorage> {
|
||||||
|
let remote_ext_config: serde_json::Value = serde_json::from_str(remote_ext_config)?;
|
||||||
|
let remote_ext_bucket = match &remote_ext_config["bucket"] {
|
||||||
|
Value::String(x) => x,
|
||||||
|
_ => bail!("remote_ext_config missing bucket"),
|
||||||
|
};
|
||||||
|
let remote_ext_region = match &remote_ext_config["region"] {
|
||||||
|
Value::String(x) => x,
|
||||||
|
_ => bail!("remote_ext_config missing region"),
|
||||||
|
};
|
||||||
|
let remote_ext_endpoint = match &remote_ext_config["endpoint"] {
|
||||||
|
Value::String(x) => Some(x.clone()),
|
||||||
|
_ => None,
|
||||||
|
};
|
||||||
|
let remote_ext_prefix = match &remote_ext_config["prefix"] {
|
||||||
|
Value::String(x) => Some(x.clone()),
|
||||||
|
_ => None,
|
||||||
|
};
|
||||||
|
|
||||||
|
// load will not be large, so default parameters are fine
|
||||||
|
let config = S3Config {
|
||||||
|
bucket_name: remote_ext_bucket.to_string(),
|
||||||
|
bucket_region: remote_ext_region.to_string(),
|
||||||
|
prefix_in_bucket: remote_ext_prefix,
|
||||||
|
endpoint: remote_ext_endpoint,
|
||||||
|
concurrency_limit: NonZeroUsize::new(100).expect("100 != 0"),
|
||||||
|
max_keys_per_list_response: None,
|
||||||
|
};
|
||||||
|
let config = RemoteStorageConfig {
|
||||||
|
max_concurrent_syncs: NonZeroUsize::new(100).expect("100 != 0"),
|
||||||
|
max_sync_errors: NonZeroU32::new(100).expect("100 != 0"),
|
||||||
|
storage: RemoteStorageKind::AwsS3(config),
|
||||||
|
};
|
||||||
|
GenericRemoteStorage::from_config(&config)
|
||||||
|
}
|
||||||
@@ -16,6 +16,8 @@ use tokio::task;
|
|||||||
use tracing::{error, info};
|
use tracing::{error, info};
|
||||||
use tracing_utils::http::OtelName;
|
use tracing_utils::http::OtelName;
|
||||||
|
|
||||||
|
use crate::extension_server::{download_extension_sql_files, download_library_file};
|
||||||
|
|
||||||
fn status_response_from_state(state: &ComputeState) -> ComputeStatusResponse {
|
fn status_response_from_state(state: &ComputeState) -> ComputeStatusResponse {
|
||||||
ComputeStatusResponse {
|
ComputeStatusResponse {
|
||||||
start_time: state.start_time,
|
start_time: state.start_time,
|
||||||
@@ -121,8 +123,68 @@ async fn routes(req: Request<Body>, compute: &Arc<ComputeNode>) -> Response<Body
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// download extension files from S3 on demand
|
||||||
|
(&Method::POST, route) if route.starts_with("/extension_server/") => {
|
||||||
|
info!("serving {:?} POST request", route);
|
||||||
|
|
||||||
|
let is_library = false;
|
||||||
|
|
||||||
|
let filename = route.split('/').last().unwrap();
|
||||||
|
|
||||||
|
info!(
|
||||||
|
"serving /extension_server POST request, filename: {:?}",
|
||||||
|
filename
|
||||||
|
);
|
||||||
|
|
||||||
|
if compute.ext_remote_storage.is_none() {
|
||||||
|
error!("Remote extension storage is not set up");
|
||||||
|
let mut resp = Response::new(Body::from("Remote extension storage is not set up"));
|
||||||
|
*resp.status_mut() = StatusCode::INTERNAL_SERVER_ERROR;
|
||||||
|
return resp;
|
||||||
|
}
|
||||||
|
let ext_storage = &compute.ext_remote_storage.unwrap();
|
||||||
|
|
||||||
|
if !is_library {
|
||||||
|
match download_extension_sql_files(
|
||||||
|
filename,
|
||||||
|
&compute.availiable_extensions,
|
||||||
|
&ext_storage,
|
||||||
|
&compute.pgbin,
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
{
|
||||||
|
Ok(_) => Response::new(Body::from("OK")),
|
||||||
|
Err(e) => {
|
||||||
|
error!("extension download failed: {}", e);
|
||||||
|
let mut resp = Response::new(Body::from(e.to_string()));
|
||||||
|
*resp.status_mut() = StatusCode::INTERNAL_SERVER_ERROR;
|
||||||
|
resp
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
match download_library_file(
|
||||||
|
filename,
|
||||||
|
&compute.availiable_libraries,
|
||||||
|
&ext_storage,
|
||||||
|
&compute.pgbin,
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
{
|
||||||
|
Ok(_) => Response::new(Body::from("OK")),
|
||||||
|
Err(e) => {
|
||||||
|
error!("library download failed: {}", e);
|
||||||
|
let mut resp = Response::new(Body::from(e.to_string()));
|
||||||
|
*resp.status_mut() = StatusCode::INTERNAL_SERVER_ERROR;
|
||||||
|
resp
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Return the `404 Not Found` for any other routes.
|
// Return the `404 Not Found` for any other routes.
|
||||||
_ => {
|
method => {
|
||||||
|
info!("404 Not Found for {:?}", method);
|
||||||
|
|
||||||
let mut not_found = Response::new(Body::from("404 Not Found"));
|
let mut not_found = Response::new(Body::from("404 Not Found"));
|
||||||
*not_found.status_mut() = StatusCode::NOT_FOUND;
|
*not_found.status_mut() = StatusCode::NOT_FOUND;
|
||||||
not_found
|
not_found
|
||||||
|
|||||||
@@ -139,6 +139,34 @@ paths:
|
|||||||
application/json:
|
application/json:
|
||||||
schema:
|
schema:
|
||||||
$ref: "#/components/schemas/GenericError"
|
$ref: "#/components/schemas/GenericError"
|
||||||
|
/extension_server:
|
||||||
|
post:
|
||||||
|
tags:
|
||||||
|
- Extension
|
||||||
|
summary: Download extension from S3 to local folder.
|
||||||
|
description: ""
|
||||||
|
operationId: downloadExtension
|
||||||
|
responses:
|
||||||
|
200:
|
||||||
|
description: Extension downloaded
|
||||||
|
content:
|
||||||
|
text/plain:
|
||||||
|
schema:
|
||||||
|
type: string
|
||||||
|
description: Error text or 'OK' if download succeeded.
|
||||||
|
example: "OK"
|
||||||
|
400:
|
||||||
|
description: Request is invalid.
|
||||||
|
content:
|
||||||
|
application/json:
|
||||||
|
schema:
|
||||||
|
$ref: "#/components/schemas/GenericError"
|
||||||
|
500:
|
||||||
|
description: Extension download request failed.
|
||||||
|
content:
|
||||||
|
application/json:
|
||||||
|
schema:
|
||||||
|
$ref: "#/components/schemas/GenericError"
|
||||||
|
|
||||||
components:
|
components:
|
||||||
securitySchemes:
|
securitySchemes:
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ pub mod http;
|
|||||||
#[macro_use]
|
#[macro_use]
|
||||||
pub mod logger;
|
pub mod logger;
|
||||||
pub mod compute;
|
pub mod compute;
|
||||||
|
pub mod extension_server;
|
||||||
pub mod monitor;
|
pub mod monitor;
|
||||||
pub mod params;
|
pub mod params;
|
||||||
pub mod pg_helpers;
|
pub mod pg_helpers;
|
||||||
|
|||||||
@@ -18,7 +18,6 @@ pub fn init_tracing_and_logging(default_log_level: &str) -> anyhow::Result<()> {
|
|||||||
.unwrap_or_else(|_| tracing_subscriber::EnvFilter::new(default_log_level));
|
.unwrap_or_else(|_| tracing_subscriber::EnvFilter::new(default_log_level));
|
||||||
|
|
||||||
let fmt_layer = tracing_subscriber::fmt::layer()
|
let fmt_layer = tracing_subscriber::fmt::layer()
|
||||||
.with_ansi(false)
|
|
||||||
.with_target(false)
|
.with_target(false)
|
||||||
.with_writer(std::io::stderr);
|
.with_writer(std::io::stderr);
|
||||||
|
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ use compute_api::spec::{Database, GenericOption, GenericOptions, PgIdent, Role};
|
|||||||
const POSTGRES_WAIT_TIMEOUT: Duration = Duration::from_millis(60 * 1000); // milliseconds
|
const POSTGRES_WAIT_TIMEOUT: Duration = Duration::from_millis(60 * 1000); // milliseconds
|
||||||
|
|
||||||
/// Escape a string for including it in a SQL literal
|
/// Escape a string for including it in a SQL literal
|
||||||
pub fn escape_literal(s: &str) -> String {
|
fn escape_literal(s: &str) -> String {
|
||||||
s.replace('\'', "''").replace('\\', "\\\\")
|
s.replace('\'', "''").replace('\\', "\\\\")
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -215,7 +215,7 @@ pub fn get_existing_dbs(client: &mut Client) -> Result<Vec<Database>> {
|
|||||||
/// Wait for Postgres to become ready to accept connections. It's ready to
|
/// Wait for Postgres to become ready to accept connections. It's ready to
|
||||||
/// accept connections when the state-field in `pgdata/postmaster.pid` says
|
/// accept connections when the state-field in `pgdata/postmaster.pid` says
|
||||||
/// 'ready'.
|
/// 'ready'.
|
||||||
#[instrument(skip_all, fields(pgdata = %pgdata.display()))]
|
#[instrument(skip(pg))]
|
||||||
pub fn wait_for_postgres(pg: &mut Child, pgdata: &Path) -> Result<()> {
|
pub fn wait_for_postgres(pg: &mut Child, pgdata: &Path) -> Result<()> {
|
||||||
let pid_path = pgdata.join("postmaster.pid");
|
let pid_path = pgdata.join("postmaster.pid");
|
||||||
|
|
||||||
|
|||||||
@@ -124,7 +124,7 @@ pub fn get_spec_from_control_plane(
|
|||||||
pub fn handle_configuration(spec: &ComputeSpec, pgdata_path: &Path) -> Result<()> {
|
pub fn handle_configuration(spec: &ComputeSpec, pgdata_path: &Path) -> Result<()> {
|
||||||
// File `postgresql.conf` is no longer included into `basebackup`, so just
|
// File `postgresql.conf` is no longer included into `basebackup`, so just
|
||||||
// always write all config into it creating new file.
|
// always write all config into it creating new file.
|
||||||
config::write_postgres_conf(&pgdata_path.join("postgresql.conf"), spec)?;
|
config::write_postgres_conf(&pgdata_path.join("postgresql.conf"), spec, None)?;
|
||||||
|
|
||||||
update_pg_hba(pgdata_path)?;
|
update_pg_hba(pgdata_path)?;
|
||||||
|
|
||||||
@@ -269,13 +269,17 @@ pub fn handle_roles(spec: &ComputeSpec, client: &mut Client) -> Result<()> {
|
|||||||
xact.execute(query.as_str(), &[])?;
|
xact.execute(query.as_str(), &[])?;
|
||||||
}
|
}
|
||||||
RoleAction::Create => {
|
RoleAction::Create => {
|
||||||
let mut query: String = format!(
|
let mut query: String = format!("CREATE ROLE {} ", name.pg_quote());
|
||||||
"CREATE ROLE {} CREATEROLE CREATEDB IN ROLE neon_superuser",
|
|
||||||
name.pg_quote()
|
|
||||||
);
|
|
||||||
info!("role create query: '{}'", &query);
|
info!("role create query: '{}'", &query);
|
||||||
query.push_str(&role.to_pg_options());
|
query.push_str(&role.to_pg_options());
|
||||||
xact.execute(query.as_str(), &[])?;
|
xact.execute(query.as_str(), &[])?;
|
||||||
|
|
||||||
|
let grant_query = format!(
|
||||||
|
"GRANT pg_read_all_data, pg_write_all_data TO {}",
|
||||||
|
name.pg_quote()
|
||||||
|
);
|
||||||
|
xact.execute(grant_query.as_str(), &[])?;
|
||||||
|
info!("role grant query: '{}'", &grant_query);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -472,11 +476,6 @@ pub fn handle_databases(spec: &ComputeSpec, client: &mut Client) -> Result<()> {
|
|||||||
query.push_str(&db.to_pg_options());
|
query.push_str(&db.to_pg_options());
|
||||||
let _guard = info_span!("executing", query).entered();
|
let _guard = info_span!("executing", query).entered();
|
||||||
client.execute(query.as_str(), &[])?;
|
client.execute(query.as_str(), &[])?;
|
||||||
let grant_query: String = format!(
|
|
||||||
"GRANT ALL PRIVILEGES ON DATABASE {} TO neon_superuser",
|
|
||||||
name.pg_quote()
|
|
||||||
);
|
|
||||||
client.execute(grant_query.as_str(), &[])?;
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -496,9 +495,35 @@ pub fn handle_databases(spec: &ComputeSpec, client: &mut Client) -> Result<()> {
|
|||||||
/// Grant CREATE ON DATABASE to the database owner and do some other alters and grants
|
/// Grant CREATE ON DATABASE to the database owner and do some other alters and grants
|
||||||
/// to allow users creating trusted extensions and re-creating `public` schema, for example.
|
/// to allow users creating trusted extensions and re-creating `public` schema, for example.
|
||||||
#[instrument(skip_all)]
|
#[instrument(skip_all)]
|
||||||
pub fn handle_grants(spec: &ComputeSpec, connstr: &str) -> Result<()> {
|
pub fn handle_grants(spec: &ComputeSpec, connstr: &str, client: &mut Client) -> Result<()> {
|
||||||
info!("cluster spec grants:");
|
info!("cluster spec grants:");
|
||||||
|
|
||||||
|
// We now have a separate `web_access` role to connect to the database
|
||||||
|
// via the web interface and proxy link auth. And also we grant a
|
||||||
|
// read / write all data privilege to every role. So also grant
|
||||||
|
// create to everyone.
|
||||||
|
// XXX: later we should stop messing with Postgres ACL in such horrible
|
||||||
|
// ways.
|
||||||
|
let roles = spec
|
||||||
|
.cluster
|
||||||
|
.roles
|
||||||
|
.iter()
|
||||||
|
.map(|r| r.name.pg_quote())
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
|
for db in &spec.cluster.databases {
|
||||||
|
let dbname = &db.name;
|
||||||
|
|
||||||
|
let query: String = format!(
|
||||||
|
"GRANT CREATE ON DATABASE {} TO {}",
|
||||||
|
dbname.pg_quote(),
|
||||||
|
roles.join(", ")
|
||||||
|
);
|
||||||
|
info!("grant query {}", &query);
|
||||||
|
|
||||||
|
client.execute(query.as_str(), &[])?;
|
||||||
|
}
|
||||||
|
|
||||||
// Do some per-database access adjustments. We'd better do this at db creation time,
|
// Do some per-database access adjustments. We'd better do this at db creation time,
|
||||||
// but CREATE DATABASE isn't transactional. So we cannot create db + do some grants
|
// but CREATE DATABASE isn't transactional. So we cannot create db + do some grants
|
||||||
// atomically.
|
// atomically.
|
||||||
|
|||||||
@@ -32,3 +32,4 @@ utils.workspace = true
|
|||||||
|
|
||||||
compute_api.workspace = true
|
compute_api.workspace = true
|
||||||
workspace_hack.workspace = true
|
workspace_hack.workspace = true
|
||||||
|
tracing.workspace = true
|
||||||
|
|||||||
@@ -180,11 +180,6 @@ pub fn stop_process(immediate: bool, process_name: &str, pid_file: &Path) -> any
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Wait until process is gone
|
// Wait until process is gone
|
||||||
wait_until_stopped(process_name, pid)?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn wait_until_stopped(process_name: &str, pid: Pid) -> anyhow::Result<()> {
|
|
||||||
for retries in 0..RETRIES {
|
for retries in 0..RETRIES {
|
||||||
match process_has_stopped(pid) {
|
match process_has_stopped(pid) {
|
||||||
Ok(true) => {
|
Ok(true) => {
|
||||||
|
|||||||
@@ -308,8 +308,7 @@ fn handle_init(init_match: &ArgMatches) -> anyhow::Result<LocalEnv> {
|
|||||||
|
|
||||||
let mut env =
|
let mut env =
|
||||||
LocalEnv::parse_config(&toml_file).context("Failed to create neon configuration")?;
|
LocalEnv::parse_config(&toml_file).context("Failed to create neon configuration")?;
|
||||||
let force = init_match.get_flag("force");
|
env.init(pg_version)
|
||||||
env.init(pg_version, force)
|
|
||||||
.context("Failed to initialize neon repository")?;
|
.context("Failed to initialize neon repository")?;
|
||||||
|
|
||||||
// Initialize pageserver, create initial tenant and timeline.
|
// Initialize pageserver, create initial tenant and timeline.
|
||||||
@@ -658,6 +657,8 @@ fn handle_endpoint(ep_match: &ArgMatches, env: &local_env::LocalEnv) -> Result<(
|
|||||||
.get_one::<String>("endpoint_id")
|
.get_one::<String>("endpoint_id")
|
||||||
.ok_or_else(|| anyhow!("No endpoint ID was provided to start"))?;
|
.ok_or_else(|| anyhow!("No endpoint ID was provided to start"))?;
|
||||||
|
|
||||||
|
let remote_ext_config = sub_args.get_one::<String>("remote-ext-config");
|
||||||
|
|
||||||
// If --safekeepers argument is given, use only the listed safekeeper nodes.
|
// If --safekeepers argument is given, use only the listed safekeeper nodes.
|
||||||
let safekeepers =
|
let safekeepers =
|
||||||
if let Some(safekeepers_str) = sub_args.get_one::<String>("safekeepers") {
|
if let Some(safekeepers_str) = sub_args.get_one::<String>("safekeepers") {
|
||||||
@@ -699,7 +700,7 @@ fn handle_endpoint(ep_match: &ArgMatches, env: &local_env::LocalEnv) -> Result<(
|
|||||||
_ => {}
|
_ => {}
|
||||||
}
|
}
|
||||||
println!("Starting existing endpoint {endpoint_id}...");
|
println!("Starting existing endpoint {endpoint_id}...");
|
||||||
endpoint.start(&auth_token, safekeepers)?;
|
endpoint.start(&auth_token, safekeepers, remote_ext_config)?;
|
||||||
} else {
|
} else {
|
||||||
let branch_name = sub_args
|
let branch_name = sub_args
|
||||||
.get_one::<String>("branch-name")
|
.get_one::<String>("branch-name")
|
||||||
@@ -743,7 +744,7 @@ fn handle_endpoint(ep_match: &ArgMatches, env: &local_env::LocalEnv) -> Result<(
|
|||||||
pg_version,
|
pg_version,
|
||||||
mode,
|
mode,
|
||||||
)?;
|
)?;
|
||||||
ep.start(&auth_token, safekeepers)?;
|
ep.start(&auth_token, safekeepers, remote_ext_config)?;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
"stop" => {
|
"stop" => {
|
||||||
@@ -1003,6 +1004,12 @@ fn cli() -> Command {
|
|||||||
.help("Additional pageserver's configuration options or overrides, refer to pageserver's 'config-override' CLI parameter docs for more")
|
.help("Additional pageserver's configuration options or overrides, refer to pageserver's 'config-override' CLI parameter docs for more")
|
||||||
.required(false);
|
.required(false);
|
||||||
|
|
||||||
|
let remote_ext_config_args = Arg::new("remote-ext-config")
|
||||||
|
.long("remote-ext-config")
|
||||||
|
.num_args(1)
|
||||||
|
.help("Configure the S3 bucket that we search for extensions in.")
|
||||||
|
.required(false);
|
||||||
|
|
||||||
let lsn_arg = Arg::new("lsn")
|
let lsn_arg = Arg::new("lsn")
|
||||||
.long("lsn")
|
.long("lsn")
|
||||||
.help("Specify Lsn on the timeline to start from. By default, end of the timeline would be used.")
|
.help("Specify Lsn on the timeline to start from. By default, end of the timeline would be used.")
|
||||||
@@ -1014,13 +1021,6 @@ fn cli() -> Command {
|
|||||||
.help("If set, the node will be a hot replica on the specified timeline")
|
.help("If set, the node will be a hot replica on the specified timeline")
|
||||||
.required(false);
|
.required(false);
|
||||||
|
|
||||||
let force_arg = Arg::new("force")
|
|
||||||
.value_parser(value_parser!(bool))
|
|
||||||
.long("force")
|
|
||||||
.action(ArgAction::SetTrue)
|
|
||||||
.help("Force initialization even if the repository is not empty")
|
|
||||||
.required(false);
|
|
||||||
|
|
||||||
Command::new("Neon CLI")
|
Command::new("Neon CLI")
|
||||||
.arg_required_else_help(true)
|
.arg_required_else_help(true)
|
||||||
.version(GIT_VERSION)
|
.version(GIT_VERSION)
|
||||||
@@ -1036,7 +1036,6 @@ fn cli() -> Command {
|
|||||||
.value_name("config"),
|
.value_name("config"),
|
||||||
)
|
)
|
||||||
.arg(pg_version_arg.clone())
|
.arg(pg_version_arg.clone())
|
||||||
.arg(force_arg)
|
|
||||||
)
|
)
|
||||||
.subcommand(
|
.subcommand(
|
||||||
Command::new("timeline")
|
Command::new("timeline")
|
||||||
@@ -1161,6 +1160,7 @@ fn cli() -> Command {
|
|||||||
.arg(pg_version_arg)
|
.arg(pg_version_arg)
|
||||||
.arg(hot_standby_arg)
|
.arg(hot_standby_arg)
|
||||||
.arg(safekeepers_arg)
|
.arg(safekeepers_arg)
|
||||||
|
.arg(remote_ext_config_args)
|
||||||
)
|
)
|
||||||
.subcommand(
|
.subcommand(
|
||||||
Command::new("stop")
|
Command::new("stop")
|
||||||
|
|||||||
@@ -405,20 +405,15 @@ impl Endpoint {
|
|||||||
String::from_utf8_lossy(&pg_ctl.stderr),
|
String::from_utf8_lossy(&pg_ctl.stderr),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Also wait for the compute_ctl process to die. It might have some cleanup
|
|
||||||
// work to do after postgres stops, like syncing safekeepers, etc.
|
|
||||||
//
|
|
||||||
// TODO use background_process::stop_process instead
|
|
||||||
let pidfile_path = self.endpoint_path().join("compute_ctl.pid");
|
|
||||||
let pid: u32 = std::fs::read_to_string(pidfile_path)?.parse()?;
|
|
||||||
let pid = nix::unistd::Pid::from_raw(pid as i32);
|
|
||||||
crate::background_process::wait_until_stopped("compute_ctl", pid)?;
|
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn start(&self, auth_token: &Option<String>, safekeepers: Vec<NodeId>) -> Result<()> {
|
pub fn start(
|
||||||
|
&self,
|
||||||
|
auth_token: &Option<String>,
|
||||||
|
safekeepers: Vec<NodeId>,
|
||||||
|
remote_ext_config: Option<&String>,
|
||||||
|
) -> Result<()> {
|
||||||
if self.status() == "running" {
|
if self.status() == "running" {
|
||||||
anyhow::bail!("The endpoint is already running");
|
anyhow::bail!("The endpoint is already running");
|
||||||
}
|
}
|
||||||
@@ -517,13 +512,10 @@ impl Endpoint {
|
|||||||
.stdin(std::process::Stdio::null())
|
.stdin(std::process::Stdio::null())
|
||||||
.stderr(logfile.try_clone()?)
|
.stderr(logfile.try_clone()?)
|
||||||
.stdout(logfile);
|
.stdout(logfile);
|
||||||
let child = cmd.spawn()?;
|
if let Some(remote_ext_config) = remote_ext_config {
|
||||||
|
cmd.args(["--remote-ext-config", remote_ext_config]);
|
||||||
// Write down the pid so we can wait for it when we want to stop
|
}
|
||||||
// TODO use background_process::start_process instead
|
let _child = cmd.spawn()?;
|
||||||
let pid = child.id();
|
|
||||||
let pidfile_path = self.endpoint_path().join("compute_ctl.pid");
|
|
||||||
std::fs::write(pidfile_path, pid.to_string())?;
|
|
||||||
|
|
||||||
// Wait for it to start
|
// Wait for it to start
|
||||||
let mut attempt = 0;
|
let mut attempt = 0;
|
||||||
|
|||||||
@@ -364,7 +364,7 @@ impl LocalEnv {
|
|||||||
//
|
//
|
||||||
// Initialize a new Neon repository
|
// Initialize a new Neon repository
|
||||||
//
|
//
|
||||||
pub fn init(&mut self, pg_version: u32, force: bool) -> anyhow::Result<()> {
|
pub fn init(&mut self, pg_version: u32) -> anyhow::Result<()> {
|
||||||
// check if config already exists
|
// check if config already exists
|
||||||
let base_path = &self.base_data_dir;
|
let base_path = &self.base_data_dir;
|
||||||
ensure!(
|
ensure!(
|
||||||
@@ -372,29 +372,11 @@ impl LocalEnv {
|
|||||||
"repository base path is missing"
|
"repository base path is missing"
|
||||||
);
|
);
|
||||||
|
|
||||||
if base_path.exists() {
|
ensure!(
|
||||||
if force {
|
!base_path.exists(),
|
||||||
println!("removing all contents of '{}'", base_path.display());
|
"directory '{}' already exists. Perhaps already initialized?",
|
||||||
// instead of directly calling `remove_dir_all`, we keep the original dir but removing
|
base_path.display()
|
||||||
// all contents inside. This helps if the developer symbol links another directory (i.e.,
|
);
|
||||||
// S3 local SSD) to the `.neon` base directory.
|
|
||||||
for entry in std::fs::read_dir(base_path)? {
|
|
||||||
let entry = entry?;
|
|
||||||
let path = entry.path();
|
|
||||||
if path.is_dir() {
|
|
||||||
fs::remove_dir_all(&path)?;
|
|
||||||
} else {
|
|
||||||
fs::remove_file(&path)?;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
bail!(
|
|
||||||
"directory '{}' already exists. Perhaps already initialized? (Hint: use --force to remove all contents)",
|
|
||||||
base_path.display()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if !self.pg_bin_dir(pg_version)?.join("postgres").exists() {
|
if !self.pg_bin_dir(pg_version)?.join("postgres").exists() {
|
||||||
bail!(
|
bail!(
|
||||||
"Can't find postgres binary at {}",
|
"Can't find postgres binary at {}",
|
||||||
@@ -410,9 +392,7 @@ impl LocalEnv {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if !base_path.exists() {
|
fs::create_dir(base_path)?;
|
||||||
fs::create_dir(base_path)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Generate keypair for JWT.
|
// Generate keypair for JWT.
|
||||||
//
|
//
|
||||||
|
|||||||
301
docs/rfcs/024-extension-loading.md
Normal file
301
docs/rfcs/024-extension-loading.md
Normal file
@@ -0,0 +1,301 @@
|
|||||||
|
# Supporting custom user Extensions
|
||||||
|
|
||||||
|
Created 2023-05-03
|
||||||
|
|
||||||
|
## Motivation
|
||||||
|
|
||||||
|
There are many extensions in the PostgreSQL ecosystem, and not all extensions
|
||||||
|
are of a quality that we can confidently support them. Additionally, our
|
||||||
|
current extension inclusion mechanism has several problems because we build all
|
||||||
|
extensions into the primary Compute image: We build the extensions every time
|
||||||
|
we build the compute image regardless of whether we actually need to rebuild
|
||||||
|
the image, and the inclusion of these extensions in the image adds a hard
|
||||||
|
dependency on all supported extensions - thus increasing the image size, and
|
||||||
|
with it the time it takes to download that image - increasing first start
|
||||||
|
latency.
|
||||||
|
|
||||||
|
This RFC proposes a dynamic loading mechanism that solves most of these
|
||||||
|
problems.
|
||||||
|
|
||||||
|
## Summary
|
||||||
|
|
||||||
|
`compute_ctl` is made responsible for loading extensions on-demand into
|
||||||
|
the container's file system for dynamically loaded extensions, and will also
|
||||||
|
make sure that the extensions in `shared_preload_libraries` are downloaded
|
||||||
|
before the compute node starts.
|
||||||
|
|
||||||
|
## Components
|
||||||
|
|
||||||
|
compute_ctl, PostgreSQL, neon (extension), Compute Host Node, Extension Store
|
||||||
|
|
||||||
|
## Requirements
|
||||||
|
|
||||||
|
Compute nodes with no extra extensions should not be negatively impacted by
|
||||||
|
the existence of support for many extensions.
|
||||||
|
|
||||||
|
Installing an extension into PostgreSQL should be easy.
|
||||||
|
|
||||||
|
Non-preloaded extensions shouldn't impact startup latency.
|
||||||
|
|
||||||
|
Uninstalled extensions shouldn't impact query latency.
|
||||||
|
|
||||||
|
A small latency penalty for dynamically loaded extensions is acceptable in
|
||||||
|
the first seconds of compute startup, but not in steady-state operations.
|
||||||
|
|
||||||
|
## Proposed implementation
|
||||||
|
|
||||||
|
### On-demand, JIT-loading of extensions
|
||||||
|
|
||||||
|
TLDR; we download extensions as soon as we need them, or when we have spare
|
||||||
|
time.
|
||||||
|
|
||||||
|
That means, we first download the extensions required to start the PostMaster
|
||||||
|
(`shared_preload_libraries` and their dependencies), then the libraries required
|
||||||
|
before a backend can start processing user input (`preload_libraries` and
|
||||||
|
dependencies), and then (with network limits applied) the remainder of the
|
||||||
|
configured extensions, with prioritization for installed extensions.
|
||||||
|
|
||||||
|
If PostgreSQL tries to load a library that is not yet fully on disk, it will
|
||||||
|
ask `compute_ctl` first if the extension has been downloaded yet, and will wait
|
||||||
|
for `compute_ctl` to finish downloading that extension. `compute_ctl` will
|
||||||
|
prioritize downloading that extension over other extensions that were not yet
|
||||||
|
requested.
|
||||||
|
|
||||||
|
#### Workflow
|
||||||
|
|
||||||
|
```mermaid
|
||||||
|
sequenceDiagram
|
||||||
|
autonumber
|
||||||
|
participant EX as External (control plane, ...)
|
||||||
|
participant CTL as compute_ctl
|
||||||
|
participant ST as extension store
|
||||||
|
actor PG as PostgreSQL
|
||||||
|
|
||||||
|
EX ->>+ CTL: Start compute with config X
|
||||||
|
|
||||||
|
note over CTL: The configuration contains a list of all <br/>extensions available to that compute node, etc.
|
||||||
|
|
||||||
|
par Optionally parallel or concurrent
|
||||||
|
loop Available extensions
|
||||||
|
CTL ->>+ ST: Download control file of extension
|
||||||
|
activate CTL
|
||||||
|
ST ->>- CTL: Finish downloading control file
|
||||||
|
CTL ->>- CTL: Put control file in extensions directory
|
||||||
|
end
|
||||||
|
|
||||||
|
loop For each extension in shared_preload_libraries
|
||||||
|
CTL ->>+ ST: Download extension's data
|
||||||
|
activate CTL
|
||||||
|
ST ->>- CTL: Finish downloading
|
||||||
|
CTL ->>- CTL: Put extension's files in the right place
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
CTL ->>+ PG: Start PostgreSQL
|
||||||
|
|
||||||
|
note over CTL: PostgreSQL can now start accepting <br/>connections. However, users may still need to wait <br/>for preload_libraries extensions to get downloaded.
|
||||||
|
|
||||||
|
par Load preload_libraries
|
||||||
|
loop For each extension in preload_libraries
|
||||||
|
CTL ->>+ ST: Download extension's data
|
||||||
|
activate CTL
|
||||||
|
ST ->>- CTL: Finish downloading
|
||||||
|
CTL ->>- CTL: Put extension's files in the right place
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
note over CTL: After this, connections don't have any hard <br/>waits for extension files left, except for those <br/>connections that override preload_libraries <br/>in their startup packet
|
||||||
|
|
||||||
|
par PG's internal_load_library(library)
|
||||||
|
alt Library is not yet loaded
|
||||||
|
PG ->>+ CTL: Load library X
|
||||||
|
CTL ->>+ ST: Download the extension that provides X
|
||||||
|
ST ->>- CTL: Finish downloading
|
||||||
|
CTL ->> CTL: Put extension's files in the right place
|
||||||
|
CTL ->>- PG: Ready
|
||||||
|
else Library is already loaded
|
||||||
|
note over PG: No-op
|
||||||
|
end
|
||||||
|
and Download all remaining extensions
|
||||||
|
loop Extension X
|
||||||
|
CTL ->>+ ST: Download not-yet-downloaded extension X
|
||||||
|
activate CTL
|
||||||
|
ST ->>- CTL: Finish downloading
|
||||||
|
CTL ->>- CTL: Put extension's files in the right place
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
deactivate PG
|
||||||
|
deactivate CTL
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Summary
|
||||||
|
|
||||||
|
Pros:
|
||||||
|
- Startup is only as slow as it takes to load all (shared_)preload_libraries
|
||||||
|
- Supports BYO Extension
|
||||||
|
|
||||||
|
Cons:
|
||||||
|
- O(sizeof(extensions)) IO requirement for loading all extensions.
|
||||||
|
|
||||||
|
### Alternative solutions
|
||||||
|
|
||||||
|
1. Allow users to add their extensions to the base image
|
||||||
|
|
||||||
|
Pros:
|
||||||
|
- Easy to deploy
|
||||||
|
|
||||||
|
Cons:
|
||||||
|
- Doesn't scale - first start size is dependent on image size;
|
||||||
|
- All extensions are shared across all users: It doesn't allow users to
|
||||||
|
bring their own restrictive-licensed extensions
|
||||||
|
|
||||||
|
2. Bring Your Own compute image
|
||||||
|
|
||||||
|
Pros:
|
||||||
|
- Still easy to deploy
|
||||||
|
- User can bring own patched version of PostgreSQL
|
||||||
|
|
||||||
|
Cons:
|
||||||
|
- First start latency is O(sizeof(extensions image))
|
||||||
|
- Warm instance pool for skipping pod schedule latency is not feasible with
|
||||||
|
O(n) custom images
|
||||||
|
- Support channels are difficult to manage
|
||||||
|
|
||||||
|
3. Download all user extensions in bulk on compute start
|
||||||
|
|
||||||
|
Pros:
|
||||||
|
- Easy to deploy
|
||||||
|
- No startup latency issues for "clean" users.
|
||||||
|
- Warm instance pool for skipping pod schedule latency is possible
|
||||||
|
|
||||||
|
Cons:
|
||||||
|
- Downloading all extensions in advance takes a lot of time, thus startup
|
||||||
|
latency issues
|
||||||
|
|
||||||
|
4. Store user's extensions in persistent storage
|
||||||
|
|
||||||
|
Pros:
|
||||||
|
- Easy to deploy
|
||||||
|
- No startup latency issues
|
||||||
|
- Warm instance pool for skipping pod schedule latency is possible
|
||||||
|
|
||||||
|
Cons:
|
||||||
|
- EC2 instances have only limited number of attachments shared between EBS
|
||||||
|
volumes, direct-attached NVMe drives, and ENIs.
|
||||||
|
- Compute instance migration isn't trivially solved for EBS mounts (e.g.
|
||||||
|
the device is unavailable whilst moving the mount between instances).
|
||||||
|
- EBS can only mount on one instance at a time (except the expensive IO2
|
||||||
|
device type).
|
||||||
|
|
||||||
|
5. Store user's extensions in network drive
|
||||||
|
|
||||||
|
Pros:
|
||||||
|
- Easy to deploy
|
||||||
|
- Few startup latency issues
|
||||||
|
- Warm instance pool for skipping pod schedule latency is possible
|
||||||
|
|
||||||
|
Cons:
|
||||||
|
- We'd need networked drives, and a lot of them, which would store many
|
||||||
|
duplicate extensions.
|
||||||
|
- **UNCHECKED:** Compute instance migration may not work nicely with
|
||||||
|
networked IOs
|
||||||
|
|
||||||
|
|
||||||
|
### Idea extensions
|
||||||
|
|
||||||
|
The extension store does not have to be S3 directly, but could be a Node-local
|
||||||
|
caching service on top of S3. This would reduce the load on the network for
|
||||||
|
popular extensions.
|
||||||
|
|
||||||
|
## Extension Store implementation
|
||||||
|
|
||||||
|
Extension Store in our case is a private S3 bucket.
|
||||||
|
Extensions are stored as tarballs in the bucket. The tarball contains the extension's control file and all the files that the extension needs to run.
|
||||||
|
|
||||||
|
We may also store the control file separately from the tarball to speed up the extension loading.
|
||||||
|
|
||||||
|
`s3://<the-bucket>/extensions/ext-name/sha-256+1234abcd1234abcd1234abcd1234abcd/bundle.tar`
|
||||||
|
|
||||||
|
where `ext-name` is an extension name and `sha-256+1234abcd1234abcd1234abcd1234abcd` is a hash of a specific extension version tarball.
|
||||||
|
|
||||||
|
To ensure security, there is no direct access to the S3 bucket from compute node.
|
||||||
|
|
||||||
|
Control plane forms a list of extensions available to the compute node
|
||||||
|
and forms a short-lived [pre-signed URL](https://docs.aws.amazon.com/AmazonS3/latest/userguide/ShareObjectPreSignedURL.html)
|
||||||
|
for each extension that is available to the compute node.
|
||||||
|
|
||||||
|
so, `compute_ctl` receives spec in the following format
|
||||||
|
|
||||||
|
```
|
||||||
|
"extensions": [{
|
||||||
|
"meta_format": 1,
|
||||||
|
"extension_name": "postgis",
|
||||||
|
"link": "https://<the-bucket>/extensions/sha-256+1234abcd1234abcd1234abcd1234abcd/bundle.tar?AWSAccessKeyId=1234abcd1234abcd1234abcd1234abcd&Expires=1234567890&Signature=1234abcd1234abcd1234abcd1234abcd",
|
||||||
|
...
|
||||||
|
}]
|
||||||
|
```
|
||||||
|
|
||||||
|
`compute_ctl` then downloads the extension from the link and unpacks it to the right place.
|
||||||
|
|
||||||
|
### How do we handle private extensions?
|
||||||
|
|
||||||
|
Private and public extensions are treated equally from the Extension Store perspective.
|
||||||
|
The only difference is that the private extensions are not listed in the user UI (managed by control plane).
|
||||||
|
|
||||||
|
### How to add new extension to the Extension Store?
|
||||||
|
|
||||||
|
Since we need to verify that the extension is compatible with the compute node and doesn't contain any malicious code,
|
||||||
|
we need to review the extension before adding it to the Extension Store.
|
||||||
|
|
||||||
|
I do not expect that we will have a lot of extensions to review, so we can do it manually for now.
|
||||||
|
|
||||||
|
Some admin UI may be added later to automate this process.
|
||||||
|
|
||||||
|
The list of extensions available to a compute node is stored in the console database.
|
||||||
|
|
||||||
|
### How is the list of available extensions managed?
|
||||||
|
|
||||||
|
We need to add new tables to the console database to store the list of available extensions, their versions and access rights.
|
||||||
|
|
||||||
|
something like this:
|
||||||
|
|
||||||
|
```
|
||||||
|
CREATE TABLE extensions (
|
||||||
|
id SERIAL PRIMARY KEY,
|
||||||
|
name VARCHAR(255) NOT NULL,
|
||||||
|
version VARCHAR(255) NOT NULL,
|
||||||
|
hash VARCHAR(255) NOT NULL, // this is the path to the extension in the Extension Store
|
||||||
|
supported_postgres_versions integer[] NOT NULL,
|
||||||
|
is_public BOOLEAN NOT NULL, // public extensions are available to all users
|
||||||
|
is_shared_preload BOOLEAN NOT NULL, // these extensions require postgres restart
|
||||||
|
is_preload BOOLEAN NOT NULL,
|
||||||
|
license VARCHAR(255) NOT NULL,
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE TABLE user_extensions (
|
||||||
|
user_id INTEGER NOT NULL,
|
||||||
|
extension_id INTEGER NOT NULL,
|
||||||
|
FOREIGN KEY (user_id) REFERENCES users (id),
|
||||||
|
FOREIGN KEY (extension_id) REFERENCES extensions (id)
|
||||||
|
);
|
||||||
|
```
|
||||||
|
|
||||||
|
When new extension is added to the Extension Store, we add a new record to the table and set permissions.
|
||||||
|
|
||||||
|
In UI, user may select the extensions that they want to use with their compute node.
|
||||||
|
|
||||||
|
NOTE: Extensions that require postgres restart will not be available until the next compute restart.
|
||||||
|
Also, currently user cannot force postgres restart. We should add this feature later.
|
||||||
|
|
||||||
|
For other extensions, we must communicate updates to `compute_ctl` and they will be downloaded in the background.
|
||||||
|
|
||||||
|
### How can user update the extension?
|
||||||
|
|
||||||
|
User can update the extension by selecting the new version of the extension in the UI.
|
||||||
|
|
||||||
|
### Alternatives
|
||||||
|
|
||||||
|
For extensions written on trusted languages we can also adopt
|
||||||
|
`dbdev` PostgreSQL Package Manager based on `pg_tle` by Supabase.
|
||||||
|
This will increase the amount supported extensions and decrease the amount of work required to support them.
|
||||||
@@ -71,7 +71,6 @@ pub struct ComputeMetrics {
|
|||||||
pub wait_for_spec_ms: u64,
|
pub wait_for_spec_ms: u64,
|
||||||
pub sync_safekeepers_ms: u64,
|
pub sync_safekeepers_ms: u64,
|
||||||
pub basebackup_ms: u64,
|
pub basebackup_ms: u64,
|
||||||
pub start_postgres_ms: u64,
|
|
||||||
pub config_ms: u64,
|
pub config_ms: u64,
|
||||||
pub total_startup_ms: u64,
|
pub total_startup_ms: u64,
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -23,7 +23,6 @@ use prometheus::{Registry, Result};
|
|||||||
pub mod launch_timestamp;
|
pub mod launch_timestamp;
|
||||||
mod wrappers;
|
mod wrappers;
|
||||||
pub use wrappers::{CountedReader, CountedWriter};
|
pub use wrappers::{CountedReader, CountedWriter};
|
||||||
pub mod metric_vec_duration;
|
|
||||||
|
|
||||||
pub type UIntGauge = GenericGauge<AtomicU64>;
|
pub type UIntGauge = GenericGauge<AtomicU64>;
|
||||||
pub type UIntGaugeVec = GenericGaugeVec<AtomicU64>;
|
pub type UIntGaugeVec = GenericGaugeVec<AtomicU64>;
|
||||||
|
|||||||
@@ -1,23 +0,0 @@
|
|||||||
//! Helpers for observing duration on HistogramVec / CounterVec / GaugeVec / MetricVec<T>.
|
|
||||||
|
|
||||||
use std::{future::Future, time::Instant};
|
|
||||||
|
|
||||||
pub trait DurationResultObserver {
|
|
||||||
fn observe_result<T, E>(&self, res: &Result<T, E>, duration: std::time::Duration);
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn observe_async_block_duration_by_result<
|
|
||||||
T,
|
|
||||||
E,
|
|
||||||
F: Future<Output = Result<T, E>>,
|
|
||||||
O: DurationResultObserver,
|
|
||||||
>(
|
|
||||||
observer: &O,
|
|
||||||
block: F,
|
|
||||||
) -> Result<T, E> {
|
|
||||||
let start = Instant::now();
|
|
||||||
let result = block.await;
|
|
||||||
let duration = start.elapsed();
|
|
||||||
observer.observe_result(&result, duration);
|
|
||||||
result
|
|
||||||
}
|
|
||||||
@@ -184,6 +184,20 @@ pub enum GenericRemoteStorage {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl GenericRemoteStorage {
|
impl GenericRemoteStorage {
|
||||||
|
// A function for listing all the files in a "directory"
|
||||||
|
// Example:
|
||||||
|
// list_files("foo/bar") = ["foo/bar/a.txt", "foo/bar/b.txt"]
|
||||||
|
pub async fn list_files(&self, folder: Option<&RemotePath>) -> anyhow::Result<Vec<RemotePath>> {
|
||||||
|
match self {
|
||||||
|
Self::LocalFs(s) => s.list_files(folder).await,
|
||||||
|
Self::AwsS3(s) => s.list_files(folder).await,
|
||||||
|
Self::Unreliable(s) => s.list_files(folder).await,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// lists common *prefixes*, if any of files
|
||||||
|
// Example:
|
||||||
|
// list_prefixes("foo123","foo567","bar123","bar432") = ["foo", "bar"]
|
||||||
pub async fn list_prefixes(
|
pub async fn list_prefixes(
|
||||||
&self,
|
&self,
|
||||||
prefix: Option<&RemotePath>,
|
prefix: Option<&RemotePath>,
|
||||||
@@ -195,14 +209,6 @@ impl GenericRemoteStorage {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn list_files(&self, folder: Option<&RemotePath>) -> anyhow::Result<Vec<RemotePath>> {
|
|
||||||
match self {
|
|
||||||
Self::LocalFs(s) => s.list_files(folder).await,
|
|
||||||
Self::AwsS3(s) => s.list_files(folder).await,
|
|
||||||
Self::Unreliable(s) => s.list_files(folder).await,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn upload(
|
pub async fn upload(
|
||||||
&self,
|
&self,
|
||||||
from: impl io::AsyncRead + Unpin + Send + Sync + 'static,
|
from: impl io::AsyncRead + Unpin + Send + Sync + 'static,
|
||||||
|
|||||||
@@ -349,6 +349,7 @@ impl RemoteStorage for S3Bucket {
|
|||||||
|
|
||||||
/// See the doc for `RemoteStorage::list_files`
|
/// See the doc for `RemoteStorage::list_files`
|
||||||
async fn list_files(&self, folder: Option<&RemotePath>) -> anyhow::Result<Vec<RemotePath>> {
|
async fn list_files(&self, folder: Option<&RemotePath>) -> anyhow::Result<Vec<RemotePath>> {
|
||||||
|
// TODO: if bucket prefix is empty, folder is prefixed with a "/" I think. Is this desired?
|
||||||
let folder_name = folder
|
let folder_name = folder
|
||||||
.map(|p| self.relative_path_to_s3_object(p))
|
.map(|p| self.relative_path_to_s3_object(p))
|
||||||
.or_else(|| self.prefix_in_bucket.clone());
|
.or_else(|| self.prefix_in_bucket.clone());
|
||||||
|
|||||||
@@ -173,15 +173,10 @@ async fn s3_delete_objects_works(ctx: &mut MaybeEnabledS3) -> anyhow::Result<()>
|
|||||||
let path2 = RemotePath::new(&PathBuf::from(format!("{}/path2", ctx.base_prefix,)))
|
let path2 = RemotePath::new(&PathBuf::from(format!("{}/path2", ctx.base_prefix,)))
|
||||||
.with_context(|| "RemotePath conversion")?;
|
.with_context(|| "RemotePath conversion")?;
|
||||||
|
|
||||||
let path3 = RemotePath::new(&PathBuf::from(format!("{}/path3", ctx.base_prefix,)))
|
|
||||||
.with_context(|| "RemotePath conversion")?;
|
|
||||||
|
|
||||||
let data1 = "remote blob data1".as_bytes();
|
let data1 = "remote blob data1".as_bytes();
|
||||||
let data1_len = data1.len();
|
let data1_len = data1.len();
|
||||||
let data2 = "remote blob data2".as_bytes();
|
let data2 = "remote blob data2".as_bytes();
|
||||||
let data2_len = data2.len();
|
let data2_len = data2.len();
|
||||||
let data3 = "remote blob data3".as_bytes();
|
|
||||||
let data3_len = data3.len();
|
|
||||||
ctx.client
|
ctx.client
|
||||||
.upload(std::io::Cursor::new(data1), data1_len, &path1, None)
|
.upload(std::io::Cursor::new(data1), data1_len, &path1, None)
|
||||||
.await?;
|
.await?;
|
||||||
@@ -190,18 +185,8 @@ async fn s3_delete_objects_works(ctx: &mut MaybeEnabledS3) -> anyhow::Result<()>
|
|||||||
.upload(std::io::Cursor::new(data2), data2_len, &path2, None)
|
.upload(std::io::Cursor::new(data2), data2_len, &path2, None)
|
||||||
.await?;
|
.await?;
|
||||||
|
|
||||||
ctx.client
|
|
||||||
.upload(std::io::Cursor::new(data3), data3_len, &path3, None)
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
ctx.client.delete_objects(&[path1, path2]).await?;
|
ctx.client.delete_objects(&[path1, path2]).await?;
|
||||||
|
|
||||||
let prefixes = ctx.client.list_prefixes(None).await?;
|
|
||||||
|
|
||||||
assert_eq!(prefixes.len(), 1);
|
|
||||||
|
|
||||||
ctx.client.delete_objects(&[path3]).await?;
|
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ edition.workspace = true
|
|||||||
license.workspace = true
|
license.workspace = true
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
|
atty.workspace = true
|
||||||
sentry.workspace = true
|
sentry.workspace = true
|
||||||
async-trait.workspace = true
|
async-trait.workspace = true
|
||||||
anyhow.workspace = true
|
anyhow.workspace = true
|
||||||
|
|||||||
@@ -84,7 +84,7 @@ pub fn init(
|
|||||||
let r = r.with({
|
let r = r.with({
|
||||||
let log_layer = tracing_subscriber::fmt::layer()
|
let log_layer = tracing_subscriber::fmt::layer()
|
||||||
.with_target(false)
|
.with_target(false)
|
||||||
.with_ansi(false)
|
.with_ansi(atty::is(atty::Stream::Stdout))
|
||||||
.with_writer(std::io::stdout);
|
.with_writer(std::io::stdout);
|
||||||
let log_layer = match log_format {
|
let log_layer = match log_format {
|
||||||
LogFormat::Json => log_layer.json().boxed(),
|
LogFormat::Json => log_layer.json().boxed(),
|
||||||
|
|||||||
@@ -1,23 +1,22 @@
|
|||||||
use pageserver::keyspace::{KeyPartitioning, KeySpace};
|
use pageserver::keyspace::{KeyPartitioning, KeySpace};
|
||||||
use pageserver::repository::Key;
|
use pageserver::repository::Key;
|
||||||
use pageserver::tenant::layer_map::LayerMap;
|
use pageserver::tenant::layer_map::LayerMap;
|
||||||
use pageserver::tenant::storage_layer::{tests::LayerDescriptor, Layer, LayerFileName};
|
use pageserver::tenant::storage_layer::{Layer, LayerDescriptor, LayerFileName};
|
||||||
use pageserver::tenant::storage_layer::{PersistentLayer, PersistentLayerDesc};
|
|
||||||
use rand::prelude::{SeedableRng, SliceRandom, StdRng};
|
use rand::prelude::{SeedableRng, SliceRandom, StdRng};
|
||||||
use std::cmp::{max, min};
|
use std::cmp::{max, min};
|
||||||
use std::fs::File;
|
use std::fs::File;
|
||||||
use std::io::{BufRead, BufReader};
|
use std::io::{BufRead, BufReader};
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
use std::str::FromStr;
|
use std::str::FromStr;
|
||||||
|
use std::sync::Arc;
|
||||||
use std::time::Instant;
|
use std::time::Instant;
|
||||||
use utils::id::{TenantId, TimelineId};
|
|
||||||
|
|
||||||
use utils::lsn::Lsn;
|
use utils::lsn::Lsn;
|
||||||
|
|
||||||
use criterion::{black_box, criterion_group, criterion_main, Criterion};
|
use criterion::{black_box, criterion_group, criterion_main, Criterion};
|
||||||
|
|
||||||
fn build_layer_map(filename_dump: PathBuf) -> LayerMap {
|
fn build_layer_map(filename_dump: PathBuf) -> LayerMap<LayerDescriptor> {
|
||||||
let mut layer_map = LayerMap::default();
|
let mut layer_map = LayerMap::<LayerDescriptor>::default();
|
||||||
|
|
||||||
let mut min_lsn = Lsn(u64::MAX);
|
let mut min_lsn = Lsn(u64::MAX);
|
||||||
let mut max_lsn = Lsn(0);
|
let mut max_lsn = Lsn(0);
|
||||||
@@ -34,7 +33,7 @@ fn build_layer_map(filename_dump: PathBuf) -> LayerMap {
|
|||||||
min_lsn = min(min_lsn, lsn_range.start);
|
min_lsn = min(min_lsn, lsn_range.start);
|
||||||
max_lsn = max(max_lsn, Lsn(lsn_range.end.0 - 1));
|
max_lsn = max(max_lsn, Lsn(lsn_range.end.0 - 1));
|
||||||
|
|
||||||
updates.insert_historic(layer.layer_desc().clone());
|
updates.insert_historic(layer.get_persistent_layer_desc(), Arc::new(layer));
|
||||||
}
|
}
|
||||||
|
|
||||||
println!("min: {min_lsn}, max: {max_lsn}");
|
println!("min: {min_lsn}, max: {max_lsn}");
|
||||||
@@ -44,7 +43,7 @@ fn build_layer_map(filename_dump: PathBuf) -> LayerMap {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Construct a layer map query pattern for benchmarks
|
/// Construct a layer map query pattern for benchmarks
|
||||||
fn uniform_query_pattern(layer_map: &LayerMap) -> Vec<(Key, Lsn)> {
|
fn uniform_query_pattern(layer_map: &LayerMap<LayerDescriptor>) -> Vec<(Key, Lsn)> {
|
||||||
// For each image layer we query one of the pages contained, at LSN right
|
// For each image layer we query one of the pages contained, at LSN right
|
||||||
// before the image layer was created. This gives us a somewhat uniform
|
// before the image layer was created. This gives us a somewhat uniform
|
||||||
// coverage of both the lsn and key space because image layers have
|
// coverage of both the lsn and key space because image layers have
|
||||||
@@ -70,7 +69,7 @@ fn uniform_query_pattern(layer_map: &LayerMap) -> Vec<(Key, Lsn)> {
|
|||||||
|
|
||||||
// Construct a partitioning for testing get_difficulty map when we
|
// Construct a partitioning for testing get_difficulty map when we
|
||||||
// don't have an exact result of `collect_keyspace` to work with.
|
// don't have an exact result of `collect_keyspace` to work with.
|
||||||
fn uniform_key_partitioning(layer_map: &LayerMap, _lsn: Lsn) -> KeyPartitioning {
|
fn uniform_key_partitioning(layer_map: &LayerMap<LayerDescriptor>, _lsn: Lsn) -> KeyPartitioning {
|
||||||
let mut parts = Vec::new();
|
let mut parts = Vec::new();
|
||||||
|
|
||||||
// We add a partition boundary at the start of each image layer,
|
// We add a partition boundary at the start of each image layer,
|
||||||
@@ -210,15 +209,13 @@ fn bench_sequential(c: &mut Criterion) {
|
|||||||
for i in 0..100_000 {
|
for i in 0..100_000 {
|
||||||
let i32 = (i as u32) % 100;
|
let i32 = (i as u32) % 100;
|
||||||
let zero = Key::from_hex("000000000000000000000000000000000000").unwrap();
|
let zero = Key::from_hex("000000000000000000000000000000000000").unwrap();
|
||||||
let layer = LayerDescriptor::from(PersistentLayerDesc::new_img(
|
let layer = LayerDescriptor {
|
||||||
TenantId::generate(),
|
key: zero.add(10 * i32)..zero.add(10 * i32 + 1),
|
||||||
TimelineId::generate(),
|
lsn: Lsn(i)..Lsn(i + 1),
|
||||||
zero.add(10 * i32)..zero.add(10 * i32 + 1),
|
is_incremental: false,
|
||||||
Lsn(i),
|
short_id: format!("Layer {}", i),
|
||||||
false,
|
};
|
||||||
0,
|
updates.insert_historic(layer.get_persistent_layer_desc(), Arc::new(layer));
|
||||||
));
|
|
||||||
updates.insert_historic(layer.layer_desc().clone());
|
|
||||||
}
|
}
|
||||||
updates.flush();
|
updates.flush();
|
||||||
println!("Finished layer map init in {:?}", now.elapsed());
|
println!("Finished layer map init in {:?}", now.elapsed());
|
||||||
|
|||||||
@@ -495,50 +495,50 @@ fn start_pageserver(
|
|||||||
Ok(())
|
Ok(())
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
}
|
|
||||||
|
|
||||||
if let Some(metric_collection_endpoint) = &conf.metric_collection_endpoint {
|
if let Some(metric_collection_endpoint) = &conf.metric_collection_endpoint {
|
||||||
let background_jobs_barrier = background_jobs_barrier;
|
let background_jobs_barrier = background_jobs_barrier;
|
||||||
let metrics_ctx = RequestContext::todo_child(
|
let metrics_ctx = RequestContext::todo_child(
|
||||||
TaskKind::MetricsCollection,
|
TaskKind::MetricsCollection,
|
||||||
// This task itself shouldn't download anything.
|
// This task itself shouldn't download anything.
|
||||||
// The actual size calculation does need downloads, and
|
// The actual size calculation does need downloads, and
|
||||||
// creates a child context with the right DownloadBehavior.
|
// creates a child context with the right DownloadBehavior.
|
||||||
DownloadBehavior::Error,
|
DownloadBehavior::Error,
|
||||||
);
|
);
|
||||||
task_mgr::spawn(
|
task_mgr::spawn(
|
||||||
crate::BACKGROUND_RUNTIME.handle(),
|
MGMT_REQUEST_RUNTIME.handle(),
|
||||||
TaskKind::MetricsCollection,
|
TaskKind::MetricsCollection,
|
||||||
None,
|
None,
|
||||||
None,
|
None,
|
||||||
"consumption metrics collection",
|
"consumption metrics collection",
|
||||||
true,
|
true,
|
||||||
async move {
|
async move {
|
||||||
// first wait until background jobs are cleared to launch.
|
// first wait until background jobs are cleared to launch.
|
||||||
//
|
//
|
||||||
// this is because we only process active tenants and timelines, and the
|
// this is because we only process active tenants and timelines, and the
|
||||||
// Timeline::get_current_logical_size will spawn the logical size calculation,
|
// Timeline::get_current_logical_size will spawn the logical size calculation,
|
||||||
// which will not be rate-limited.
|
// which will not be rate-limited.
|
||||||
let cancel = task_mgr::shutdown_token();
|
let cancel = task_mgr::shutdown_token();
|
||||||
|
|
||||||
tokio::select! {
|
tokio::select! {
|
||||||
_ = cancel.cancelled() => { return Ok(()); },
|
_ = cancel.cancelled() => { return Ok(()); },
|
||||||
_ = background_jobs_barrier.wait() => {}
|
_ = background_jobs_barrier.wait() => {}
|
||||||
};
|
};
|
||||||
|
|
||||||
pageserver::consumption_metrics::collect_metrics(
|
pageserver::consumption_metrics::collect_metrics(
|
||||||
metric_collection_endpoint,
|
metric_collection_endpoint,
|
||||||
conf.metric_collection_interval,
|
conf.metric_collection_interval,
|
||||||
conf.cached_metric_collection_interval,
|
conf.cached_metric_collection_interval,
|
||||||
conf.synthetic_size_calculation_interval,
|
conf.synthetic_size_calculation_interval,
|
||||||
conf.id,
|
conf.id,
|
||||||
metrics_ctx,
|
metrics_ctx,
|
||||||
)
|
)
|
||||||
.instrument(info_span!("metrics_collection"))
|
.instrument(info_span!("metrics_collection"))
|
||||||
.await?;
|
.await?;
|
||||||
Ok(())
|
Ok(())
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Spawn a task to listen for libpq connections. It will spawn further tasks
|
// Spawn a task to listen for libpq connections. It will spawn further tasks
|
||||||
|
|||||||
@@ -96,12 +96,12 @@ pub mod defaults {
|
|||||||
|
|
||||||
#background_task_maximum_delay = '{DEFAULT_BACKGROUND_TASK_MAXIMUM_DELAY}'
|
#background_task_maximum_delay = '{DEFAULT_BACKGROUND_TASK_MAXIMUM_DELAY}'
|
||||||
|
|
||||||
[tenant_config]
|
# [tenant_config]
|
||||||
#checkpoint_distance = {DEFAULT_CHECKPOINT_DISTANCE} # in bytes
|
#checkpoint_distance = {DEFAULT_CHECKPOINT_DISTANCE} # in bytes
|
||||||
#checkpoint_timeout = {DEFAULT_CHECKPOINT_TIMEOUT}
|
#checkpoint_timeout = {DEFAULT_CHECKPOINT_TIMEOUT}
|
||||||
#compaction_target_size = {DEFAULT_COMPACTION_TARGET_SIZE} # in bytes
|
#compaction_target_size = {DEFAULT_COMPACTION_TARGET_SIZE} # in bytes
|
||||||
#compaction_period = '{DEFAULT_COMPACTION_PERIOD}'
|
#compaction_period = '{DEFAULT_COMPACTION_PERIOD}'
|
||||||
#compaction_threshold = {DEFAULT_COMPACTION_THRESHOLD}
|
#compaction_threshold = '{DEFAULT_COMPACTION_THRESHOLD}'
|
||||||
|
|
||||||
#gc_period = '{DEFAULT_GC_PERIOD}'
|
#gc_period = '{DEFAULT_GC_PERIOD}'
|
||||||
#gc_horizon = {DEFAULT_GC_HORIZON}
|
#gc_horizon = {DEFAULT_GC_HORIZON}
|
||||||
@@ -111,8 +111,7 @@ pub mod defaults {
|
|||||||
#min_resident_size_override = .. # in bytes
|
#min_resident_size_override = .. # in bytes
|
||||||
#evictions_low_residence_duration_metric_threshold = '{DEFAULT_EVICTIONS_LOW_RESIDENCE_DURATION_METRIC_THRESHOLD}'
|
#evictions_low_residence_duration_metric_threshold = '{DEFAULT_EVICTIONS_LOW_RESIDENCE_DURATION_METRIC_THRESHOLD}'
|
||||||
#gc_feedback = false
|
#gc_feedback = false
|
||||||
|
# [remote_storage]
|
||||||
[remote_storage]
|
|
||||||
|
|
||||||
"###
|
"###
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -24,8 +24,6 @@ const RESIDENT_SIZE: &str = "resident_size";
|
|||||||
const REMOTE_STORAGE_SIZE: &str = "remote_storage_size";
|
const REMOTE_STORAGE_SIZE: &str = "remote_storage_size";
|
||||||
const TIMELINE_LOGICAL_SIZE: &str = "timeline_logical_size";
|
const TIMELINE_LOGICAL_SIZE: &str = "timeline_logical_size";
|
||||||
|
|
||||||
const DEFAULT_HTTP_REPORTING_TIMEOUT: Duration = Duration::from_secs(60);
|
|
||||||
|
|
||||||
#[serde_as]
|
#[serde_as]
|
||||||
#[derive(Serialize, Debug)]
|
#[derive(Serialize, Debug)]
|
||||||
struct Ids {
|
struct Ids {
|
||||||
@@ -75,10 +73,7 @@ pub async fn collect_metrics(
|
|||||||
);
|
);
|
||||||
|
|
||||||
// define client here to reuse it for all requests
|
// define client here to reuse it for all requests
|
||||||
let client = reqwest::ClientBuilder::new()
|
let client = reqwest::Client::new();
|
||||||
.timeout(DEFAULT_HTTP_REPORTING_TIMEOUT)
|
|
||||||
.build()
|
|
||||||
.expect("Failed to create http client with timeout");
|
|
||||||
let mut cached_metrics: HashMap<PageserverConsumptionMetricsKey, u64> = HashMap::new();
|
let mut cached_metrics: HashMap<PageserverConsumptionMetricsKey, u64> = HashMap::new();
|
||||||
let mut prev_iteration_time: std::time::Instant = std::time::Instant::now();
|
let mut prev_iteration_time: std::time::Instant = std::time::Instant::now();
|
||||||
|
|
||||||
@@ -88,7 +83,7 @@ pub async fn collect_metrics(
|
|||||||
info!("collect_metrics received cancellation request");
|
info!("collect_metrics received cancellation request");
|
||||||
return Ok(());
|
return Ok(());
|
||||||
},
|
},
|
||||||
tick_at = ticker.tick() => {
|
_ = ticker.tick() => {
|
||||||
|
|
||||||
// send cached metrics every cached_metric_collection_interval
|
// send cached metrics every cached_metric_collection_interval
|
||||||
let send_cached = prev_iteration_time.elapsed() >= cached_metric_collection_interval;
|
let send_cached = prev_iteration_time.elapsed() >= cached_metric_collection_interval;
|
||||||
@@ -98,12 +93,6 @@ pub async fn collect_metrics(
|
|||||||
}
|
}
|
||||||
|
|
||||||
collect_metrics_iteration(&client, &mut cached_metrics, metric_collection_endpoint, node_id, &ctx, send_cached).await;
|
collect_metrics_iteration(&client, &mut cached_metrics, metric_collection_endpoint, node_id, &ctx, send_cached).await;
|
||||||
|
|
||||||
crate::tenant::tasks::warn_when_period_overrun(
|
|
||||||
tick_at.elapsed(),
|
|
||||||
metric_collection_interval,
|
|
||||||
"consumption_metrics_collect_metrics",
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -284,43 +273,32 @@ pub async fn collect_metrics_iteration(
|
|||||||
})
|
})
|
||||||
.expect("PageserverConsumptionMetric should not fail serialization");
|
.expect("PageserverConsumptionMetric should not fail serialization");
|
||||||
|
|
||||||
const MAX_RETRIES: u32 = 3;
|
let res = client
|
||||||
|
.post(metric_collection_endpoint.clone())
|
||||||
|
.json(&chunk_json)
|
||||||
|
.send()
|
||||||
|
.await;
|
||||||
|
|
||||||
for attempt in 0..MAX_RETRIES {
|
match res {
|
||||||
let res = client
|
Ok(res) => {
|
||||||
.post(metric_collection_endpoint.clone())
|
if res.status().is_success() {
|
||||||
.json(&chunk_json)
|
// update cached metrics after they were sent successfully
|
||||||
.send()
|
for (curr_key, curr_val) in chunk.iter() {
|
||||||
.await;
|
cached_metrics.insert(curr_key.clone(), *curr_val);
|
||||||
|
}
|
||||||
match res {
|
} else {
|
||||||
Ok(res) => {
|
error!("metrics endpoint refused the sent metrics: {:?}", res);
|
||||||
if res.status().is_success() {
|
for metric in chunk_to_send.iter() {
|
||||||
// update cached metrics after they were sent successfully
|
// Report if the metric value is suspiciously large
|
||||||
for (curr_key, curr_val) in chunk.iter() {
|
if metric.value > (1u64 << 40) {
|
||||||
cached_metrics.insert(curr_key.clone(), *curr_val);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
error!("metrics endpoint refused the sent metrics: {:?}", res);
|
|
||||||
for metric in chunk_to_send
|
|
||||||
.iter()
|
|
||||||
.filter(|metric| metric.value > (1u64 << 40))
|
|
||||||
{
|
|
||||||
// Report if the metric value is suspiciously large
|
|
||||||
error!("potentially abnormal metric value: {:?}", metric);
|
error!("potentially abnormal metric value: {:?}", metric);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
|
||||||
}
|
|
||||||
Err(err) if err.is_timeout() => {
|
|
||||||
error!(attempt, "timeout sending metrics, retrying immediately");
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
Err(err) => {
|
|
||||||
error!(attempt, ?err, "failed to send metrics");
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Err(err) => {
|
||||||
|
error!("failed to send metrics: {:?}", err);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -339,7 +317,7 @@ pub async fn calculate_synthetic_size_worker(
|
|||||||
_ = task_mgr::shutdown_watcher() => {
|
_ = task_mgr::shutdown_watcher() => {
|
||||||
return Ok(());
|
return Ok(());
|
||||||
},
|
},
|
||||||
tick_at = ticker.tick() => {
|
_ = ticker.tick() => {
|
||||||
|
|
||||||
let tenants = match mgr::list_tenants().await {
|
let tenants = match mgr::list_tenants().await {
|
||||||
Ok(tenants) => tenants,
|
Ok(tenants) => tenants,
|
||||||
@@ -365,12 +343,6 @@ pub async fn calculate_synthetic_size_worker(
|
|||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
crate::tenant::tasks::warn_when_period_overrun(
|
|
||||||
tick_at.elapsed(),
|
|
||||||
synthetic_size_calculation_interval,
|
|
||||||
"consumption_metrics_synthetic_size_worker",
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -110,6 +110,7 @@ pub fn launch_disk_usage_global_eviction_task(
|
|||||||
|
|
||||||
disk_usage_eviction_task(&state, task_config, storage, &conf.tenants_path(), cancel)
|
disk_usage_eviction_task(&state, task_config, storage, &conf.tenants_path(), cancel)
|
||||||
.await;
|
.await;
|
||||||
|
info!("disk usage based eviction task finishing");
|
||||||
Ok(())
|
Ok(())
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
@@ -125,16 +126,13 @@ async fn disk_usage_eviction_task(
|
|||||||
tenants_dir: &Path,
|
tenants_dir: &Path,
|
||||||
cancel: CancellationToken,
|
cancel: CancellationToken,
|
||||||
) {
|
) {
|
||||||
scopeguard::defer! {
|
|
||||||
info!("disk usage based eviction task finishing");
|
|
||||||
};
|
|
||||||
|
|
||||||
use crate::tenant::tasks::random_init_delay;
|
use crate::tenant::tasks::random_init_delay;
|
||||||
{
|
{
|
||||||
if random_init_delay(task_config.period, &cancel)
|
if random_init_delay(task_config.period, &cancel)
|
||||||
.await
|
.await
|
||||||
.is_err()
|
.is_err()
|
||||||
{
|
{
|
||||||
|
info!("shutting down");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -169,6 +167,7 @@ async fn disk_usage_eviction_task(
|
|||||||
tokio::select! {
|
tokio::select! {
|
||||||
_ = tokio::time::sleep_until(sleep_until) => {},
|
_ = tokio::time::sleep_until(sleep_until) => {},
|
||||||
_ = cancel.cancelled() => {
|
_ = cancel.cancelled() => {
|
||||||
|
info!("shutting down");
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -315,7 +314,7 @@ pub async fn disk_usage_eviction_task_iteration_impl<U: Usage>(
|
|||||||
partition,
|
partition,
|
||||||
candidate.layer.get_tenant_id(),
|
candidate.layer.get_tenant_id(),
|
||||||
candidate.layer.get_timeline_id(),
|
candidate.layer.get_timeline_id(),
|
||||||
candidate.layer,
|
candidate.layer.filename().file_name(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -186,8 +186,10 @@ paths:
|
|||||||
schema:
|
schema:
|
||||||
$ref: "#/components/schemas/Error"
|
$ref: "#/components/schemas/Error"
|
||||||
delete:
|
delete:
|
||||||
description: "Attempts to delete specified timeline. 500 and 409 errors should be retried"
|
description: "Attempts to delete specified timeline. On 500 errors should be retried"
|
||||||
responses:
|
responses:
|
||||||
|
"200":
|
||||||
|
description: Ok
|
||||||
"400":
|
"400":
|
||||||
description: Error when no tenant id found in path or no timeline id
|
description: Error when no tenant id found in path or no timeline id
|
||||||
content:
|
content:
|
||||||
@@ -212,12 +214,6 @@ paths:
|
|||||||
application/json:
|
application/json:
|
||||||
schema:
|
schema:
|
||||||
$ref: "#/components/schemas/NotFoundError"
|
$ref: "#/components/schemas/NotFoundError"
|
||||||
"409":
|
|
||||||
description: Deletion is already in progress, continue polling
|
|
||||||
content:
|
|
||||||
application/json:
|
|
||||||
schema:
|
|
||||||
$ref: "#/components/schemas/ConflictError"
|
|
||||||
"412":
|
"412":
|
||||||
description: Tenant is missing, or timeline has children
|
description: Tenant is missing, or timeline has children
|
||||||
content:
|
content:
|
||||||
@@ -722,12 +718,6 @@ paths:
|
|||||||
application/json:
|
application/json:
|
||||||
schema:
|
schema:
|
||||||
$ref: "#/components/schemas/ForbiddenError"
|
$ref: "#/components/schemas/ForbiddenError"
|
||||||
"406":
|
|
||||||
description: Permanently unsatisfiable request, don't retry.
|
|
||||||
content:
|
|
||||||
application/json:
|
|
||||||
schema:
|
|
||||||
$ref: "#/components/schemas/Error"
|
|
||||||
"409":
|
"409":
|
||||||
description: Timeline already exists, creation skipped
|
description: Timeline already exists, creation skipped
|
||||||
content:
|
content:
|
||||||
|
|||||||
@@ -23,6 +23,7 @@ use super::models::{
|
|||||||
TimelineCreateRequest, TimelineGcRequest, TimelineInfo,
|
TimelineCreateRequest, TimelineGcRequest, TimelineInfo,
|
||||||
};
|
};
|
||||||
use crate::context::{DownloadBehavior, RequestContext};
|
use crate::context::{DownloadBehavior, RequestContext};
|
||||||
|
use crate::disk_usage_eviction_task;
|
||||||
use crate::metrics::{StorageTimeOperation, STORAGE_TIME_GLOBAL};
|
use crate::metrics::{StorageTimeOperation, STORAGE_TIME_GLOBAL};
|
||||||
use crate::pgdatadir_mapping::LsnForTimestamp;
|
use crate::pgdatadir_mapping::LsnForTimestamp;
|
||||||
use crate::task_mgr::TaskKind;
|
use crate::task_mgr::TaskKind;
|
||||||
@@ -34,7 +35,6 @@ use crate::tenant::size::ModelInputs;
|
|||||||
use crate::tenant::storage_layer::LayerAccessStatsReset;
|
use crate::tenant::storage_layer::LayerAccessStatsReset;
|
||||||
use crate::tenant::{LogicalSizeCalculationCause, PageReconstructError, Timeline};
|
use crate::tenant::{LogicalSizeCalculationCause, PageReconstructError, Timeline};
|
||||||
use crate::{config::PageServerConf, tenant::mgr};
|
use crate::{config::PageServerConf, tenant::mgr};
|
||||||
use crate::{disk_usage_eviction_task, tenant};
|
|
||||||
use utils::{
|
use utils::{
|
||||||
auth::JwtAuth,
|
auth::JwtAuth,
|
||||||
http::{
|
http::{
|
||||||
@@ -187,7 +187,6 @@ impl From<crate::tenant::DeleteTimelineError> for ApiError {
|
|||||||
format!("Cannot delete timeline which has child timelines: {children:?}")
|
format!("Cannot delete timeline which has child timelines: {children:?}")
|
||||||
.into_boxed_str(),
|
.into_boxed_str(),
|
||||||
),
|
),
|
||||||
a @ AlreadyInProgress => ApiError::Conflict(a.to_string()),
|
|
||||||
Other(e) => ApiError::InternalServerError(e),
|
Other(e) => ApiError::InternalServerError(e),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -328,22 +327,15 @@ async fn timeline_create_handler(
|
|||||||
&ctx,
|
&ctx,
|
||||||
)
|
)
|
||||||
.await {
|
.await {
|
||||||
Ok(new_timeline) => {
|
Ok(Some(new_timeline)) => {
|
||||||
// Created. Construct a TimelineInfo for it.
|
// Created. Construct a TimelineInfo for it.
|
||||||
let timeline_info = build_timeline_info_common(&new_timeline, &ctx)
|
let timeline_info = build_timeline_info_common(&new_timeline, &ctx)
|
||||||
.await
|
.await
|
||||||
.map_err(ApiError::InternalServerError)?;
|
.map_err(ApiError::InternalServerError)?;
|
||||||
json_response(StatusCode::CREATED, timeline_info)
|
json_response(StatusCode::CREATED, timeline_info)
|
||||||
}
|
}
|
||||||
Err(tenant::CreateTimelineError::AlreadyExists) => {
|
Ok(None) => json_response(StatusCode::CONFLICT, ()), // timeline already exists
|
||||||
json_response(StatusCode::CONFLICT, ())
|
Err(err) => Err(ApiError::InternalServerError(err)),
|
||||||
}
|
|
||||||
Err(tenant::CreateTimelineError::AncestorLsn(err)) => {
|
|
||||||
json_response(StatusCode::NOT_ACCEPTABLE, HttpErrorBody::from_msg(
|
|
||||||
format!("{err:#}")
|
|
||||||
))
|
|
||||||
}
|
|
||||||
Err(tenant::CreateTimelineError::Other(err)) => Err(ApiError::InternalServerError(err)),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
.instrument(info_span!("timeline_create", tenant = %tenant_id, timeline_id = %new_timeline_id, lsn=?request_data.ancestor_start_lsn, pg_version=?request_data.pg_version))
|
.instrument(info_span!("timeline_create", tenant = %tenant_id, timeline_id = %new_timeline_id, lsn=?request_data.ancestor_start_lsn, pg_version=?request_data.pg_version))
|
||||||
@@ -1136,6 +1128,8 @@ async fn disk_usage_eviction_run(
|
|||||||
freed_bytes: 0,
|
freed_bytes: 0,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
use crate::task_mgr::MGMT_REQUEST_RUNTIME;
|
||||||
|
|
||||||
let (tx, rx) = tokio::sync::oneshot::channel();
|
let (tx, rx) = tokio::sync::oneshot::channel();
|
||||||
|
|
||||||
let state = get_state(&r);
|
let state = get_state(&r);
|
||||||
@@ -1153,7 +1147,7 @@ async fn disk_usage_eviction_run(
|
|||||||
let _g = cancel.drop_guard();
|
let _g = cancel.drop_guard();
|
||||||
|
|
||||||
crate::task_mgr::spawn(
|
crate::task_mgr::spawn(
|
||||||
crate::task_mgr::BACKGROUND_RUNTIME.handle(),
|
MGMT_REQUEST_RUNTIME.handle(),
|
||||||
TaskKind::DiskUsageEviction,
|
TaskKind::DiskUsageEviction,
|
||||||
None,
|
None,
|
||||||
None,
|
None,
|
||||||
|
|||||||
@@ -1,9 +1,8 @@
|
|||||||
use metrics::metric_vec_duration::DurationResultObserver;
|
|
||||||
use metrics::{
|
use metrics::{
|
||||||
register_counter_vec, register_histogram, register_histogram_vec, register_int_counter,
|
register_counter_vec, register_histogram, register_histogram_vec, register_int_counter,
|
||||||
register_int_counter_vec, register_int_gauge, register_int_gauge_vec, register_uint_gauge,
|
register_int_counter_vec, register_int_gauge, register_int_gauge_vec, register_uint_gauge_vec,
|
||||||
register_uint_gauge_vec, Counter, CounterVec, Histogram, HistogramVec, IntCounter,
|
Counter, CounterVec, Histogram, HistogramVec, IntCounter, IntCounterVec, IntGauge, IntGaugeVec,
|
||||||
IntCounterVec, IntGauge, IntGaugeVec, UIntGauge, UIntGaugeVec,
|
UIntGauge, UIntGaugeVec,
|
||||||
};
|
};
|
||||||
use once_cell::sync::Lazy;
|
use once_cell::sync::Lazy;
|
||||||
use pageserver_api::models::TenantState;
|
use pageserver_api::models::TenantState;
|
||||||
@@ -130,122 +129,6 @@ pub static MATERIALIZED_PAGE_CACHE_HIT: Lazy<IntCounter> = Lazy::new(|| {
|
|||||||
.expect("failed to define a metric")
|
.expect("failed to define a metric")
|
||||||
});
|
});
|
||||||
|
|
||||||
pub struct PageCacheMetrics {
|
|
||||||
pub read_accesses_materialized_page: IntCounter,
|
|
||||||
pub read_accesses_ephemeral: IntCounter,
|
|
||||||
pub read_accesses_immutable: IntCounter,
|
|
||||||
|
|
||||||
pub read_hits_ephemeral: IntCounter,
|
|
||||||
pub read_hits_immutable: IntCounter,
|
|
||||||
pub read_hits_materialized_page_exact: IntCounter,
|
|
||||||
pub read_hits_materialized_page_older_lsn: IntCounter,
|
|
||||||
}
|
|
||||||
|
|
||||||
static PAGE_CACHE_READ_HITS: Lazy<IntCounterVec> = Lazy::new(|| {
|
|
||||||
register_int_counter_vec!(
|
|
||||||
"pageserver_page_cache_read_hits_total",
|
|
||||||
"Number of read accesses to the page cache that hit",
|
|
||||||
&["key_kind", "hit_kind"]
|
|
||||||
)
|
|
||||||
.expect("failed to define a metric")
|
|
||||||
});
|
|
||||||
|
|
||||||
static PAGE_CACHE_READ_ACCESSES: Lazy<IntCounterVec> = Lazy::new(|| {
|
|
||||||
register_int_counter_vec!(
|
|
||||||
"pageserver_page_cache_read_accesses_total",
|
|
||||||
"Number of read accesses to the page cache",
|
|
||||||
&["key_kind"]
|
|
||||||
)
|
|
||||||
.expect("failed to define a metric")
|
|
||||||
});
|
|
||||||
|
|
||||||
pub static PAGE_CACHE: Lazy<PageCacheMetrics> = Lazy::new(|| PageCacheMetrics {
|
|
||||||
read_accesses_materialized_page: {
|
|
||||||
PAGE_CACHE_READ_ACCESSES
|
|
||||||
.get_metric_with_label_values(&["materialized_page"])
|
|
||||||
.unwrap()
|
|
||||||
},
|
|
||||||
|
|
||||||
read_accesses_ephemeral: {
|
|
||||||
PAGE_CACHE_READ_ACCESSES
|
|
||||||
.get_metric_with_label_values(&["ephemeral"])
|
|
||||||
.unwrap()
|
|
||||||
},
|
|
||||||
|
|
||||||
read_accesses_immutable: {
|
|
||||||
PAGE_CACHE_READ_ACCESSES
|
|
||||||
.get_metric_with_label_values(&["immutable"])
|
|
||||||
.unwrap()
|
|
||||||
},
|
|
||||||
|
|
||||||
read_hits_ephemeral: {
|
|
||||||
PAGE_CACHE_READ_HITS
|
|
||||||
.get_metric_with_label_values(&["ephemeral", "-"])
|
|
||||||
.unwrap()
|
|
||||||
},
|
|
||||||
|
|
||||||
read_hits_immutable: {
|
|
||||||
PAGE_CACHE_READ_HITS
|
|
||||||
.get_metric_with_label_values(&["immutable", "-"])
|
|
||||||
.unwrap()
|
|
||||||
},
|
|
||||||
|
|
||||||
read_hits_materialized_page_exact: {
|
|
||||||
PAGE_CACHE_READ_HITS
|
|
||||||
.get_metric_with_label_values(&["materialized_page", "exact"])
|
|
||||||
.unwrap()
|
|
||||||
},
|
|
||||||
|
|
||||||
read_hits_materialized_page_older_lsn: {
|
|
||||||
PAGE_CACHE_READ_HITS
|
|
||||||
.get_metric_with_label_values(&["materialized_page", "older_lsn"])
|
|
||||||
.unwrap()
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
pub struct PageCacheSizeMetrics {
|
|
||||||
pub max_bytes: UIntGauge,
|
|
||||||
|
|
||||||
pub current_bytes_ephemeral: UIntGauge,
|
|
||||||
pub current_bytes_immutable: UIntGauge,
|
|
||||||
pub current_bytes_materialized_page: UIntGauge,
|
|
||||||
}
|
|
||||||
|
|
||||||
static PAGE_CACHE_SIZE_CURRENT_BYTES: Lazy<UIntGaugeVec> = Lazy::new(|| {
|
|
||||||
register_uint_gauge_vec!(
|
|
||||||
"pageserver_page_cache_size_current_bytes",
|
|
||||||
"Current size of the page cache in bytes, by key kind",
|
|
||||||
&["key_kind"]
|
|
||||||
)
|
|
||||||
.expect("failed to define a metric")
|
|
||||||
});
|
|
||||||
|
|
||||||
pub static PAGE_CACHE_SIZE: Lazy<PageCacheSizeMetrics> = Lazy::new(|| PageCacheSizeMetrics {
|
|
||||||
max_bytes: {
|
|
||||||
register_uint_gauge!(
|
|
||||||
"pageserver_page_cache_size_max_bytes",
|
|
||||||
"Maximum size of the page cache in bytes"
|
|
||||||
)
|
|
||||||
.expect("failed to define a metric")
|
|
||||||
},
|
|
||||||
|
|
||||||
current_bytes_ephemeral: {
|
|
||||||
PAGE_CACHE_SIZE_CURRENT_BYTES
|
|
||||||
.get_metric_with_label_values(&["ephemeral"])
|
|
||||||
.unwrap()
|
|
||||||
},
|
|
||||||
current_bytes_immutable: {
|
|
||||||
PAGE_CACHE_SIZE_CURRENT_BYTES
|
|
||||||
.get_metric_with_label_values(&["immutable"])
|
|
||||||
.unwrap()
|
|
||||||
},
|
|
||||||
current_bytes_materialized_page: {
|
|
||||||
PAGE_CACHE_SIZE_CURRENT_BYTES
|
|
||||||
.get_metric_with_label_values(&["materialized_page"])
|
|
||||||
.unwrap()
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
static WAIT_LSN_TIME: Lazy<HistogramVec> = Lazy::new(|| {
|
static WAIT_LSN_TIME: Lazy<HistogramVec> = Lazy::new(|| {
|
||||||
register_histogram_vec!(
|
register_histogram_vec!(
|
||||||
"pageserver_wait_lsn_seconds",
|
"pageserver_wait_lsn_seconds",
|
||||||
@@ -320,11 +203,11 @@ pub static TENANT_STATE_METRIC: Lazy<UIntGaugeVec> = Lazy::new(|| {
|
|||||||
|
|
||||||
pub static TENANT_SYNTHETIC_SIZE_METRIC: Lazy<UIntGaugeVec> = Lazy::new(|| {
|
pub static TENANT_SYNTHETIC_SIZE_METRIC: Lazy<UIntGaugeVec> = Lazy::new(|| {
|
||||||
register_uint_gauge_vec!(
|
register_uint_gauge_vec!(
|
||||||
"pageserver_tenant_synthetic_cached_size_bytes",
|
"pageserver_tenant_synthetic_size",
|
||||||
"Synthetic size of each tenant in bytes",
|
"Synthetic size of each tenant",
|
||||||
&["tenant_id"]
|
&["tenant_id"]
|
||||||
)
|
)
|
||||||
.expect("Failed to register pageserver_tenant_synthetic_cached_size_bytes metric")
|
.expect("Failed to register pageserver_tenant_synthetic_size metric")
|
||||||
});
|
});
|
||||||
|
|
||||||
// Metrics for cloud upload. These metrics reflect data uploaded to cloud storage,
|
// Metrics for cloud upload. These metrics reflect data uploaded to cloud storage,
|
||||||
@@ -541,27 +424,6 @@ pub static SMGR_QUERY_TIME: Lazy<HistogramVec> = Lazy::new(|| {
|
|||||||
.expect("failed to define a metric")
|
.expect("failed to define a metric")
|
||||||
});
|
});
|
||||||
|
|
||||||
pub struct BasebackupQueryTime(HistogramVec);
|
|
||||||
pub static BASEBACKUP_QUERY_TIME: Lazy<BasebackupQueryTime> = Lazy::new(|| {
|
|
||||||
BasebackupQueryTime({
|
|
||||||
register_histogram_vec!(
|
|
||||||
"pageserver_basebackup_query_seconds",
|
|
||||||
"Histogram of basebackup queries durations, by result type",
|
|
||||||
&["result"],
|
|
||||||
CRITICAL_OP_BUCKETS.into(),
|
|
||||||
)
|
|
||||||
.expect("failed to define a metric")
|
|
||||||
})
|
|
||||||
});
|
|
||||||
|
|
||||||
impl DurationResultObserver for BasebackupQueryTime {
|
|
||||||
fn observe_result<T, E>(&self, res: &Result<T, E>, duration: std::time::Duration) {
|
|
||||||
let label_value = if res.is_ok() { "ok" } else { "error" };
|
|
||||||
let metric = self.0.get_metric_with_label_values(&[label_value]).unwrap();
|
|
||||||
metric.observe(duration.as_secs_f64());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub static LIVE_CONNECTIONS_COUNT: Lazy<IntGaugeVec> = Lazy::new(|| {
|
pub static LIVE_CONNECTIONS_COUNT: Lazy<IntGaugeVec> = Lazy::new(|| {
|
||||||
register_int_gauge_vec!(
|
register_int_gauge_vec!(
|
||||||
"pageserver_live_connections",
|
"pageserver_live_connections",
|
||||||
@@ -961,6 +823,11 @@ impl TimelineMetrics {
|
|||||||
let evictions_with_low_residence_duration =
|
let evictions_with_low_residence_duration =
|
||||||
evictions_with_low_residence_duration_builder.build(&tenant_id, &timeline_id);
|
evictions_with_low_residence_duration_builder.build(&tenant_id, &timeline_id);
|
||||||
|
|
||||||
|
// TODO(chi): remove this once we remove Lazy for all metrics. Otherwise this will not appear in the exporter
|
||||||
|
// and integration test will error.
|
||||||
|
MATERIALIZED_PAGE_CACHE_HIT_DIRECT.get();
|
||||||
|
MATERIALIZED_PAGE_CACHE_HIT.get();
|
||||||
|
|
||||||
TimelineMetrics {
|
TimelineMetrics {
|
||||||
tenant_id,
|
tenant_id,
|
||||||
timeline_id,
|
timeline_id,
|
||||||
@@ -1084,6 +951,7 @@ impl RemoteTimelineClientMetrics {
|
|||||||
op_kind: &RemoteOpKind,
|
op_kind: &RemoteOpKind,
|
||||||
status: &'static str,
|
status: &'static str,
|
||||||
) -> Histogram {
|
) -> Histogram {
|
||||||
|
// XXX would be nice to have an upgradable RwLock
|
||||||
let mut guard = self.remote_operation_time.lock().unwrap();
|
let mut guard = self.remote_operation_time.lock().unwrap();
|
||||||
let key = (file_kind.as_str(), op_kind.as_str(), status);
|
let key = (file_kind.as_str(), op_kind.as_str(), status);
|
||||||
let metric = guard.entry(key).or_insert_with(move || {
|
let metric = guard.entry(key).or_insert_with(move || {
|
||||||
@@ -1105,6 +973,7 @@ impl RemoteTimelineClientMetrics {
|
|||||||
file_kind: &RemoteOpFileKind,
|
file_kind: &RemoteOpFileKind,
|
||||||
op_kind: &RemoteOpKind,
|
op_kind: &RemoteOpKind,
|
||||||
) -> IntGauge {
|
) -> IntGauge {
|
||||||
|
// XXX would be nice to have an upgradable RwLock
|
||||||
let mut guard = self.calls_unfinished_gauge.lock().unwrap();
|
let mut guard = self.calls_unfinished_gauge.lock().unwrap();
|
||||||
let key = (file_kind.as_str(), op_kind.as_str());
|
let key = (file_kind.as_str(), op_kind.as_str());
|
||||||
let metric = guard.entry(key).or_insert_with(move || {
|
let metric = guard.entry(key).or_insert_with(move || {
|
||||||
@@ -1125,6 +994,7 @@ impl RemoteTimelineClientMetrics {
|
|||||||
file_kind: &RemoteOpFileKind,
|
file_kind: &RemoteOpFileKind,
|
||||||
op_kind: &RemoteOpKind,
|
op_kind: &RemoteOpKind,
|
||||||
) -> Histogram {
|
) -> Histogram {
|
||||||
|
// XXX would be nice to have an upgradable RwLock
|
||||||
let mut guard = self.calls_started_hist.lock().unwrap();
|
let mut guard = self.calls_started_hist.lock().unwrap();
|
||||||
let key = (file_kind.as_str(), op_kind.as_str());
|
let key = (file_kind.as_str(), op_kind.as_str());
|
||||||
let metric = guard.entry(key).or_insert_with(move || {
|
let metric = guard.entry(key).or_insert_with(move || {
|
||||||
@@ -1145,6 +1015,7 @@ impl RemoteTimelineClientMetrics {
|
|||||||
file_kind: &RemoteOpFileKind,
|
file_kind: &RemoteOpFileKind,
|
||||||
op_kind: &RemoteOpKind,
|
op_kind: &RemoteOpKind,
|
||||||
) -> IntCounter {
|
) -> IntCounter {
|
||||||
|
// XXX would be nice to have an upgradable RwLock
|
||||||
let mut guard = self.bytes_started_counter.lock().unwrap();
|
let mut guard = self.bytes_started_counter.lock().unwrap();
|
||||||
let key = (file_kind.as_str(), op_kind.as_str());
|
let key = (file_kind.as_str(), op_kind.as_str());
|
||||||
let metric = guard.entry(key).or_insert_with(move || {
|
let metric = guard.entry(key).or_insert_with(move || {
|
||||||
@@ -1165,6 +1036,7 @@ impl RemoteTimelineClientMetrics {
|
|||||||
file_kind: &RemoteOpFileKind,
|
file_kind: &RemoteOpFileKind,
|
||||||
op_kind: &RemoteOpKind,
|
op_kind: &RemoteOpKind,
|
||||||
) -> IntCounter {
|
) -> IntCounter {
|
||||||
|
// XXX would be nice to have an upgradable RwLock
|
||||||
let mut guard = self.bytes_finished_counter.lock().unwrap();
|
let mut guard = self.bytes_finished_counter.lock().unwrap();
|
||||||
let key = (file_kind.as_str(), op_kind.as_str());
|
let key = (file_kind.as_str(), op_kind.as_str());
|
||||||
let metric = guard.entry(key).or_insert_with(move || {
|
let metric = guard.entry(key).or_insert_with(move || {
|
||||||
@@ -1430,8 +1302,4 @@ pub fn preinitialize_metrics() {
|
|||||||
|
|
||||||
// Same as above for this metric, but, it's a Vec-type metric for which we don't know all the labels.
|
// Same as above for this metric, but, it's a Vec-type metric for which we don't know all the labels.
|
||||||
BACKGROUND_LOOP_PERIOD_OVERRUN_COUNT.reset();
|
BACKGROUND_LOOP_PERIOD_OVERRUN_COUNT.reset();
|
||||||
|
|
||||||
// Python tests need these.
|
|
||||||
MATERIALIZED_PAGE_CACHE_HIT_DIRECT.get();
|
|
||||||
MATERIALIZED_PAGE_CACHE_HIT.get();
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -53,8 +53,8 @@ use utils::{
|
|||||||
lsn::Lsn,
|
lsn::Lsn,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
use crate::repository::Key;
|
||||||
use crate::tenant::writeback_ephemeral_file;
|
use crate::tenant::writeback_ephemeral_file;
|
||||||
use crate::{metrics::PageCacheSizeMetrics, repository::Key};
|
|
||||||
|
|
||||||
static PAGE_CACHE: OnceCell<PageCache> = OnceCell::new();
|
static PAGE_CACHE: OnceCell<PageCache> = OnceCell::new();
|
||||||
const TEST_PAGE_CACHE_SIZE: usize = 50;
|
const TEST_PAGE_CACHE_SIZE: usize = 50;
|
||||||
@@ -187,8 +187,6 @@ pub struct PageCache {
|
|||||||
/// Index of the next candidate to evict, for the Clock replacement algorithm.
|
/// Index of the next candidate to evict, for the Clock replacement algorithm.
|
||||||
/// This is interpreted modulo the page cache size.
|
/// This is interpreted modulo the page cache size.
|
||||||
next_evict_slot: AtomicUsize,
|
next_evict_slot: AtomicUsize,
|
||||||
|
|
||||||
size_metrics: &'static PageCacheSizeMetrics,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
///
|
///
|
||||||
@@ -315,10 +313,6 @@ impl PageCache {
|
|||||||
key: &Key,
|
key: &Key,
|
||||||
lsn: Lsn,
|
lsn: Lsn,
|
||||||
) -> Option<(Lsn, PageReadGuard)> {
|
) -> Option<(Lsn, PageReadGuard)> {
|
||||||
crate::metrics::PAGE_CACHE
|
|
||||||
.read_accesses_materialized_page
|
|
||||||
.inc();
|
|
||||||
|
|
||||||
let mut cache_key = CacheKey::MaterializedPage {
|
let mut cache_key = CacheKey::MaterializedPage {
|
||||||
hash_key: MaterializedPageHashKey {
|
hash_key: MaterializedPageHashKey {
|
||||||
tenant_id,
|
tenant_id,
|
||||||
@@ -329,21 +323,8 @@ impl PageCache {
|
|||||||
};
|
};
|
||||||
|
|
||||||
if let Some(guard) = self.try_lock_for_read(&mut cache_key) {
|
if let Some(guard) = self.try_lock_for_read(&mut cache_key) {
|
||||||
if let CacheKey::MaterializedPage {
|
if let CacheKey::MaterializedPage { hash_key: _, lsn } = cache_key {
|
||||||
hash_key: _,
|
Some((lsn, guard))
|
||||||
lsn: available_lsn,
|
|
||||||
} = cache_key
|
|
||||||
{
|
|
||||||
if available_lsn == lsn {
|
|
||||||
crate::metrics::PAGE_CACHE
|
|
||||||
.read_hits_materialized_page_exact
|
|
||||||
.inc();
|
|
||||||
} else {
|
|
||||||
crate::metrics::PAGE_CACHE
|
|
||||||
.read_hits_materialized_page_older_lsn
|
|
||||||
.inc();
|
|
||||||
}
|
|
||||||
Some((available_lsn, guard))
|
|
||||||
} else {
|
} else {
|
||||||
panic!("unexpected key type in slot");
|
panic!("unexpected key type in slot");
|
||||||
}
|
}
|
||||||
@@ -518,31 +499,11 @@ impl PageCache {
|
|||||||
/// ```
|
/// ```
|
||||||
///
|
///
|
||||||
fn lock_for_read(&self, cache_key: &mut CacheKey) -> anyhow::Result<ReadBufResult> {
|
fn lock_for_read(&self, cache_key: &mut CacheKey) -> anyhow::Result<ReadBufResult> {
|
||||||
let (read_access, hit) = match cache_key {
|
|
||||||
CacheKey::MaterializedPage { .. } => {
|
|
||||||
unreachable!("Materialized pages use lookup_materialized_page")
|
|
||||||
}
|
|
||||||
CacheKey::EphemeralPage { .. } => (
|
|
||||||
&crate::metrics::PAGE_CACHE.read_accesses_ephemeral,
|
|
||||||
&crate::metrics::PAGE_CACHE.read_hits_ephemeral,
|
|
||||||
),
|
|
||||||
CacheKey::ImmutableFilePage { .. } => (
|
|
||||||
&crate::metrics::PAGE_CACHE.read_accesses_immutable,
|
|
||||||
&crate::metrics::PAGE_CACHE.read_hits_immutable,
|
|
||||||
),
|
|
||||||
};
|
|
||||||
read_access.inc();
|
|
||||||
|
|
||||||
let mut is_first_iteration = true;
|
|
||||||
loop {
|
loop {
|
||||||
// First check if the key already exists in the cache.
|
// First check if the key already exists in the cache.
|
||||||
if let Some(read_guard) = self.try_lock_for_read(cache_key) {
|
if let Some(read_guard) = self.try_lock_for_read(cache_key) {
|
||||||
if is_first_iteration {
|
|
||||||
hit.inc();
|
|
||||||
}
|
|
||||||
return Ok(ReadBufResult::Found(read_guard));
|
return Ok(ReadBufResult::Found(read_guard));
|
||||||
}
|
}
|
||||||
is_first_iteration = false;
|
|
||||||
|
|
||||||
// Not found. Find a victim buffer
|
// Not found. Find a victim buffer
|
||||||
let (slot_idx, mut inner) =
|
let (slot_idx, mut inner) =
|
||||||
@@ -720,9 +681,6 @@ impl PageCache {
|
|||||||
|
|
||||||
if let Ok(version_idx) = versions.binary_search_by_key(old_lsn, |v| v.lsn) {
|
if let Ok(version_idx) = versions.binary_search_by_key(old_lsn, |v| v.lsn) {
|
||||||
versions.remove(version_idx);
|
versions.remove(version_idx);
|
||||||
self.size_metrics
|
|
||||||
.current_bytes_materialized_page
|
|
||||||
.sub_page_sz(1);
|
|
||||||
if versions.is_empty() {
|
if versions.is_empty() {
|
||||||
old_entry.remove_entry();
|
old_entry.remove_entry();
|
||||||
}
|
}
|
||||||
@@ -735,13 +693,11 @@ impl PageCache {
|
|||||||
let mut map = self.ephemeral_page_map.write().unwrap();
|
let mut map = self.ephemeral_page_map.write().unwrap();
|
||||||
map.remove(&(*file_id, *blkno))
|
map.remove(&(*file_id, *blkno))
|
||||||
.expect("could not find old key in mapping");
|
.expect("could not find old key in mapping");
|
||||||
self.size_metrics.current_bytes_ephemeral.sub_page_sz(1);
|
|
||||||
}
|
}
|
||||||
CacheKey::ImmutableFilePage { file_id, blkno } => {
|
CacheKey::ImmutableFilePage { file_id, blkno } => {
|
||||||
let mut map = self.immutable_page_map.write().unwrap();
|
let mut map = self.immutable_page_map.write().unwrap();
|
||||||
map.remove(&(*file_id, *blkno))
|
map.remove(&(*file_id, *blkno))
|
||||||
.expect("could not find old key in mapping");
|
.expect("could not find old key in mapping");
|
||||||
self.size_metrics.current_bytes_immutable.sub_page_sz(1);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -769,9 +725,6 @@ impl PageCache {
|
|||||||
slot_idx,
|
slot_idx,
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
self.size_metrics
|
|
||||||
.current_bytes_materialized_page
|
|
||||||
.add_page_sz(1);
|
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -782,7 +735,6 @@ impl PageCache {
|
|||||||
Entry::Occupied(entry) => Some(*entry.get()),
|
Entry::Occupied(entry) => Some(*entry.get()),
|
||||||
Entry::Vacant(entry) => {
|
Entry::Vacant(entry) => {
|
||||||
entry.insert(slot_idx);
|
entry.insert(slot_idx);
|
||||||
self.size_metrics.current_bytes_ephemeral.add_page_sz(1);
|
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -793,7 +745,6 @@ impl PageCache {
|
|||||||
Entry::Occupied(entry) => Some(*entry.get()),
|
Entry::Occupied(entry) => Some(*entry.get()),
|
||||||
Entry::Vacant(entry) => {
|
Entry::Vacant(entry) => {
|
||||||
entry.insert(slot_idx);
|
entry.insert(slot_idx);
|
||||||
self.size_metrics.current_bytes_immutable.add_page_sz(1);
|
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -893,12 +844,6 @@ impl PageCache {
|
|||||||
|
|
||||||
let page_buffer = Box::leak(vec![0u8; num_pages * PAGE_SZ].into_boxed_slice());
|
let page_buffer = Box::leak(vec![0u8; num_pages * PAGE_SZ].into_boxed_slice());
|
||||||
|
|
||||||
let size_metrics = &crate::metrics::PAGE_CACHE_SIZE;
|
|
||||||
size_metrics.max_bytes.set_page_sz(num_pages);
|
|
||||||
size_metrics.current_bytes_ephemeral.set_page_sz(0);
|
|
||||||
size_metrics.current_bytes_immutable.set_page_sz(0);
|
|
||||||
size_metrics.current_bytes_materialized_page.set_page_sz(0);
|
|
||||||
|
|
||||||
let slots = page_buffer
|
let slots = page_buffer
|
||||||
.chunks_exact_mut(PAGE_SZ)
|
.chunks_exact_mut(PAGE_SZ)
|
||||||
.map(|chunk| {
|
.map(|chunk| {
|
||||||
@@ -921,30 +866,6 @@ impl PageCache {
|
|||||||
immutable_page_map: Default::default(),
|
immutable_page_map: Default::default(),
|
||||||
slots,
|
slots,
|
||||||
next_evict_slot: AtomicUsize::new(0),
|
next_evict_slot: AtomicUsize::new(0),
|
||||||
size_metrics,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
trait PageSzBytesMetric {
|
|
||||||
fn set_page_sz(&self, count: usize);
|
|
||||||
fn add_page_sz(&self, count: usize);
|
|
||||||
fn sub_page_sz(&self, count: usize);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline(always)]
|
|
||||||
fn count_times_page_sz(count: usize) -> u64 {
|
|
||||||
u64::try_from(count).unwrap() * u64::try_from(PAGE_SZ).unwrap()
|
|
||||||
}
|
|
||||||
|
|
||||||
impl PageSzBytesMetric for metrics::UIntGauge {
|
|
||||||
fn set_page_sz(&self, count: usize) {
|
|
||||||
self.set(count_times_page_sz(count));
|
|
||||||
}
|
|
||||||
fn add_page_sz(&self, count: usize) {
|
|
||||||
self.add(count_times_page_sz(count));
|
|
||||||
}
|
|
||||||
fn sub_page_sz(&self, count: usize) {
|
|
||||||
self.sub(count_times_page_sz(count));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -904,7 +904,7 @@ where
|
|||||||
|
|
||||||
self.check_permission(Some(tenant_id))?;
|
self.check_permission(Some(tenant_id))?;
|
||||||
|
|
||||||
let lsn = if params.len() >= 3 {
|
let lsn = if params.len() == 3 {
|
||||||
Some(
|
Some(
|
||||||
Lsn::from_str(params[2])
|
Lsn::from_str(params[2])
|
||||||
.with_context(|| format!("Failed to parse Lsn from {}", params[2]))?,
|
.with_context(|| format!("Failed to parse Lsn from {}", params[2]))?,
|
||||||
@@ -913,24 +913,10 @@ where
|
|||||||
None
|
None
|
||||||
};
|
};
|
||||||
|
|
||||||
metrics::metric_vec_duration::observe_async_block_duration_by_result(
|
// Check that the timeline exists
|
||||||
&*crate::metrics::BASEBACKUP_QUERY_TIME,
|
self.handle_basebackup_request(pgb, tenant_id, timeline_id, lsn, None, false, ctx)
|
||||||
async move {
|
.await?;
|
||||||
self.handle_basebackup_request(
|
pgb.write_message_noflush(&BeMessage::CommandComplete(b"SELECT 1"))?;
|
||||||
pgb,
|
|
||||||
tenant_id,
|
|
||||||
timeline_id,
|
|
||||||
lsn,
|
|
||||||
None,
|
|
||||||
false,
|
|
||||||
ctx,
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
pgb.write_message_noflush(&BeMessage::CommandComplete(b"SELECT 1"))?;
|
|
||||||
anyhow::Ok(())
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
}
|
}
|
||||||
// return pair of prev_lsn and last_lsn
|
// return pair of prev_lsn and last_lsn
|
||||||
else if query_string.starts_with("get_last_record_rlsn ") {
|
else if query_string.starts_with("get_last_record_rlsn ") {
|
||||||
|
|||||||
@@ -887,7 +887,7 @@ impl<'a> DatadirModification<'a> {
|
|||||||
ctx: &RequestContext,
|
ctx: &RequestContext,
|
||||||
) -> Result<(), RelationError> {
|
) -> Result<(), RelationError> {
|
||||||
if rel.relnode == 0 {
|
if rel.relnode == 0 {
|
||||||
return Err(RelationError::InvalidRelnode);
|
return Err(RelationError::AlreadyExists);
|
||||||
}
|
}
|
||||||
// It's possible that this is the first rel for this db in this
|
// It's possible that this is the first rel for this db in this
|
||||||
// tablespace. Create the reldir entry for it if so.
|
// tablespace. Create the reldir entry for it if so.
|
||||||
|
|||||||
@@ -506,17 +506,17 @@ pub async fn shutdown_tasks(
|
|||||||
warn!(name = task.name, tenant_id = ?tenant_id, timeline_id = ?timeline_id, kind = ?task_kind, "stopping left-over");
|
warn!(name = task.name, tenant_id = ?tenant_id, timeline_id = ?timeline_id, kind = ?task_kind, "stopping left-over");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
let join_handle = tokio::select! {
|
let completed = tokio::select! {
|
||||||
biased;
|
biased;
|
||||||
_ = &mut join_handle => { None },
|
_ = &mut join_handle => { true },
|
||||||
_ = tokio::time::sleep(std::time::Duration::from_secs(1)) => {
|
_ = tokio::time::sleep(std::time::Duration::from_secs(1)) => {
|
||||||
// allow some time to elapse before logging to cut down the number of log
|
// allow some time to elapse before logging to cut down the number of log
|
||||||
// lines.
|
// lines.
|
||||||
info!("waiting for {} to shut down", task.name);
|
info!("waiting for {} to shut down", task.name);
|
||||||
Some(join_handle)
|
false
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
if let Some(join_handle) = join_handle {
|
if !completed {
|
||||||
// we never handled this return value, but:
|
// we never handled this return value, but:
|
||||||
// - we don't deschedule which would lead to is_cancelled
|
// - we don't deschedule which would lead to is_cancelled
|
||||||
// - panics are already logged (is_panicked)
|
// - panics are already logged (is_panicked)
|
||||||
|
|||||||
@@ -440,13 +440,8 @@ pub enum GetTimelineError {
|
|||||||
pub enum DeleteTimelineError {
|
pub enum DeleteTimelineError {
|
||||||
#[error("NotFound")]
|
#[error("NotFound")]
|
||||||
NotFound,
|
NotFound,
|
||||||
|
|
||||||
#[error("HasChildren")]
|
#[error("HasChildren")]
|
||||||
HasChildren(Vec<TimelineId>),
|
HasChildren(Vec<TimelineId>),
|
||||||
|
|
||||||
#[error("Timeline deletion is already in progress")]
|
|
||||||
AlreadyInProgress,
|
|
||||||
|
|
||||||
#[error(transparent)]
|
#[error(transparent)]
|
||||||
Other(#[from] anyhow::Error),
|
Other(#[from] anyhow::Error),
|
||||||
}
|
}
|
||||||
@@ -501,16 +496,6 @@ impl DeletionGuard {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(thiserror::Error, Debug)]
|
|
||||||
pub enum CreateTimelineError {
|
|
||||||
#[error("a timeline with the given ID already exists")]
|
|
||||||
AlreadyExists,
|
|
||||||
#[error(transparent)]
|
|
||||||
AncestorLsn(anyhow::Error),
|
|
||||||
#[error(transparent)]
|
|
||||||
Other(#[from] anyhow::Error),
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Tenant {
|
impl Tenant {
|
||||||
/// Yet another helper for timeline initialization.
|
/// Yet another helper for timeline initialization.
|
||||||
/// Contains the common part of `load_local_timeline` and `load_remote_timeline`.
|
/// Contains the common part of `load_local_timeline` and `load_remote_timeline`.
|
||||||
@@ -600,7 +585,6 @@ impl Tenant {
|
|||||||
.layers
|
.layers
|
||||||
.read()
|
.read()
|
||||||
.await
|
.await
|
||||||
.0
|
|
||||||
.iter_historic_layers()
|
.iter_historic_layers()
|
||||||
.next()
|
.next()
|
||||||
.is_some(),
|
.is_some(),
|
||||||
@@ -1385,7 +1369,8 @@ impl Tenant {
|
|||||||
/// Returns the new timeline ID and reference to its Timeline object.
|
/// Returns the new timeline ID and reference to its Timeline object.
|
||||||
///
|
///
|
||||||
/// If the caller specified the timeline ID to use (`new_timeline_id`), and timeline with
|
/// If the caller specified the timeline ID to use (`new_timeline_id`), and timeline with
|
||||||
/// the same timeline ID already exists, returns CreateTimelineError::AlreadyExists.
|
/// the same timeline ID already exists, returns None. If `new_timeline_id` is not given,
|
||||||
|
/// a new unique ID is generated.
|
||||||
pub async fn create_timeline(
|
pub async fn create_timeline(
|
||||||
&self,
|
&self,
|
||||||
new_timeline_id: TimelineId,
|
new_timeline_id: TimelineId,
|
||||||
@@ -1394,12 +1379,11 @@ impl Tenant {
|
|||||||
pg_version: u32,
|
pg_version: u32,
|
||||||
broker_client: storage_broker::BrokerClientChannel,
|
broker_client: storage_broker::BrokerClientChannel,
|
||||||
ctx: &RequestContext,
|
ctx: &RequestContext,
|
||||||
) -> Result<Arc<Timeline>, CreateTimelineError> {
|
) -> anyhow::Result<Option<Arc<Timeline>>> {
|
||||||
if !self.is_active() {
|
anyhow::ensure!(
|
||||||
return Err(CreateTimelineError::Other(anyhow::anyhow!(
|
self.is_active(),
|
||||||
"Cannot create timelines on inactive tenant"
|
"Cannot create timelines on inactive tenant"
|
||||||
)));
|
);
|
||||||
}
|
|
||||||
|
|
||||||
if let Ok(existing) = self.get_timeline(new_timeline_id, false) {
|
if let Ok(existing) = self.get_timeline(new_timeline_id, false) {
|
||||||
debug!("timeline {new_timeline_id} already exists");
|
debug!("timeline {new_timeline_id} already exists");
|
||||||
@@ -1419,7 +1403,7 @@ impl Tenant {
|
|||||||
.context("wait for timeline uploads to complete")?;
|
.context("wait for timeline uploads to complete")?;
|
||||||
}
|
}
|
||||||
|
|
||||||
return Err(CreateTimelineError::AlreadyExists);
|
return Ok(None);
|
||||||
}
|
}
|
||||||
|
|
||||||
let loaded_timeline = match ancestor_timeline_id {
|
let loaded_timeline = match ancestor_timeline_id {
|
||||||
@@ -1434,12 +1418,12 @@ impl Tenant {
|
|||||||
let ancestor_ancestor_lsn = ancestor_timeline.get_ancestor_lsn();
|
let ancestor_ancestor_lsn = ancestor_timeline.get_ancestor_lsn();
|
||||||
if ancestor_ancestor_lsn > *lsn {
|
if ancestor_ancestor_lsn > *lsn {
|
||||||
// can we safely just branch from the ancestor instead?
|
// can we safely just branch from the ancestor instead?
|
||||||
return Err(CreateTimelineError::AncestorLsn(anyhow::anyhow!(
|
bail!(
|
||||||
"invalid start lsn {} for ancestor timeline {}: less than timeline ancestor lsn {}",
|
"invalid start lsn {} for ancestor timeline {}: less than timeline ancestor lsn {}",
|
||||||
lsn,
|
lsn,
|
||||||
ancestor_timeline_id,
|
ancestor_timeline_id,
|
||||||
ancestor_ancestor_lsn,
|
ancestor_ancestor_lsn,
|
||||||
)));
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Wait for the WAL to arrive and be processed on the parent branch up
|
// Wait for the WAL to arrive and be processed on the parent branch up
|
||||||
@@ -1473,7 +1457,7 @@ impl Tenant {
|
|||||||
})?;
|
})?;
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(loaded_timeline)
|
Ok(Some(loaded_timeline))
|
||||||
}
|
}
|
||||||
|
|
||||||
/// perform one garbage collection iteration, removing old data files from disk.
|
/// perform one garbage collection iteration, removing old data files from disk.
|
||||||
@@ -1771,11 +1755,14 @@ impl Tenant {
|
|||||||
timeline = Arc::clone(timeline_entry.get());
|
timeline = Arc::clone(timeline_entry.get());
|
||||||
|
|
||||||
// Prevent two tasks from trying to delete the timeline at the same time.
|
// Prevent two tasks from trying to delete the timeline at the same time.
|
||||||
delete_lock_guard = DeletionGuard(
|
delete_lock_guard =
|
||||||
Arc::clone(&timeline.delete_lock)
|
DeletionGuard(Arc::clone(&timeline.delete_lock).try_lock_owned().map_err(
|
||||||
.try_lock_owned()
|
|_| {
|
||||||
.map_err(|_| DeleteTimelineError::AlreadyInProgress)?,
|
DeleteTimelineError::Other(anyhow::anyhow!(
|
||||||
);
|
"timeline deletion is already in progress"
|
||||||
|
))
|
||||||
|
},
|
||||||
|
)?);
|
||||||
|
|
||||||
// If another task finished the deletion just before we acquired the lock,
|
// If another task finished the deletion just before we acquired the lock,
|
||||||
// return success.
|
// return success.
|
||||||
@@ -2717,7 +2704,7 @@ impl Tenant {
|
|||||||
dst_id: TimelineId,
|
dst_id: TimelineId,
|
||||||
start_lsn: Option<Lsn>,
|
start_lsn: Option<Lsn>,
|
||||||
ctx: &RequestContext,
|
ctx: &RequestContext,
|
||||||
) -> Result<Arc<Timeline>, CreateTimelineError> {
|
) -> anyhow::Result<Arc<Timeline>> {
|
||||||
let tl = self
|
let tl = self
|
||||||
.branch_timeline_impl(src_timeline, dst_id, start_lsn, ctx)
|
.branch_timeline_impl(src_timeline, dst_id, start_lsn, ctx)
|
||||||
.await?;
|
.await?;
|
||||||
@@ -2734,7 +2721,7 @@ impl Tenant {
|
|||||||
dst_id: TimelineId,
|
dst_id: TimelineId,
|
||||||
start_lsn: Option<Lsn>,
|
start_lsn: Option<Lsn>,
|
||||||
ctx: &RequestContext,
|
ctx: &RequestContext,
|
||||||
) -> Result<Arc<Timeline>, CreateTimelineError> {
|
) -> anyhow::Result<Arc<Timeline>> {
|
||||||
self.branch_timeline_impl(src_timeline, dst_id, start_lsn, ctx)
|
self.branch_timeline_impl(src_timeline, dst_id, start_lsn, ctx)
|
||||||
.await
|
.await
|
||||||
}
|
}
|
||||||
@@ -2745,7 +2732,7 @@ impl Tenant {
|
|||||||
dst_id: TimelineId,
|
dst_id: TimelineId,
|
||||||
start_lsn: Option<Lsn>,
|
start_lsn: Option<Lsn>,
|
||||||
_ctx: &RequestContext,
|
_ctx: &RequestContext,
|
||||||
) -> Result<Arc<Timeline>, CreateTimelineError> {
|
) -> anyhow::Result<Arc<Timeline>> {
|
||||||
let src_id = src_timeline.timeline_id;
|
let src_id = src_timeline.timeline_id;
|
||||||
|
|
||||||
// If no start LSN is specified, we branch the new timeline from the source timeline's last record LSN
|
// If no start LSN is specified, we branch the new timeline from the source timeline's last record LSN
|
||||||
@@ -2785,17 +2772,16 @@ impl Tenant {
|
|||||||
.context(format!(
|
.context(format!(
|
||||||
"invalid branch start lsn: less than latest GC cutoff {}",
|
"invalid branch start lsn: less than latest GC cutoff {}",
|
||||||
*latest_gc_cutoff_lsn,
|
*latest_gc_cutoff_lsn,
|
||||||
))
|
))?;
|
||||||
.map_err(CreateTimelineError::AncestorLsn)?;
|
|
||||||
|
|
||||||
// and then the planned GC cutoff
|
// and then the planned GC cutoff
|
||||||
{
|
{
|
||||||
let gc_info = src_timeline.gc_info.read().unwrap();
|
let gc_info = src_timeline.gc_info.read().unwrap();
|
||||||
let cutoff = min(gc_info.pitr_cutoff, gc_info.horizon_cutoff);
|
let cutoff = min(gc_info.pitr_cutoff, gc_info.horizon_cutoff);
|
||||||
if start_lsn < cutoff {
|
if start_lsn < cutoff {
|
||||||
return Err(CreateTimelineError::AncestorLsn(anyhow::anyhow!(
|
bail!(format!(
|
||||||
"invalid branch start lsn: less than planned GC cutoff {cutoff}"
|
"invalid branch start lsn: less than planned GC cutoff {cutoff}"
|
||||||
)));
|
));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -3828,9 +3814,6 @@ mod tests {
|
|||||||
{
|
{
|
||||||
Ok(_) => panic!("branching should have failed"),
|
Ok(_) => panic!("branching should have failed"),
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
let CreateTimelineError::AncestorLsn(err) = err else {
|
|
||||||
panic!("wrong error type")
|
|
||||||
};
|
|
||||||
assert!(err.to_string().contains("invalid branch start lsn"));
|
assert!(err.to_string().contains("invalid branch start lsn"));
|
||||||
assert!(err
|
assert!(err
|
||||||
.source()
|
.source()
|
||||||
@@ -3860,9 +3843,6 @@ mod tests {
|
|||||||
{
|
{
|
||||||
Ok(_) => panic!("branching should have failed"),
|
Ok(_) => panic!("branching should have failed"),
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
let CreateTimelineError::AncestorLsn(err) = err else {
|
|
||||||
panic!("wrong error type");
|
|
||||||
};
|
|
||||||
assert!(&err.to_string().contains("invalid branch start lsn"));
|
assert!(&err.to_string().contains("invalid branch start lsn"));
|
||||||
assert!(&err
|
assert!(&err
|
||||||
.source()
|
.source()
|
||||||
|
|||||||
@@ -51,23 +51,25 @@ use crate::keyspace::KeyPartitioning;
|
|||||||
use crate::repository::Key;
|
use crate::repository::Key;
|
||||||
use crate::tenant::storage_layer::InMemoryLayer;
|
use crate::tenant::storage_layer::InMemoryLayer;
|
||||||
use crate::tenant::storage_layer::Layer;
|
use crate::tenant::storage_layer::Layer;
|
||||||
|
use anyhow::Context;
|
||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
|
use std::collections::HashMap;
|
||||||
use std::collections::VecDeque;
|
use std::collections::VecDeque;
|
||||||
use std::ops::Range;
|
use std::ops::Range;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use utils::lsn::Lsn;
|
use utils::lsn::Lsn;
|
||||||
|
|
||||||
use historic_layer_coverage::BufferedHistoricLayerCoverage;
|
use historic_layer_coverage::BufferedHistoricLayerCoverage;
|
||||||
pub use historic_layer_coverage::LayerKey;
|
pub use historic_layer_coverage::Replacement;
|
||||||
|
|
||||||
use super::storage_layer::range_eq;
|
use super::storage_layer::range_eq;
|
||||||
use super::storage_layer::PersistentLayerDesc;
|
use super::storage_layer::PersistentLayerDesc;
|
||||||
|
use super::storage_layer::PersistentLayerKey;
|
||||||
|
|
||||||
///
|
///
|
||||||
/// LayerMap tracks what layers exist on a timeline.
|
/// LayerMap tracks what layers exist on a timeline.
|
||||||
///
|
///
|
||||||
#[derive(Default)]
|
pub struct LayerMap<L: ?Sized> {
|
||||||
pub struct LayerMap {
|
|
||||||
//
|
//
|
||||||
// 'open_layer' holds the current InMemoryLayer that is accepting new
|
// 'open_layer' holds the current InMemoryLayer that is accepting new
|
||||||
// records. If it is None, 'next_open_layer_at' will be set instead, indicating
|
// records. If it is None, 'next_open_layer_at' will be set instead, indicating
|
||||||
@@ -93,6 +95,24 @@ pub struct LayerMap {
|
|||||||
/// L0 layers have key range Key::MIN..Key::MAX, and locating them using R-Tree search is very inefficient.
|
/// L0 layers have key range Key::MIN..Key::MAX, and locating them using R-Tree search is very inefficient.
|
||||||
/// So L0 layers are held in l0_delta_layers vector, in addition to the R-tree.
|
/// So L0 layers are held in l0_delta_layers vector, in addition to the R-tree.
|
||||||
l0_delta_layers: Vec<Arc<PersistentLayerDesc>>,
|
l0_delta_layers: Vec<Arc<PersistentLayerDesc>>,
|
||||||
|
|
||||||
|
/// Mapping from persistent layer key to the actual layer object. Currently, it stores delta, image, and
|
||||||
|
/// remote layers. In future refactors, this will be eventually moved out of LayerMap into Timeline, and
|
||||||
|
/// RemoteLayer will be removed.
|
||||||
|
mapping: HashMap<PersistentLayerKey, Arc<L>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<L: ?Sized> Default for LayerMap<L> {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self {
|
||||||
|
open_layer: None,
|
||||||
|
next_open_layer_at: None,
|
||||||
|
frozen_layers: VecDeque::default(),
|
||||||
|
l0_delta_layers: Vec::default(),
|
||||||
|
historic: BufferedHistoricLayerCoverage::default(),
|
||||||
|
mapping: HashMap::default(),
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// The primary update API for the layer map.
|
/// The primary update API for the layer map.
|
||||||
@@ -100,21 +120,24 @@ pub struct LayerMap {
|
|||||||
/// Batching historic layer insertions and removals is good for
|
/// Batching historic layer insertions and removals is good for
|
||||||
/// performance and this struct helps us do that correctly.
|
/// performance and this struct helps us do that correctly.
|
||||||
#[must_use]
|
#[must_use]
|
||||||
pub struct BatchedUpdates<'a> {
|
pub struct BatchedUpdates<'a, L: ?Sized + Layer> {
|
||||||
// While we hold this exclusive reference to the layer map the type checker
|
// While we hold this exclusive reference to the layer map the type checker
|
||||||
// will prevent us from accidentally reading any unflushed updates.
|
// will prevent us from accidentally reading any unflushed updates.
|
||||||
layer_map: &'a mut LayerMap,
|
layer_map: &'a mut LayerMap<L>,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Provide ability to batch more updates while hiding the read
|
/// Provide ability to batch more updates while hiding the read
|
||||||
/// API so we don't accidentally read without flushing.
|
/// API so we don't accidentally read without flushing.
|
||||||
impl BatchedUpdates<'_> {
|
impl<L> BatchedUpdates<'_, L>
|
||||||
|
where
|
||||||
|
L: ?Sized + Layer,
|
||||||
|
{
|
||||||
///
|
///
|
||||||
/// Insert an on-disk layer.
|
/// Insert an on-disk layer.
|
||||||
///
|
///
|
||||||
// TODO remove the `layer` argument when `mapping` is refactored out of `LayerMap`
|
// TODO remove the `layer` argument when `mapping` is refactored out of `LayerMap`
|
||||||
pub fn insert_historic(&mut self, layer_desc: PersistentLayerDesc) {
|
pub fn insert_historic(&mut self, layer_desc: PersistentLayerDesc, layer: Arc<L>) {
|
||||||
self.layer_map.insert_historic_noflush(layer_desc)
|
self.layer_map.insert_historic_noflush(layer_desc, layer)
|
||||||
}
|
}
|
||||||
|
|
||||||
///
|
///
|
||||||
@@ -122,8 +145,31 @@ impl BatchedUpdates<'_> {
|
|||||||
///
|
///
|
||||||
/// This should be called when the corresponding file on disk has been deleted.
|
/// This should be called when the corresponding file on disk has been deleted.
|
||||||
///
|
///
|
||||||
pub fn remove_historic(&mut self, layer_desc: PersistentLayerDesc) {
|
pub fn remove_historic(&mut self, layer_desc: PersistentLayerDesc, layer: Arc<L>) {
|
||||||
self.layer_map.remove_historic_noflush(layer_desc)
|
self.layer_map.remove_historic_noflush(layer_desc, layer)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Replaces existing layer iff it is the `expected`.
|
||||||
|
///
|
||||||
|
/// If the expected layer has been removed it will not be inserted by this function.
|
||||||
|
///
|
||||||
|
/// Returned `Replacement` describes succeeding in replacement or the reason why it could not
|
||||||
|
/// be done.
|
||||||
|
///
|
||||||
|
/// TODO replacement can be done without buffering and rebuilding layer map updates.
|
||||||
|
/// One way to do that is to add a layer of indirection for returned values, so
|
||||||
|
/// that we can replace values only by updating a hashmap.
|
||||||
|
pub fn replace_historic(
|
||||||
|
&mut self,
|
||||||
|
expected_desc: PersistentLayerDesc,
|
||||||
|
expected: &Arc<L>,
|
||||||
|
new_desc: PersistentLayerDesc,
|
||||||
|
new: Arc<L>,
|
||||||
|
) -> anyhow::Result<Replacement<Arc<L>>> {
|
||||||
|
fail::fail_point!("layermap-replace-notfound", |_| Ok(Replacement::NotFound));
|
||||||
|
|
||||||
|
self.layer_map
|
||||||
|
.replace_historic_noflush(expected_desc, expected, new_desc, new)
|
||||||
}
|
}
|
||||||
|
|
||||||
// We will flush on drop anyway, but this method makes it
|
// We will flush on drop anyway, but this method makes it
|
||||||
@@ -139,19 +185,25 @@ impl BatchedUpdates<'_> {
|
|||||||
// than panic later or read without flushing.
|
// than panic later or read without flushing.
|
||||||
//
|
//
|
||||||
// TODO maybe warn if flush hasn't explicitly been called
|
// TODO maybe warn if flush hasn't explicitly been called
|
||||||
impl Drop for BatchedUpdates<'_> {
|
impl<L> Drop for BatchedUpdates<'_, L>
|
||||||
|
where
|
||||||
|
L: ?Sized + Layer,
|
||||||
|
{
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
self.layer_map.flush_updates();
|
self.layer_map.flush_updates();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Return value of LayerMap::search
|
/// Return value of LayerMap::search
|
||||||
pub struct SearchResult {
|
pub struct SearchResult<L: ?Sized> {
|
||||||
pub layer: Arc<PersistentLayerDesc>,
|
pub layer: Arc<L>,
|
||||||
pub lsn_floor: Lsn,
|
pub lsn_floor: Lsn,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl LayerMap {
|
impl<L> LayerMap<L>
|
||||||
|
where
|
||||||
|
L: ?Sized + Layer,
|
||||||
|
{
|
||||||
///
|
///
|
||||||
/// Find the latest layer (by lsn.end) that covers the given
|
/// Find the latest layer (by lsn.end) that covers the given
|
||||||
/// 'key', with lsn.start < 'end_lsn'.
|
/// 'key', with lsn.start < 'end_lsn'.
|
||||||
@@ -183,7 +235,7 @@ impl LayerMap {
|
|||||||
/// NOTE: This only searches the 'historic' layers, *not* the
|
/// NOTE: This only searches the 'historic' layers, *not* the
|
||||||
/// 'open' and 'frozen' layers!
|
/// 'open' and 'frozen' layers!
|
||||||
///
|
///
|
||||||
pub fn search(&self, key: Key, end_lsn: Lsn) -> Option<SearchResult> {
|
pub fn search(&self, key: Key, end_lsn: Lsn) -> Option<SearchResult<L>> {
|
||||||
let version = self.historic.get().unwrap().get_version(end_lsn.0 - 1)?;
|
let version = self.historic.get().unwrap().get_version(end_lsn.0 - 1)?;
|
||||||
let latest_delta = version.delta_coverage.query(key.to_i128());
|
let latest_delta = version.delta_coverage.query(key.to_i128());
|
||||||
let latest_image = version.image_coverage.query(key.to_i128());
|
let latest_image = version.image_coverage.query(key.to_i128());
|
||||||
@@ -192,6 +244,7 @@ impl LayerMap {
|
|||||||
(None, None) => None,
|
(None, None) => None,
|
||||||
(None, Some(image)) => {
|
(None, Some(image)) => {
|
||||||
let lsn_floor = image.get_lsn_range().start;
|
let lsn_floor = image.get_lsn_range().start;
|
||||||
|
let image = self.get_layer_from_mapping(&image.key()).clone();
|
||||||
Some(SearchResult {
|
Some(SearchResult {
|
||||||
layer: image,
|
layer: image,
|
||||||
lsn_floor,
|
lsn_floor,
|
||||||
@@ -199,6 +252,7 @@ impl LayerMap {
|
|||||||
}
|
}
|
||||||
(Some(delta), None) => {
|
(Some(delta), None) => {
|
||||||
let lsn_floor = delta.get_lsn_range().start;
|
let lsn_floor = delta.get_lsn_range().start;
|
||||||
|
let delta = self.get_layer_from_mapping(&delta.key()).clone();
|
||||||
Some(SearchResult {
|
Some(SearchResult {
|
||||||
layer: delta,
|
layer: delta,
|
||||||
lsn_floor,
|
lsn_floor,
|
||||||
@@ -209,6 +263,7 @@ impl LayerMap {
|
|||||||
let image_is_newer = image.get_lsn_range().end >= delta.get_lsn_range().end;
|
let image_is_newer = image.get_lsn_range().end >= delta.get_lsn_range().end;
|
||||||
let image_exact_match = img_lsn + 1 == end_lsn;
|
let image_exact_match = img_lsn + 1 == end_lsn;
|
||||||
if image_is_newer || image_exact_match {
|
if image_is_newer || image_exact_match {
|
||||||
|
let image = self.get_layer_from_mapping(&image.key()).clone();
|
||||||
Some(SearchResult {
|
Some(SearchResult {
|
||||||
layer: image,
|
layer: image,
|
||||||
lsn_floor: img_lsn,
|
lsn_floor: img_lsn,
|
||||||
@@ -216,6 +271,7 @@ impl LayerMap {
|
|||||||
} else {
|
} else {
|
||||||
let lsn_floor =
|
let lsn_floor =
|
||||||
std::cmp::max(delta.get_lsn_range().start, image.get_lsn_range().start + 1);
|
std::cmp::max(delta.get_lsn_range().start, image.get_lsn_range().start + 1);
|
||||||
|
let delta = self.get_layer_from_mapping(&delta.key()).clone();
|
||||||
Some(SearchResult {
|
Some(SearchResult {
|
||||||
layer: delta,
|
layer: delta,
|
||||||
lsn_floor,
|
lsn_floor,
|
||||||
@@ -226,7 +282,7 @@ impl LayerMap {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Start a batch of updates, applied on drop
|
/// Start a batch of updates, applied on drop
|
||||||
pub fn batch_update(&mut self) -> BatchedUpdates<'_> {
|
pub fn batch_update(&mut self) -> BatchedUpdates<'_, L> {
|
||||||
BatchedUpdates { layer_map: self }
|
BatchedUpdates { layer_map: self }
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -236,32 +292,48 @@ impl LayerMap {
|
|||||||
/// Helper function for BatchedUpdates::insert_historic
|
/// Helper function for BatchedUpdates::insert_historic
|
||||||
///
|
///
|
||||||
/// TODO(chi): remove L generic so that we do not need to pass layer object.
|
/// TODO(chi): remove L generic so that we do not need to pass layer object.
|
||||||
pub(self) fn insert_historic_noflush(&mut self, layer_desc: PersistentLayerDesc) {
|
pub(self) fn insert_historic_noflush(
|
||||||
|
&mut self,
|
||||||
|
layer_desc: PersistentLayerDesc,
|
||||||
|
layer: Arc<L>,
|
||||||
|
) {
|
||||||
|
self.mapping.insert(layer_desc.key(), layer.clone());
|
||||||
|
|
||||||
// TODO: See #3869, resulting #4088, attempted fix and repro #4094
|
// TODO: See #3869, resulting #4088, attempted fix and repro #4094
|
||||||
|
|
||||||
if Self::is_l0(&layer_desc) {
|
if Self::is_l0(&layer) {
|
||||||
self.l0_delta_layers.push(layer_desc.clone().into());
|
self.l0_delta_layers.push(layer_desc.clone().into());
|
||||||
}
|
}
|
||||||
|
|
||||||
self.historic.insert(
|
self.historic.insert(
|
||||||
historic_layer_coverage::LayerKey::from(&layer_desc),
|
historic_layer_coverage::LayerKey::from(&*layer),
|
||||||
layer_desc.into(),
|
layer_desc.into(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn get_layer_from_mapping(&self, key: &PersistentLayerKey) -> &Arc<L> {
|
||||||
|
let layer = self
|
||||||
|
.mapping
|
||||||
|
.get(key)
|
||||||
|
.with_context(|| format!("{key:?}"))
|
||||||
|
.expect("inconsistent layer mapping");
|
||||||
|
layer
|
||||||
|
}
|
||||||
|
|
||||||
///
|
///
|
||||||
/// Remove an on-disk layer from the map.
|
/// Remove an on-disk layer from the map.
|
||||||
///
|
///
|
||||||
/// Helper function for BatchedUpdates::remove_historic
|
/// Helper function for BatchedUpdates::remove_historic
|
||||||
///
|
///
|
||||||
pub fn remove_historic_noflush(&mut self, layer_desc: PersistentLayerDesc) {
|
pub fn remove_historic_noflush(&mut self, layer_desc: PersistentLayerDesc, layer: Arc<L>) {
|
||||||
self.historic
|
self.historic
|
||||||
.remove(historic_layer_coverage::LayerKey::from(&layer_desc));
|
.remove(historic_layer_coverage::LayerKey::from(&*layer));
|
||||||
let layer_key = layer_desc.key();
|
if Self::is_l0(&layer) {
|
||||||
if Self::is_l0(&layer_desc) {
|
|
||||||
let len_before = self.l0_delta_layers.len();
|
let len_before = self.l0_delta_layers.len();
|
||||||
let mut l0_delta_layers = std::mem::take(&mut self.l0_delta_layers);
|
let mut l0_delta_layers = std::mem::take(&mut self.l0_delta_layers);
|
||||||
l0_delta_layers.retain(|other| other.key() != layer_key);
|
l0_delta_layers.retain(|other| {
|
||||||
|
!Self::compare_arced_layers(self.get_layer_from_mapping(&other.key()), &layer)
|
||||||
|
});
|
||||||
self.l0_delta_layers = l0_delta_layers;
|
self.l0_delta_layers = l0_delta_layers;
|
||||||
// this assertion is related to use of Arc::ptr_eq in Self::compare_arced_layers,
|
// this assertion is related to use of Arc::ptr_eq in Self::compare_arced_layers,
|
||||||
// there's a chance that the comparison fails at runtime due to it comparing (pointer,
|
// there's a chance that the comparison fails at runtime due to it comparing (pointer,
|
||||||
@@ -272,6 +344,69 @@ impl LayerMap {
|
|||||||
"failed to locate removed historic layer from l0_delta_layers"
|
"failed to locate removed historic layer from l0_delta_layers"
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
self.mapping.remove(&layer_desc.key());
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(self) fn replace_historic_noflush(
|
||||||
|
&mut self,
|
||||||
|
expected_desc: PersistentLayerDesc,
|
||||||
|
expected: &Arc<L>,
|
||||||
|
new_desc: PersistentLayerDesc,
|
||||||
|
new: Arc<L>,
|
||||||
|
) -> anyhow::Result<Replacement<Arc<L>>> {
|
||||||
|
let key = historic_layer_coverage::LayerKey::from(&**expected);
|
||||||
|
let other = historic_layer_coverage::LayerKey::from(&*new);
|
||||||
|
|
||||||
|
let expected_l0 = Self::is_l0(expected);
|
||||||
|
let new_l0 = Self::is_l0(&new);
|
||||||
|
|
||||||
|
anyhow::ensure!(
|
||||||
|
key == other,
|
||||||
|
"expected and new must have equal LayerKeys: {key:?} != {other:?}"
|
||||||
|
);
|
||||||
|
|
||||||
|
anyhow::ensure!(
|
||||||
|
expected_l0 == new_l0,
|
||||||
|
"expected and new must both be l0 deltas or neither should be: {expected_l0} != {new_l0}"
|
||||||
|
);
|
||||||
|
|
||||||
|
let l0_index = if expected_l0 {
|
||||||
|
// find the index in case replace worked, we need to replace that as well
|
||||||
|
let pos = self.l0_delta_layers.iter().position(|slot| {
|
||||||
|
Self::compare_arced_layers(self.get_layer_from_mapping(&slot.key()), expected)
|
||||||
|
});
|
||||||
|
|
||||||
|
if pos.is_none() {
|
||||||
|
return Ok(Replacement::NotFound);
|
||||||
|
}
|
||||||
|
pos
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
};
|
||||||
|
|
||||||
|
let new_desc = Arc::new(new_desc);
|
||||||
|
let replaced = self.historic.replace(&key, new_desc.clone(), |existing| {
|
||||||
|
**existing == expected_desc
|
||||||
|
});
|
||||||
|
|
||||||
|
if let Replacement::Replaced { .. } = &replaced {
|
||||||
|
self.mapping.remove(&expected_desc.key());
|
||||||
|
self.mapping.insert(new_desc.key(), new);
|
||||||
|
if let Some(index) = l0_index {
|
||||||
|
self.l0_delta_layers[index] = new_desc;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let replaced = match replaced {
|
||||||
|
Replacement::Replaced { in_buffered } => Replacement::Replaced { in_buffered },
|
||||||
|
Replacement::NotFound => Replacement::NotFound,
|
||||||
|
Replacement::RemovalBuffered => Replacement::RemovalBuffered,
|
||||||
|
Replacement::Unexpected(x) => {
|
||||||
|
Replacement::Unexpected(self.get_layer_from_mapping(&x.key()).clone())
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(replaced)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Helper function for BatchedUpdates::drop.
|
/// Helper function for BatchedUpdates::drop.
|
||||||
@@ -319,8 +454,10 @@ impl LayerMap {
|
|||||||
Ok(true)
|
Ok(true)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn iter_historic_layers(&self) -> impl '_ + Iterator<Item = Arc<PersistentLayerDesc>> {
|
pub fn iter_historic_layers(&self) -> impl '_ + Iterator<Item = Arc<L>> {
|
||||||
self.historic.iter()
|
self.historic
|
||||||
|
.iter()
|
||||||
|
.map(|x| self.get_layer_from_mapping(&x.key()).clone())
|
||||||
}
|
}
|
||||||
|
|
||||||
///
|
///
|
||||||
@@ -335,7 +472,7 @@ impl LayerMap {
|
|||||||
&self,
|
&self,
|
||||||
key_range: &Range<Key>,
|
key_range: &Range<Key>,
|
||||||
lsn: Lsn,
|
lsn: Lsn,
|
||||||
) -> Result<Vec<(Range<Key>, Option<Arc<PersistentLayerDesc>>)>> {
|
) -> Result<Vec<(Range<Key>, Option<Arc<L>>)>> {
|
||||||
let version = match self.historic.get().unwrap().get_version(lsn.0) {
|
let version = match self.historic.get().unwrap().get_version(lsn.0) {
|
||||||
Some(v) => v,
|
Some(v) => v,
|
||||||
None => return Ok(vec![]),
|
None => return Ok(vec![]),
|
||||||
@@ -345,26 +482,36 @@ impl LayerMap {
|
|||||||
let end = key_range.end.to_i128();
|
let end = key_range.end.to_i128();
|
||||||
|
|
||||||
// Initialize loop variables
|
// Initialize loop variables
|
||||||
let mut coverage: Vec<(Range<Key>, Option<Arc<PersistentLayerDesc>>)> = vec![];
|
let mut coverage: Vec<(Range<Key>, Option<Arc<L>>)> = vec![];
|
||||||
let mut current_key = start;
|
let mut current_key = start;
|
||||||
let mut current_val = version.image_coverage.query(start);
|
let mut current_val = version.image_coverage.query(start);
|
||||||
|
|
||||||
// Loop through the change events and push intervals
|
// Loop through the change events and push intervals
|
||||||
for (change_key, change_val) in version.image_coverage.range(start..end) {
|
for (change_key, change_val) in version.image_coverage.range(start..end) {
|
||||||
let kr = Key::from_i128(current_key)..Key::from_i128(change_key);
|
let kr = Key::from_i128(current_key)..Key::from_i128(change_key);
|
||||||
coverage.push((kr, current_val.take()));
|
coverage.push((
|
||||||
|
kr,
|
||||||
|
current_val
|
||||||
|
.take()
|
||||||
|
.map(|l| self.get_layer_from_mapping(&l.key()).clone()),
|
||||||
|
));
|
||||||
current_key = change_key;
|
current_key = change_key;
|
||||||
current_val = change_val.clone();
|
current_val = change_val.clone();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add the final interval
|
// Add the final interval
|
||||||
let kr = Key::from_i128(current_key)..Key::from_i128(end);
|
let kr = Key::from_i128(current_key)..Key::from_i128(end);
|
||||||
coverage.push((kr, current_val.take()));
|
coverage.push((
|
||||||
|
kr,
|
||||||
|
current_val
|
||||||
|
.take()
|
||||||
|
.map(|l| self.get_layer_from_mapping(&l.key()).clone()),
|
||||||
|
));
|
||||||
|
|
||||||
Ok(coverage)
|
Ok(coverage)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn is_l0(layer: &PersistentLayerDesc) -> bool {
|
pub fn is_l0(layer: &L) -> bool {
|
||||||
range_eq(&layer.get_key_range(), &(Key::MIN..Key::MAX))
|
range_eq(&layer.get_key_range(), &(Key::MIN..Key::MAX))
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -390,7 +537,7 @@ impl LayerMap {
|
|||||||
/// TODO The optimal number should probably be slightly higher than 1, but to
|
/// TODO The optimal number should probably be slightly higher than 1, but to
|
||||||
/// implement that we need to plumb a lot more context into this function
|
/// implement that we need to plumb a lot more context into this function
|
||||||
/// than just the current partition_range.
|
/// than just the current partition_range.
|
||||||
pub fn is_reimage_worthy(layer: &PersistentLayerDesc, partition_range: &Range<Key>) -> bool {
|
pub fn is_reimage_worthy(layer: &L, partition_range: &Range<Key>) -> bool {
|
||||||
// Case 1
|
// Case 1
|
||||||
if !Self::is_l0(layer) {
|
if !Self::is_l0(layer) {
|
||||||
return true;
|
return true;
|
||||||
@@ -448,7 +595,9 @@ impl LayerMap {
|
|||||||
let kr = Key::from_i128(current_key)..Key::from_i128(change_key);
|
let kr = Key::from_i128(current_key)..Key::from_i128(change_key);
|
||||||
let lr = lsn.start..val.get_lsn_range().start;
|
let lr = lsn.start..val.get_lsn_range().start;
|
||||||
if !kr.is_empty() {
|
if !kr.is_empty() {
|
||||||
let base_count = Self::is_reimage_worthy(&val, key) as usize;
|
let base_count =
|
||||||
|
Self::is_reimage_worthy(self.get_layer_from_mapping(&val.key()), key)
|
||||||
|
as usize;
|
||||||
let new_limit = limit.map(|l| l - base_count);
|
let new_limit = limit.map(|l| l - base_count);
|
||||||
let max_stacked_deltas_underneath =
|
let max_stacked_deltas_underneath =
|
||||||
self.count_deltas(&kr, &lr, new_limit)?;
|
self.count_deltas(&kr, &lr, new_limit)?;
|
||||||
@@ -471,7 +620,9 @@ impl LayerMap {
|
|||||||
let lr = lsn.start..val.get_lsn_range().start;
|
let lr = lsn.start..val.get_lsn_range().start;
|
||||||
|
|
||||||
if !kr.is_empty() {
|
if !kr.is_empty() {
|
||||||
let base_count = Self::is_reimage_worthy(&val, key) as usize;
|
let base_count =
|
||||||
|
Self::is_reimage_worthy(self.get_layer_from_mapping(&val.key()), key)
|
||||||
|
as usize;
|
||||||
let new_limit = limit.map(|l| l - base_count);
|
let new_limit = limit.map(|l| l - base_count);
|
||||||
let max_stacked_deltas_underneath = self.count_deltas(&kr, &lr, new_limit)?;
|
let max_stacked_deltas_underneath = self.count_deltas(&kr, &lr, new_limit)?;
|
||||||
max_stacked_deltas = std::cmp::max(
|
max_stacked_deltas = std::cmp::max(
|
||||||
@@ -621,8 +772,12 @@ impl LayerMap {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Return all L0 delta layers
|
/// Return all L0 delta layers
|
||||||
pub fn get_level0_deltas(&self) -> Result<Vec<Arc<PersistentLayerDesc>>> {
|
pub fn get_level0_deltas(&self) -> Result<Vec<Arc<L>>> {
|
||||||
Ok(self.l0_delta_layers.to_vec())
|
Ok(self
|
||||||
|
.l0_delta_layers
|
||||||
|
.iter()
|
||||||
|
.map(|x| self.get_layer_from_mapping(&x.key()).clone())
|
||||||
|
.collect())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// debugging function to print out the contents of the layer map
|
/// debugging function to print out the contents of the layer map
|
||||||
@@ -647,51 +802,72 @@ impl LayerMap {
|
|||||||
println!("End dump LayerMap");
|
println!("End dump LayerMap");
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Similar to `Arc::ptr_eq`, but only compares the object pointers, not vtables.
|
||||||
|
///
|
||||||
|
/// Returns `true` if the two `Arc` point to the same layer, false otherwise.
|
||||||
|
#[inline(always)]
|
||||||
|
pub fn compare_arced_layers(left: &Arc<L>, right: &Arc<L>) -> bool {
|
||||||
|
// "dyn Trait" objects are "fat pointers" in that they have two components:
|
||||||
|
// - pointer to the object
|
||||||
|
// - pointer to the vtable
|
||||||
|
//
|
||||||
|
// rust does not provide a guarantee that these vtables are unique, but however
|
||||||
|
// `Arc::ptr_eq` as of writing (at least up to 1.67) uses a comparison where both the
|
||||||
|
// pointer and the vtable need to be equal.
|
||||||
|
//
|
||||||
|
// See: https://github.com/rust-lang/rust/issues/103763
|
||||||
|
//
|
||||||
|
// A future version of rust will most likely use this form below, where we cast each
|
||||||
|
// pointer into a pointer to unit, which drops the inaccessible vtable pointer, making it
|
||||||
|
// not affect the comparison.
|
||||||
|
//
|
||||||
|
// See: https://github.com/rust-lang/rust/pull/106450
|
||||||
|
let left = Arc::as_ptr(left) as *const ();
|
||||||
|
let right = Arc::as_ptr(right) as *const ();
|
||||||
|
|
||||||
|
left == right
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use super::LayerMap;
|
use super::{LayerMap, Replacement};
|
||||||
use crate::tenant::storage_layer::{tests::LayerDescriptor, LayerFileName};
|
use crate::tenant::storage_layer::{Layer, LayerDescriptor, LayerFileName};
|
||||||
use std::str::FromStr;
|
use std::str::FromStr;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
mod l0_delta_layers_updated {
|
mod l0_delta_layers_updated {
|
||||||
|
|
||||||
use crate::tenant::{
|
|
||||||
storage_layer::{PersistentLayer, PersistentLayerDesc},
|
|
||||||
timeline::LayerFileManager,
|
|
||||||
};
|
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn for_full_range_delta() {
|
fn for_full_range_delta() {
|
||||||
// l0_delta_layers are used by compaction, and should observe all buffered updates
|
// l0_delta_layers are used by compaction, and should observe all buffered updates
|
||||||
l0_delta_layers_updated_scenario(
|
l0_delta_layers_updated_scenario(
|
||||||
"000000000000000000000000000000000000-FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF__0000000053423C21-0000000053424D69",
|
"000000000000000000000000000000000000-FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF__0000000053423C21-0000000053424D69",
|
||||||
true
|
true
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn for_non_full_range_delta() {
|
fn for_non_full_range_delta() {
|
||||||
// has minimal uncovered areas compared to l0_delta_layers_updated_on_insert_replace_remove_for_full_range_delta
|
// has minimal uncovered areas compared to l0_delta_layers_updated_on_insert_replace_remove_for_full_range_delta
|
||||||
l0_delta_layers_updated_scenario(
|
l0_delta_layers_updated_scenario(
|
||||||
"000000000000000000000000000000000001-FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFE__0000000053423C21-0000000053424D69",
|
"000000000000000000000000000000000001-FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFE__0000000053423C21-0000000053424D69",
|
||||||
// because not full range
|
// because not full range
|
||||||
false
|
false
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn for_image() {
|
fn for_image() {
|
||||||
l0_delta_layers_updated_scenario(
|
l0_delta_layers_updated_scenario(
|
||||||
"000000000000000000000000000000000000-000000000000000000000000000000010000__0000000053424D69",
|
"000000000000000000000000000000000000-000000000000000000000000000000010000__0000000053424D69",
|
||||||
// code only checks if it is a full range layer, doesn't care about images, which must
|
// code only checks if it is a full range layer, doesn't care about images, which must
|
||||||
// mean we should in practice never have full range images
|
// mean we should in practice never have full range images
|
||||||
false
|
false
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
@@ -707,16 +883,16 @@ mod tests {
|
|||||||
let not_found = Arc::new(layer.clone());
|
let not_found = Arc::new(layer.clone());
|
||||||
let new_version = Arc::new(layer);
|
let new_version = Arc::new(layer);
|
||||||
|
|
||||||
// after the immutable storage state refactor, the replace operation
|
let mut map = LayerMap::default();
|
||||||
// will not use layer map any more. We keep it here for consistency in test cases
|
|
||||||
// and can remove it in the future.
|
|
||||||
let _map = LayerMap::default();
|
|
||||||
|
|
||||||
let mut mapping = LayerFileManager::new();
|
let res = map.batch_update().replace_historic(
|
||||||
|
not_found.get_persistent_layer_desc(),
|
||||||
|
¬_found,
|
||||||
|
new_version.get_persistent_layer_desc(),
|
||||||
|
new_version,
|
||||||
|
);
|
||||||
|
|
||||||
mapping
|
assert!(matches!(res, Ok(Replacement::NotFound)), "{res:?}");
|
||||||
.replace_and_verify(not_found, new_version)
|
|
||||||
.unwrap_err();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn l0_delta_layers_updated_scenario(layer_name: &str, expected_l0: bool) {
|
fn l0_delta_layers_updated_scenario(layer_name: &str, expected_l0: bool) {
|
||||||
@@ -727,44 +903,49 @@ mod tests {
|
|||||||
let downloaded = Arc::new(skeleton);
|
let downloaded = Arc::new(skeleton);
|
||||||
|
|
||||||
let mut map = LayerMap::default();
|
let mut map = LayerMap::default();
|
||||||
let mut mapping = LayerFileManager::new();
|
|
||||||
|
|
||||||
// two disjoint Arcs in different lifecycle phases. even if it seems they must be the
|
// two disjoint Arcs in different lifecycle phases. even if it seems they must be the
|
||||||
// same layer, we use LayerMap::compare_arced_layers as the identity of layers.
|
// same layer, we use LayerMap::compare_arced_layers as the identity of layers.
|
||||||
assert_eq!(remote.layer_desc(), downloaded.layer_desc());
|
assert!(!LayerMap::compare_arced_layers(&remote, &downloaded));
|
||||||
|
|
||||||
let expected_in_counts = (1, usize::from(expected_l0));
|
let expected_in_counts = (1, usize::from(expected_l0));
|
||||||
|
|
||||||
map.batch_update()
|
map.batch_update()
|
||||||
.insert_historic(remote.layer_desc().clone());
|
.insert_historic(remote.get_persistent_layer_desc(), remote.clone());
|
||||||
mapping.insert(remote.clone());
|
assert_eq!(count_layer_in(&map, &remote), expected_in_counts);
|
||||||
assert_eq!(
|
|
||||||
count_layer_in(&map, remote.layer_desc()),
|
|
||||||
expected_in_counts
|
|
||||||
);
|
|
||||||
|
|
||||||
mapping
|
let replaced = map
|
||||||
.replace_and_verify(remote, downloaded.clone())
|
.batch_update()
|
||||||
|
.replace_historic(
|
||||||
|
remote.get_persistent_layer_desc(),
|
||||||
|
&remote,
|
||||||
|
downloaded.get_persistent_layer_desc(),
|
||||||
|
downloaded.clone(),
|
||||||
|
)
|
||||||
.expect("name derived attributes are the same");
|
.expect("name derived attributes are the same");
|
||||||
assert_eq!(
|
assert!(
|
||||||
count_layer_in(&map, downloaded.layer_desc()),
|
matches!(replaced, Replacement::Replaced { .. }),
|
||||||
expected_in_counts
|
"{replaced:?}"
|
||||||
);
|
);
|
||||||
|
assert_eq!(count_layer_in(&map, &downloaded), expected_in_counts);
|
||||||
|
|
||||||
map.batch_update()
|
map.batch_update()
|
||||||
.remove_historic(downloaded.layer_desc().clone());
|
.remove_historic(downloaded.get_persistent_layer_desc(), downloaded.clone());
|
||||||
assert_eq!(count_layer_in(&map, downloaded.layer_desc()), (0, 0));
|
assert_eq!(count_layer_in(&map, &downloaded), (0, 0));
|
||||||
}
|
}
|
||||||
|
|
||||||
fn count_layer_in(map: &LayerMap, layer: &PersistentLayerDesc) -> (usize, usize) {
|
fn count_layer_in<L: Layer + ?Sized>(map: &LayerMap<L>, layer: &Arc<L>) -> (usize, usize) {
|
||||||
let historic = map
|
let historic = map
|
||||||
.iter_historic_layers()
|
.iter_historic_layers()
|
||||||
.filter(|x| x.key() == layer.key())
|
.filter(|x| LayerMap::compare_arced_layers(x, layer))
|
||||||
.count();
|
.count();
|
||||||
let l0s = map
|
let l0s = map
|
||||||
.get_level0_deltas()
|
.get_level0_deltas()
|
||||||
.expect("why does this return a result");
|
.expect("why does this return a result");
|
||||||
let l0 = l0s.iter().filter(|x| x.key() == layer.key()).count();
|
let l0 = l0s
|
||||||
|
.iter()
|
||||||
|
.filter(|x| LayerMap::compare_arced_layers(x, layer))
|
||||||
|
.count();
|
||||||
|
|
||||||
(historic, l0)
|
(historic, l0)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,8 +3,6 @@ use std::ops::Range;
|
|||||||
|
|
||||||
use tracing::info;
|
use tracing::info;
|
||||||
|
|
||||||
use crate::tenant::storage_layer::PersistentLayerDesc;
|
|
||||||
|
|
||||||
use super::layer_coverage::LayerCoverageTuple;
|
use super::layer_coverage::LayerCoverageTuple;
|
||||||
|
|
||||||
/// Layers in this module are identified and indexed by this data.
|
/// Layers in this module are identified and indexed by this data.
|
||||||
@@ -43,8 +41,8 @@ impl Ord for LayerKey {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<&PersistentLayerDesc> for LayerKey {
|
impl<'a, L: crate::tenant::storage_layer::Layer + ?Sized> From<&'a L> for LayerKey {
|
||||||
fn from(layer: &PersistentLayerDesc) -> Self {
|
fn from(layer: &'a L) -> Self {
|
||||||
let kr = layer.get_key_range();
|
let kr = layer.get_key_range();
|
||||||
let lr = layer.get_lsn_range();
|
let lr = layer.get_lsn_range();
|
||||||
LayerKey {
|
LayerKey {
|
||||||
@@ -456,6 +454,59 @@ impl<Value: Clone> BufferedHistoricLayerCoverage<Value> {
|
|||||||
self.buffer.insert(layer_key, None);
|
self.buffer.insert(layer_key, None);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Replaces a previous layer with a new layer value.
|
||||||
|
///
|
||||||
|
/// The replacement is conditional on:
|
||||||
|
/// - there is an existing `LayerKey` record
|
||||||
|
/// - there is no buffered removal for the given `LayerKey`
|
||||||
|
/// - the given closure returns true for the current `Value`
|
||||||
|
///
|
||||||
|
/// The closure is used to compare the latest value (buffered insert, or existing layer)
|
||||||
|
/// against some expectation. This allows to use `Arc::ptr_eq` or similar which would be
|
||||||
|
/// inaccessible via `PartialEq` trait.
|
||||||
|
///
|
||||||
|
/// Returns a `Replacement` value describing the outcome; only the case of
|
||||||
|
/// `Replacement::Replaced` modifies the map and requires a rebuild.
|
||||||
|
pub fn replace<F>(
|
||||||
|
&mut self,
|
||||||
|
layer_key: &LayerKey,
|
||||||
|
new: Value,
|
||||||
|
check_expected: F,
|
||||||
|
) -> Replacement<Value>
|
||||||
|
where
|
||||||
|
F: FnOnce(&Value) -> bool,
|
||||||
|
{
|
||||||
|
let (slot, in_buffered) = match self.buffer.get(layer_key) {
|
||||||
|
Some(inner @ Some(_)) => {
|
||||||
|
// we compare against the buffered version, because there will be a later
|
||||||
|
// rebuild before querying
|
||||||
|
(inner.as_ref(), true)
|
||||||
|
}
|
||||||
|
Some(None) => {
|
||||||
|
// buffer has removal for this key; it will not be equivalent by any check_expected.
|
||||||
|
return Replacement::RemovalBuffered;
|
||||||
|
}
|
||||||
|
None => {
|
||||||
|
// no pending modification for the key, check layers
|
||||||
|
(self.layers.get(layer_key), false)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
match slot {
|
||||||
|
Some(existing) if !check_expected(existing) => {
|
||||||
|
// unfortunate clone here, but otherwise the nll borrowck grows the region of
|
||||||
|
// 'a to cover the whole function, and we could not mutate in the other
|
||||||
|
// Some(existing) branch
|
||||||
|
Replacement::Unexpected(existing.clone())
|
||||||
|
}
|
||||||
|
None => Replacement::NotFound,
|
||||||
|
Some(_existing) => {
|
||||||
|
self.insert(layer_key.to_owned(), new);
|
||||||
|
Replacement::Replaced { in_buffered }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub fn rebuild(&mut self) {
|
pub fn rebuild(&mut self) {
|
||||||
// Find the first LSN that needs to be rebuilt
|
// Find the first LSN that needs to be rebuilt
|
||||||
let rebuild_since: u64 = match self.buffer.iter().next() {
|
let rebuild_since: u64 = match self.buffer.iter().next() {
|
||||||
@@ -524,6 +575,22 @@ impl<Value: Clone> BufferedHistoricLayerCoverage<Value> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Outcome of the replace operation.
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub enum Replacement<Value> {
|
||||||
|
/// Previous value was replaced with the new value.
|
||||||
|
Replaced {
|
||||||
|
/// Replacement happened for a scheduled insert.
|
||||||
|
in_buffered: bool,
|
||||||
|
},
|
||||||
|
/// Key was not found buffered updates or existing layers.
|
||||||
|
NotFound,
|
||||||
|
/// Key has been scheduled for removal, it was not replaced.
|
||||||
|
RemovalBuffered,
|
||||||
|
/// Previous value was rejected by the closure.
|
||||||
|
Unexpected(Value),
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_retroactive_regression_1() {
|
fn test_retroactive_regression_1() {
|
||||||
let mut map = BufferedHistoricLayerCoverage::new();
|
let mut map = BufferedHistoricLayerCoverage::new();
|
||||||
@@ -632,3 +699,139 @@ fn test_retroactive_simple() {
|
|||||||
assert_eq!(version.image_coverage.query(8), Some("Image 4".to_string()));
|
assert_eq!(version.image_coverage.query(8), Some("Image 4".to_string()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_retroactive_replacement() {
|
||||||
|
let mut map = BufferedHistoricLayerCoverage::new();
|
||||||
|
|
||||||
|
let keys = [
|
||||||
|
LayerKey {
|
||||||
|
key: 0..5,
|
||||||
|
lsn: 100..101,
|
||||||
|
is_image: true,
|
||||||
|
},
|
||||||
|
LayerKey {
|
||||||
|
key: 3..9,
|
||||||
|
lsn: 110..111,
|
||||||
|
is_image: true,
|
||||||
|
},
|
||||||
|
LayerKey {
|
||||||
|
key: 4..6,
|
||||||
|
lsn: 120..121,
|
||||||
|
is_image: true,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
let layers = [
|
||||||
|
"Image 1".to_string(),
|
||||||
|
"Image 2".to_string(),
|
||||||
|
"Image 3".to_string(),
|
||||||
|
];
|
||||||
|
|
||||||
|
for (key, layer) in keys.iter().zip(layers.iter()) {
|
||||||
|
map.insert(key.to_owned(), layer.to_owned());
|
||||||
|
}
|
||||||
|
|
||||||
|
// rebuild is not necessary here, because replace works for both buffered updates and existing
|
||||||
|
// layers.
|
||||||
|
|
||||||
|
for (key, orig_layer) in keys.iter().zip(layers.iter()) {
|
||||||
|
let replacement = format!("Remote {orig_layer}");
|
||||||
|
|
||||||
|
// evict
|
||||||
|
let ret = map.replace(key, replacement.clone(), |l| l == orig_layer);
|
||||||
|
assert!(
|
||||||
|
matches!(ret, Replacement::Replaced { .. }),
|
||||||
|
"replace {orig_layer}: {ret:?}"
|
||||||
|
);
|
||||||
|
map.rebuild();
|
||||||
|
|
||||||
|
let at = key.lsn.end + 1;
|
||||||
|
|
||||||
|
let version = map.get().expect("rebuilt").get_version(at).unwrap();
|
||||||
|
assert_eq!(
|
||||||
|
version.image_coverage.query(4).as_deref(),
|
||||||
|
Some(replacement.as_str()),
|
||||||
|
"query for 4 at version {at} after eviction",
|
||||||
|
);
|
||||||
|
|
||||||
|
// download
|
||||||
|
let ret = map.replace(key, orig_layer.clone(), |l| l == &replacement);
|
||||||
|
assert!(
|
||||||
|
matches!(ret, Replacement::Replaced { .. }),
|
||||||
|
"replace {orig_layer} back: {ret:?}"
|
||||||
|
);
|
||||||
|
map.rebuild();
|
||||||
|
let version = map.get().expect("rebuilt").get_version(at).unwrap();
|
||||||
|
assert_eq!(
|
||||||
|
version.image_coverage.query(4).as_deref(),
|
||||||
|
Some(orig_layer.as_str()),
|
||||||
|
"query for 4 at version {at} after download",
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn missing_key_is_not_inserted_with_replace() {
|
||||||
|
let mut map = BufferedHistoricLayerCoverage::new();
|
||||||
|
let key = LayerKey {
|
||||||
|
key: 0..5,
|
||||||
|
lsn: 100..101,
|
||||||
|
is_image: true,
|
||||||
|
};
|
||||||
|
|
||||||
|
let ret = map.replace(&key, "should not replace", |_| true);
|
||||||
|
assert!(matches!(ret, Replacement::NotFound), "{ret:?}");
|
||||||
|
map.rebuild();
|
||||||
|
assert!(map
|
||||||
|
.get()
|
||||||
|
.expect("no changes to rebuild")
|
||||||
|
.get_version(102)
|
||||||
|
.is_none());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn replacing_buffered_insert_and_remove() {
|
||||||
|
let mut map = BufferedHistoricLayerCoverage::new();
|
||||||
|
let key = LayerKey {
|
||||||
|
key: 0..5,
|
||||||
|
lsn: 100..101,
|
||||||
|
is_image: true,
|
||||||
|
};
|
||||||
|
|
||||||
|
map.insert(key.clone(), "Image 1");
|
||||||
|
let ret = map.replace(&key, "Remote Image 1", |&l| l == "Image 1");
|
||||||
|
assert!(
|
||||||
|
matches!(ret, Replacement::Replaced { in_buffered: true }),
|
||||||
|
"{ret:?}"
|
||||||
|
);
|
||||||
|
map.rebuild();
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
map.get()
|
||||||
|
.expect("rebuilt")
|
||||||
|
.get_version(102)
|
||||||
|
.unwrap()
|
||||||
|
.image_coverage
|
||||||
|
.query(4),
|
||||||
|
Some("Remote Image 1")
|
||||||
|
);
|
||||||
|
|
||||||
|
map.remove(key.clone());
|
||||||
|
let ret = map.replace(&key, "should not replace", |_| true);
|
||||||
|
assert!(
|
||||||
|
matches!(ret, Replacement::RemovalBuffered),
|
||||||
|
"cannot replace after scheduled remove: {ret:?}"
|
||||||
|
);
|
||||||
|
|
||||||
|
map.rebuild();
|
||||||
|
|
||||||
|
let ret = map.replace(&key, "should not replace", |_| true);
|
||||||
|
assert!(
|
||||||
|
matches!(ret, Replacement::NotFound),
|
||||||
|
"cannot replace after remove + rebuild: {ret:?}"
|
||||||
|
);
|
||||||
|
|
||||||
|
let at_version = map.get().expect("rebuilt").get_version(102);
|
||||||
|
assert!(at_version.is_none());
|
||||||
|
}
|
||||||
|
|||||||
@@ -608,7 +608,10 @@ impl RemoteTimelineClient {
|
|||||||
self.calls_unfinished_metric_begin(&op);
|
self.calls_unfinished_metric_begin(&op);
|
||||||
upload_queue.queued_operations.push_back(op);
|
upload_queue.queued_operations.push_back(op);
|
||||||
|
|
||||||
info!("scheduled layer file upload {layer_file_name}");
|
info!(
|
||||||
|
"scheduled layer file upload {}",
|
||||||
|
layer_file_name.file_name()
|
||||||
|
);
|
||||||
|
|
||||||
// Launch the task immediately, if possible
|
// Launch the task immediately, if possible
|
||||||
self.launch_queued_tasks(upload_queue);
|
self.launch_queued_tasks(upload_queue);
|
||||||
@@ -661,7 +664,7 @@ impl RemoteTimelineClient {
|
|||||||
});
|
});
|
||||||
self.calls_unfinished_metric_begin(&op);
|
self.calls_unfinished_metric_begin(&op);
|
||||||
upload_queue.queued_operations.push_back(op);
|
upload_queue.queued_operations.push_back(op);
|
||||||
info!("scheduled layer file deletion {name}");
|
info!("scheduled layer file deletion {}", name.file_name());
|
||||||
}
|
}
|
||||||
|
|
||||||
// Launch the tasks immediately, if possible
|
// Launch the tasks immediately, if possible
|
||||||
@@ -825,7 +828,7 @@ impl RemoteTimelineClient {
|
|||||||
.queued_operations
|
.queued_operations
|
||||||
.push_back(op);
|
.push_back(op);
|
||||||
|
|
||||||
info!("scheduled layer file deletion {name}");
|
info!("scheduled layer file deletion {}", name.file_name());
|
||||||
deletions_queued += 1;
|
deletions_queued += 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -859,8 +862,10 @@ impl RemoteTimelineClient {
|
|||||||
"Found {} files not bound to index_file.json, proceeding with their deletion",
|
"Found {} files not bound to index_file.json, proceeding with their deletion",
|
||||||
remaining.len()
|
remaining.len()
|
||||||
);
|
);
|
||||||
warn!("About to remove {} files", remaining.len());
|
for file in remaining {
|
||||||
self.storage_impl.delete_objects(&remaining).await?;
|
warn!("Removing {}", file.object_name().unwrap_or_default());
|
||||||
|
self.storage_impl.delete(&file).await?;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let index_file_path = timeline_storage_path.join(Path::new(IndexPart::FILE_NAME));
|
let index_file_path = timeline_storage_path.join(Path::new(IndexPart::FILE_NAME));
|
||||||
|
|||||||
@@ -176,10 +176,13 @@ impl LayerAccessStats {
|
|||||||
/// Create an empty stats object and record a [`LayerLoad`] event with the given residence status.
|
/// Create an empty stats object and record a [`LayerLoad`] event with the given residence status.
|
||||||
///
|
///
|
||||||
/// See [`record_residence_event`] for why you need to do this while holding the layer map lock.
|
/// See [`record_residence_event`] for why you need to do this while holding the layer map lock.
|
||||||
pub(crate) fn for_loading_layer(
|
pub(crate) fn for_loading_layer<L>(
|
||||||
layer_map_lock_held_witness: &BatchedUpdates<'_>,
|
layer_map_lock_held_witness: &BatchedUpdates<'_, L>,
|
||||||
status: LayerResidenceStatus,
|
status: LayerResidenceStatus,
|
||||||
) -> Self {
|
) -> Self
|
||||||
|
where
|
||||||
|
L: ?Sized + Layer,
|
||||||
|
{
|
||||||
let new = LayerAccessStats(Mutex::new(LayerAccessStatsLocked::default()));
|
let new = LayerAccessStats(Mutex::new(LayerAccessStatsLocked::default()));
|
||||||
new.record_residence_event(
|
new.record_residence_event(
|
||||||
layer_map_lock_held_witness,
|
layer_map_lock_held_witness,
|
||||||
@@ -194,11 +197,14 @@ impl LayerAccessStats {
|
|||||||
/// The `new_status` is not recorded in `self`.
|
/// The `new_status` is not recorded in `self`.
|
||||||
///
|
///
|
||||||
/// See [`record_residence_event`] for why you need to do this while holding the layer map lock.
|
/// See [`record_residence_event`] for why you need to do this while holding the layer map lock.
|
||||||
pub(crate) fn clone_for_residence_change(
|
pub(crate) fn clone_for_residence_change<L>(
|
||||||
&self,
|
&self,
|
||||||
layer_map_lock_held_witness: &BatchedUpdates<'_>,
|
layer_map_lock_held_witness: &BatchedUpdates<'_, L>,
|
||||||
new_status: LayerResidenceStatus,
|
new_status: LayerResidenceStatus,
|
||||||
) -> LayerAccessStats {
|
) -> LayerAccessStats
|
||||||
|
where
|
||||||
|
L: ?Sized + Layer,
|
||||||
|
{
|
||||||
let clone = {
|
let clone = {
|
||||||
let inner = self.0.lock().unwrap();
|
let inner = self.0.lock().unwrap();
|
||||||
inner.clone()
|
inner.clone()
|
||||||
@@ -226,12 +232,14 @@ impl LayerAccessStats {
|
|||||||
/// - Compact: Grab layer map lock, add the new L1 to layer map and remove the L0s, release layer map lock.
|
/// - Compact: Grab layer map lock, add the new L1 to layer map and remove the L0s, release layer map lock.
|
||||||
/// - Eviction: observes the new L1 layer whose only activity timestamp is the LayerCreate event.
|
/// - Eviction: observes the new L1 layer whose only activity timestamp is the LayerCreate event.
|
||||||
///
|
///
|
||||||
pub(crate) fn record_residence_event(
|
pub(crate) fn record_residence_event<L>(
|
||||||
&self,
|
&self,
|
||||||
_layer_map_lock_held_witness: &BatchedUpdates<'_>,
|
_layer_map_lock_held_witness: &BatchedUpdates<'_, L>,
|
||||||
status: LayerResidenceStatus,
|
status: LayerResidenceStatus,
|
||||||
reason: LayerResidenceEventReason,
|
reason: LayerResidenceEventReason,
|
||||||
) {
|
) where
|
||||||
|
L: ?Sized + Layer,
|
||||||
|
{
|
||||||
let mut locked = self.0.lock().unwrap();
|
let mut locked = self.0.lock().unwrap();
|
||||||
locked.iter_mut().for_each(|inner| {
|
locked.iter_mut().for_each(|inner| {
|
||||||
inner
|
inner
|
||||||
@@ -335,7 +343,7 @@ impl LayerAccessStats {
|
|||||||
/// All layers should implement a minimal `std::fmt::Debug` without tenant or
|
/// All layers should implement a minimal `std::fmt::Debug` without tenant or
|
||||||
/// timeline names, because those are known in the context of which the layers
|
/// timeline names, because those are known in the context of which the layers
|
||||||
/// are used in (timeline).
|
/// are used in (timeline).
|
||||||
pub trait Layer: std::fmt::Debug + std::fmt::Display + Send + Sync {
|
pub trait Layer: std::fmt::Debug + Send + Sync {
|
||||||
/// Range of keys that this layer covers
|
/// Range of keys that this layer covers
|
||||||
fn get_key_range(&self) -> Range<Key>;
|
fn get_key_range(&self) -> Range<Key>;
|
||||||
|
|
||||||
@@ -373,6 +381,9 @@ pub trait Layer: std::fmt::Debug + std::fmt::Display + Send + Sync {
|
|||||||
ctx: &RequestContext,
|
ctx: &RequestContext,
|
||||||
) -> Result<ValueReconstructResult>;
|
) -> Result<ValueReconstructResult>;
|
||||||
|
|
||||||
|
/// A short ID string that uniquely identifies the given layer within a [`LayerMap`].
|
||||||
|
fn short_id(&self) -> String;
|
||||||
|
|
||||||
/// Dump summary of the contents of the layer to stdout
|
/// Dump summary of the contents of the layer to stdout
|
||||||
fn dump(&self, verbose: bool, ctx: &RequestContext) -> Result<()>;
|
fn dump(&self, verbose: bool, ctx: &RequestContext) -> Result<()>;
|
||||||
}
|
}
|
||||||
@@ -462,127 +473,94 @@ pub fn downcast_remote_layer(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub mod tests {
|
/// Holds metadata about a layer without any content. Used mostly for testing.
|
||||||
use super::*;
|
///
|
||||||
|
/// To use filenames as fixtures, parse them as [`LayerFileName`] then convert from that to a
|
||||||
|
/// LayerDescriptor.
|
||||||
|
#[derive(Clone, Debug)]
|
||||||
|
pub struct LayerDescriptor {
|
||||||
|
pub key: Range<Key>,
|
||||||
|
pub lsn: Range<Lsn>,
|
||||||
|
pub is_incremental: bool,
|
||||||
|
pub short_id: String,
|
||||||
|
}
|
||||||
|
|
||||||
/// Holds metadata about a layer without any content. Used mostly for testing.
|
impl LayerDescriptor {
|
||||||
///
|
/// `LayerDescriptor` is only used for testing purpose so it does not matter whether it is image / delta,
|
||||||
/// To use filenames as fixtures, parse them as [`LayerFileName`] then convert from that to a
|
/// and the tenant / timeline id does not matter.
|
||||||
/// LayerDescriptor.
|
pub fn get_persistent_layer_desc(&self) -> PersistentLayerDesc {
|
||||||
#[derive(Clone, Debug)]
|
PersistentLayerDesc::new_delta(
|
||||||
pub struct LayerDescriptor {
|
TenantId::from_array([0; 16]),
|
||||||
base: PersistentLayerDesc,
|
TimelineId::from_array([0; 16]),
|
||||||
|
self.key.clone(),
|
||||||
|
self.lsn.clone(),
|
||||||
|
233,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Layer for LayerDescriptor {
|
||||||
|
fn get_key_range(&self) -> Range<Key> {
|
||||||
|
self.key.clone()
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<PersistentLayerDesc> for LayerDescriptor {
|
fn get_lsn_range(&self) -> Range<Lsn> {
|
||||||
fn from(base: PersistentLayerDesc) -> Self {
|
self.lsn.clone()
|
||||||
Self { base }
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Layer for LayerDescriptor {
|
fn is_incremental(&self) -> bool {
|
||||||
fn get_value_reconstruct_data(
|
self.is_incremental
|
||||||
&self,
|
|
||||||
_key: Key,
|
|
||||||
_lsn_range: Range<Lsn>,
|
|
||||||
_reconstruct_data: &mut ValueReconstructState,
|
|
||||||
_ctx: &RequestContext,
|
|
||||||
) -> Result<ValueReconstructResult> {
|
|
||||||
todo!("This method shouldn't be part of the Layer trait")
|
|
||||||
}
|
|
||||||
|
|
||||||
fn dump(&self, _verbose: bool, _ctx: &RequestContext) -> Result<()> {
|
|
||||||
todo!()
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Boilerplate to implement the Layer trait, always use layer_desc for persistent layers.
|
|
||||||
fn get_key_range(&self) -> Range<Key> {
|
|
||||||
self.layer_desc().key_range.clone()
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Boilerplate to implement the Layer trait, always use layer_desc for persistent layers.
|
|
||||||
fn get_lsn_range(&self) -> Range<Lsn> {
|
|
||||||
self.layer_desc().lsn_range.clone()
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Boilerplate to implement the Layer trait, always use layer_desc for persistent layers.
|
|
||||||
fn is_incremental(&self) -> bool {
|
|
||||||
self.layer_desc().is_incremental
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Boilerplate to implement the Layer trait, always use layer_desc for persistent layers.
|
fn get_value_reconstruct_data(
|
||||||
impl std::fmt::Display for LayerDescriptor {
|
&self,
|
||||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
_key: Key,
|
||||||
write!(f, "{}", self.layer_desc().short_id())
|
_lsn_range: Range<Lsn>,
|
||||||
}
|
_reconstruct_data: &mut ValueReconstructState,
|
||||||
|
_ctx: &RequestContext,
|
||||||
|
) -> Result<ValueReconstructResult> {
|
||||||
|
todo!("This method shouldn't be part of the Layer trait")
|
||||||
}
|
}
|
||||||
|
|
||||||
impl PersistentLayer for LayerDescriptor {
|
fn short_id(&self) -> String {
|
||||||
fn layer_desc(&self) -> &PersistentLayerDesc {
|
self.short_id.clone()
|
||||||
&self.base
|
|
||||||
}
|
|
||||||
|
|
||||||
fn local_path(&self) -> Option<PathBuf> {
|
|
||||||
unimplemented!()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn iter(&self, _: &RequestContext) -> Result<LayerIter<'_>> {
|
|
||||||
unimplemented!()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn key_iter(&self, _: &RequestContext) -> Result<LayerKeyIter<'_>> {
|
|
||||||
unimplemented!()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn delete_resident_layer_file(&self) -> Result<()> {
|
|
||||||
unimplemented!()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn info(&self, _: LayerAccessStatsReset) -> HistoricLayerInfo {
|
|
||||||
unimplemented!()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn access_stats(&self) -> &LayerAccessStats {
|
|
||||||
unimplemented!()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<DeltaFileName> for LayerDescriptor {
|
fn dump(&self, _verbose: bool, _ctx: &RequestContext) -> Result<()> {
|
||||||
fn from(value: DeltaFileName) -> Self {
|
todo!()
|
||||||
LayerDescriptor {
|
}
|
||||||
base: PersistentLayerDesc::new_delta(
|
}
|
||||||
TenantId::from_array([0; 16]),
|
|
||||||
TimelineId::from_array([0; 16]),
|
impl From<DeltaFileName> for LayerDescriptor {
|
||||||
value.key_range,
|
fn from(value: DeltaFileName) -> Self {
|
||||||
value.lsn_range,
|
let short_id = value.to_string();
|
||||||
233,
|
LayerDescriptor {
|
||||||
),
|
key: value.key_range,
|
||||||
}
|
lsn: value.lsn_range,
|
||||||
|
is_incremental: true,
|
||||||
|
short_id,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl From<ImageFileName> for LayerDescriptor {
|
impl From<ImageFileName> for LayerDescriptor {
|
||||||
fn from(value: ImageFileName) -> Self {
|
fn from(value: ImageFileName) -> Self {
|
||||||
LayerDescriptor {
|
let short_id = value.to_string();
|
||||||
base: PersistentLayerDesc::new_img(
|
let lsn = value.lsn_as_range();
|
||||||
TenantId::from_array([0; 16]),
|
LayerDescriptor {
|
||||||
TimelineId::from_array([0; 16]),
|
key: value.key_range,
|
||||||
value.key_range,
|
lsn,
|
||||||
value.lsn,
|
is_incremental: false,
|
||||||
false,
|
short_id,
|
||||||
233,
|
|
||||||
),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl From<LayerFileName> for LayerDescriptor {
|
impl From<LayerFileName> for LayerDescriptor {
|
||||||
fn from(value: LayerFileName) -> Self {
|
fn from(value: LayerFileName) -> Self {
|
||||||
match value {
|
match value {
|
||||||
LayerFileName::Delta(d) => Self::from(d),
|
LayerFileName::Delta(d) => Self::from(d),
|
||||||
LayerFileName::Image(i) => Self::from(i),
|
LayerFileName::Image(i) => Self::from(i),
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -394,11 +394,10 @@ impl Layer for DeltaLayer {
|
|||||||
fn is_incremental(&self) -> bool {
|
fn is_incremental(&self) -> bool {
|
||||||
self.layer_desc().is_incremental
|
self.layer_desc().is_incremental
|
||||||
}
|
}
|
||||||
}
|
|
||||||
/// Boilerplate to implement the Layer trait, always use layer_desc for persistent layers.
|
/// Boilerplate to implement the Layer trait, always use layer_desc for persistent layers.
|
||||||
impl std::fmt::Display for DeltaLayer {
|
fn short_id(&self) -> String {
|
||||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
self.layer_desc().short_id()
|
||||||
write!(f, "{}", self.layer_desc().short_id())
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -210,15 +210,9 @@ pub enum LayerFileName {
|
|||||||
|
|
||||||
impl LayerFileName {
|
impl LayerFileName {
|
||||||
pub fn file_name(&self) -> String {
|
pub fn file_name(&self) -> String {
|
||||||
self.to_string()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl fmt::Display for LayerFileName {
|
|
||||||
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
||||||
match self {
|
match self {
|
||||||
Self::Image(fname) => write!(f, "{fname}"),
|
Self::Image(fname) => fname.to_string(),
|
||||||
Self::Delta(fname) => write!(f, "{fname}"),
|
Self::Delta(fname) => fname.to_string(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -230,12 +230,10 @@ impl Layer for ImageLayer {
|
|||||||
fn is_incremental(&self) -> bool {
|
fn is_incremental(&self) -> bool {
|
||||||
self.layer_desc().is_incremental
|
self.layer_desc().is_incremental
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
/// Boilerplate to implement the Layer trait, always use layer_desc for persistent layers.
|
/// Boilerplate to implement the Layer trait, always use layer_desc for persistent layers.
|
||||||
impl std::fmt::Display for ImageLayer {
|
fn short_id(&self) -> String {
|
||||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
self.layer_desc().short_id()
|
||||||
write!(f, "{}", self.layer_desc().short_id())
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -131,6 +131,13 @@ impl Layer for InMemoryLayer {
|
|||||||
true
|
true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn short_id(&self) -> String {
|
||||||
|
let inner = self.inner.read().unwrap();
|
||||||
|
|
||||||
|
let end_lsn = inner.end_lsn.unwrap_or(Lsn(u64::MAX));
|
||||||
|
format!("inmem-{:016X}-{:016X}", self.start_lsn.0, end_lsn.0)
|
||||||
|
}
|
||||||
|
|
||||||
/// debugging function to print out the contents of the layer
|
/// debugging function to print out the contents of the layer
|
||||||
fn dump(&self, verbose: bool, _ctx: &RequestContext) -> Result<()> {
|
fn dump(&self, verbose: bool, _ctx: &RequestContext) -> Result<()> {
|
||||||
let inner = self.inner.read().unwrap();
|
let inner = self.inner.read().unwrap();
|
||||||
@@ -233,15 +240,6 @@ impl Layer for InMemoryLayer {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl std::fmt::Display for InMemoryLayer {
|
|
||||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
||||||
let inner = self.inner.read().unwrap();
|
|
||||||
|
|
||||||
let end_lsn = inner.end_lsn.unwrap_or(Lsn(u64::MAX));
|
|
||||||
write!(f, "inmem-{:016X}-{:016X}", self.start_lsn.0, end_lsn.0)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl InMemoryLayer {
|
impl InMemoryLayer {
|
||||||
///
|
///
|
||||||
/// Get layer size on the disk
|
/// Get layer size on the disk
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
use core::fmt::Display;
|
|
||||||
use std::ops::Range;
|
use std::ops::Range;
|
||||||
use utils::{
|
use utils::{
|
||||||
id::{TenantId, TimelineId},
|
id::{TenantId, TimelineId},
|
||||||
@@ -49,8 +48,8 @@ impl PersistentLayerDesc {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn short_id(&self) -> impl Display {
|
pub fn short_id(&self) -> String {
|
||||||
self.filename()
|
self.filename().file_name()
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
|
|||||||
@@ -71,7 +71,10 @@ impl Layer for RemoteLayer {
|
|||||||
_reconstruct_state: &mut ValueReconstructState,
|
_reconstruct_state: &mut ValueReconstructState,
|
||||||
_ctx: &RequestContext,
|
_ctx: &RequestContext,
|
||||||
) -> Result<ValueReconstructResult> {
|
) -> Result<ValueReconstructResult> {
|
||||||
bail!("layer {self} needs to be downloaded");
|
bail!(
|
||||||
|
"layer {} needs to be downloaded",
|
||||||
|
self.filename().file_name()
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// debugging function to print out the contents of the layer
|
/// debugging function to print out the contents of the layer
|
||||||
@@ -103,12 +106,10 @@ impl Layer for RemoteLayer {
|
|||||||
fn is_incremental(&self) -> bool {
|
fn is_incremental(&self) -> bool {
|
||||||
self.layer_desc().is_incremental
|
self.layer_desc().is_incremental
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
/// Boilerplate to implement the Layer trait, always use layer_desc for persistent layers.
|
/// Boilerplate to implement the Layer trait, always use layer_desc for persistent layers.
|
||||||
impl std::fmt::Display for RemoteLayer {
|
fn short_id(&self) -> String {
|
||||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
self.layer_desc().short_id()
|
||||||
write!(f, "{}", self.layer_desc().short_id())
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -217,12 +218,15 @@ impl RemoteLayer {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Create a Layer struct representing this layer, after it has been downloaded.
|
/// Create a Layer struct representing this layer, after it has been downloaded.
|
||||||
pub fn create_downloaded_layer(
|
pub fn create_downloaded_layer<L>(
|
||||||
&self,
|
&self,
|
||||||
layer_map_lock_held_witness: &BatchedUpdates<'_>,
|
layer_map_lock_held_witness: &BatchedUpdates<'_, L>,
|
||||||
conf: &'static PageServerConf,
|
conf: &'static PageServerConf,
|
||||||
file_size: u64,
|
file_size: u64,
|
||||||
) -> Arc<dyn PersistentLayer> {
|
) -> Arc<dyn PersistentLayer>
|
||||||
|
where
|
||||||
|
L: ?Sized + Layer,
|
||||||
|
{
|
||||||
if self.desc.is_delta {
|
if self.desc.is_delta {
|
||||||
let fname = self.desc.delta_file_name();
|
let fname = self.desc.delta_file_name();
|
||||||
Arc::new(DeltaLayer::new(
|
Arc::new(DeltaLayer::new(
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -70,6 +70,7 @@ impl Timeline {
|
|||||||
};
|
};
|
||||||
|
|
||||||
self_clone.eviction_task(cancel).await;
|
self_clone.eviction_task(cancel).await;
|
||||||
|
info!("eviction task finishing");
|
||||||
Ok(())
|
Ok(())
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
@@ -77,9 +78,6 @@ impl Timeline {
|
|||||||
|
|
||||||
#[instrument(skip_all, fields(tenant_id = %self.tenant_id, timeline_id = %self.timeline_id))]
|
#[instrument(skip_all, fields(tenant_id = %self.tenant_id, timeline_id = %self.timeline_id))]
|
||||||
async fn eviction_task(self: Arc<Self>, cancel: CancellationToken) {
|
async fn eviction_task(self: Arc<Self>, cancel: CancellationToken) {
|
||||||
scopeguard::defer! {
|
|
||||||
info!("eviction task finishing");
|
|
||||||
}
|
|
||||||
use crate::tenant::tasks::random_init_delay;
|
use crate::tenant::tasks::random_init_delay;
|
||||||
{
|
{
|
||||||
let policy = self.get_eviction_policy();
|
let policy = self.get_eviction_policy();
|
||||||
@@ -88,6 +86,7 @@ impl Timeline {
|
|||||||
EvictionPolicy::NoEviction => Duration::from_secs(10),
|
EvictionPolicy::NoEviction => Duration::from_secs(10),
|
||||||
};
|
};
|
||||||
if random_init_delay(period, &cancel).await.is_err() {
|
if random_init_delay(period, &cancel).await.is_err() {
|
||||||
|
info!("shutting down");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -102,6 +101,7 @@ impl Timeline {
|
|||||||
ControlFlow::Continue(sleep_until) => {
|
ControlFlow::Continue(sleep_until) => {
|
||||||
tokio::select! {
|
tokio::select! {
|
||||||
_ = cancel.cancelled() => {
|
_ = cancel.cancelled() => {
|
||||||
|
info!("shutting down");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
_ = tokio::time::sleep_until(sleep_until) => { }
|
_ = tokio::time::sleep_until(sleep_until) => { }
|
||||||
@@ -197,11 +197,9 @@ impl Timeline {
|
|||||||
// We don't want to hold the layer map lock during eviction.
|
// We don't want to hold the layer map lock during eviction.
|
||||||
// So, we just need to deal with this.
|
// So, we just need to deal with this.
|
||||||
let candidates: Vec<Arc<dyn PersistentLayer>> = {
|
let candidates: Vec<Arc<dyn PersistentLayer>> = {
|
||||||
let guard = self.layers.read().await;
|
let layers = self.layers.read().await;
|
||||||
let (layers, mapping) = &*guard;
|
|
||||||
let mut candidates = Vec::new();
|
let mut candidates = Vec::new();
|
||||||
for hist_layer in layers.iter_historic_layers() {
|
for hist_layer in layers.iter_historic_layers() {
|
||||||
let hist_layer = mapping.get_from_desc(&hist_layer);
|
|
||||||
if hist_layer.is_remote_layer() {
|
if hist_layer.is_remote_layer() {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@@ -209,7 +207,7 @@ impl Timeline {
|
|||||||
let last_activity_ts = hist_layer.access_stats().latest_activity().unwrap_or_else(|| {
|
let last_activity_ts = hist_layer.access_stats().latest_activity().unwrap_or_else(|| {
|
||||||
// We only use this fallback if there's an implementation error.
|
// We only use this fallback if there's an implementation error.
|
||||||
// `latest_activity` already does rate-limited warn!() log.
|
// `latest_activity` already does rate-limited warn!() log.
|
||||||
debug!(layer=%hist_layer, "last_activity returns None, using SystemTime::now");
|
debug!(layer=%hist_layer.filename().file_name(), "last_activity returns None, using SystemTime::now");
|
||||||
SystemTime::now()
|
SystemTime::now()
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -71,8 +71,6 @@ pub(super) async fn handle_walreceiver_connection(
|
|||||||
ctx: RequestContext,
|
ctx: RequestContext,
|
||||||
node: NodeId,
|
node: NodeId,
|
||||||
) -> anyhow::Result<()> {
|
) -> anyhow::Result<()> {
|
||||||
debug_assert_current_span_has_tenant_and_timeline_id();
|
|
||||||
|
|
||||||
WALRECEIVER_STARTED_CONNECTIONS.inc();
|
WALRECEIVER_STARTED_CONNECTIONS.inc();
|
||||||
|
|
||||||
// Connect to the database in replication mode.
|
// Connect to the database in replication mode.
|
||||||
@@ -142,9 +140,6 @@ pub(super) async fn handle_walreceiver_connection(
|
|||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
// Enrich the log lines emitted by this closure with meaningful context.
|
|
||||||
// TODO: technically, this task outlives the surrounding function, so, the
|
|
||||||
// spans won't be properly nested.
|
|
||||||
.instrument(tracing::info_span!("poller")),
|
.instrument(tracing::info_span!("poller")),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|||||||
@@ -302,6 +302,15 @@ impl VirtualFile {
|
|||||||
.observe_closure_duration(|| self.open_options.open(&self.path))?;
|
.observe_closure_duration(|| self.open_options.open(&self.path))?;
|
||||||
|
|
||||||
// Perform the requested operation on it
|
// Perform the requested operation on it
|
||||||
|
//
|
||||||
|
// TODO: We could downgrade the locks to read mode before calling
|
||||||
|
// 'func', to allow a little bit more concurrency, but the standard
|
||||||
|
// library RwLock doesn't allow downgrading without releasing the lock,
|
||||||
|
// and that doesn't seem worth the trouble.
|
||||||
|
//
|
||||||
|
// XXX: `parking_lot::RwLock` can enable such downgrades, yet its implementation is fair and
|
||||||
|
// may deadlock on subsequent read calls.
|
||||||
|
// Simply replacing all `RwLock` in project causes deadlocks, so use it sparingly.
|
||||||
let result = STORAGE_IO_TIME
|
let result = STORAGE_IO_TIME
|
||||||
.with_label_values(&[op, &self.tenant_id, &self.timeline_id])
|
.with_label_values(&[op, &self.tenant_id, &self.timeline_id])
|
||||||
.observe_closure_duration(|| func(&file));
|
.observe_closure_duration(|| func(&file));
|
||||||
|
|||||||
@@ -122,43 +122,6 @@ hnsw_populate(HierarchicalNSW* hnsw, Relation indexRel, Relation heapRel)
|
|||||||
true, true, hnsw_build_callback, (void *) hnsw, NULL);
|
true, true, hnsw_build_callback, (void *) hnsw, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef __APPLE__
|
|
||||||
|
|
||||||
#include <sys/types.h>
|
|
||||||
#include <sys/sysctl.h>
|
|
||||||
|
|
||||||
static void
|
|
||||||
hnsw_check_available_memory(Size requested)
|
|
||||||
{
|
|
||||||
size_t total;
|
|
||||||
if (sysctlbyname("hw.memsize", NULL, &total, NULL, 0) < 0)
|
|
||||||
elog(ERROR, "Failed to get amount of RAM: %m");
|
|
||||||
|
|
||||||
if ((Size)NBuffers*BLCKSZ + requested >= total)
|
|
||||||
elog(ERROR, "HNSW index requeries %ld bytes while only %ld are available",
|
|
||||||
requested, total - (Size)NBuffers*BLCKSZ);
|
|
||||||
}
|
|
||||||
|
|
||||||
#else
|
|
||||||
|
|
||||||
#include <sys/sysinfo.h>
|
|
||||||
|
|
||||||
static void
|
|
||||||
hnsw_check_available_memory(Size requested)
|
|
||||||
{
|
|
||||||
struct sysinfo si;
|
|
||||||
Size total;
|
|
||||||
if (sysinfo(&si) < 0)
|
|
||||||
elog(ERROR, "Failed to get amount of RAM: %n");
|
|
||||||
|
|
||||||
total = si.totalram*si.mem_unit;
|
|
||||||
if ((Size)NBuffers*BLCKSZ + requested >= total)
|
|
||||||
elog(ERROR, "HNSW index requeries %ld bytes while only %ld are available",
|
|
||||||
requested, total - (Size)NBuffers*BLCKSZ);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
static HierarchicalNSW*
|
static HierarchicalNSW*
|
||||||
hnsw_get_index(Relation indexRel, Relation heapRel)
|
hnsw_get_index(Relation indexRel, Relation heapRel)
|
||||||
{
|
{
|
||||||
@@ -193,8 +156,6 @@ hnsw_get_index(Relation indexRel, Relation heapRel)
|
|||||||
size_data_per_element = size_links_level0 + data_size + sizeof(label_t);
|
size_data_per_element = size_links_level0 + data_size + sizeof(label_t);
|
||||||
shmem_size = hnsw_sizeof() + maxelements * size_data_per_element;
|
shmem_size = hnsw_sizeof() + maxelements * size_data_per_element;
|
||||||
|
|
||||||
hnsw_check_available_memory(shmem_size);
|
|
||||||
|
|
||||||
/* first try to attach to existed index */
|
/* first try to attach to existed index */
|
||||||
if (!dsm_impl_op(DSM_OP_ATTACH, handle, 0, &impl_private,
|
if (!dsm_impl_op(DSM_OP_ATTACH, handle, 0, &impl_private,
|
||||||
&mapped_address, &mapped_size, DEBUG1))
|
&mapped_address, &mapped_size, DEBUG1))
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
comment = 'hnsw index'
|
comment = 'hNsw index'
|
||||||
default_version = '0.1.0'
|
default_version = '0.1.0'
|
||||||
module_pathname = '$libdir/hnsw'
|
module_pathname = '$libdir/hnsw'
|
||||||
relocatable = true
|
relocatable = true
|
||||||
|
|||||||
@@ -4,6 +4,7 @@
|
|||||||
MODULE_big = neon
|
MODULE_big = neon
|
||||||
OBJS = \
|
OBJS = \
|
||||||
$(WIN32RES) \
|
$(WIN32RES) \
|
||||||
|
extension_server.o \
|
||||||
file_cache.o \
|
file_cache.o \
|
||||||
libpagestore.o \
|
libpagestore.o \
|
||||||
libpqwalproposer.o \
|
libpqwalproposer.o \
|
||||||
|
|||||||
@@ -32,7 +32,6 @@
|
|||||||
#include "port.h"
|
#include "port.h"
|
||||||
#include <curl/curl.h>
|
#include <curl/curl.h>
|
||||||
#include "utils/jsonb.h"
|
#include "utils/jsonb.h"
|
||||||
#include "libpq/crypt.h"
|
|
||||||
|
|
||||||
static ProcessUtility_hook_type PreviousProcessUtilityHook = NULL;
|
static ProcessUtility_hook_type PreviousProcessUtilityHook = NULL;
|
||||||
|
|
||||||
@@ -162,22 +161,7 @@ ConstructDeltaMessage()
|
|||||||
PushKeyValue(&state, "name", entry->name);
|
PushKeyValue(&state, "name", entry->name);
|
||||||
if (entry->password)
|
if (entry->password)
|
||||||
{
|
{
|
||||||
#if PG_MAJORVERSION_NUM == 14
|
|
||||||
char *logdetail;
|
|
||||||
#else
|
|
||||||
const char *logdetail;
|
|
||||||
#endif
|
|
||||||
PushKeyValue(&state, "password", (char *) entry->password);
|
PushKeyValue(&state, "password", (char *) entry->password);
|
||||||
char *encrypted_password = get_role_password(entry->name, &logdetail);
|
|
||||||
|
|
||||||
if (encrypted_password)
|
|
||||||
{
|
|
||||||
PushKeyValue(&state, "encrypted_password", encrypted_password);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
elog(ERROR, "Failed to get encrypted password: %s", logdetail);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if (entry->old_name[0] != '\0')
|
if (entry->old_name[0] != '\0')
|
||||||
{
|
{
|
||||||
|
|||||||
91
pgxn/neon/extension_server.c
Normal file
91
pgxn/neon/extension_server.c
Normal file
@@ -0,0 +1,91 @@
|
|||||||
|
|
||||||
|
/*-------------------------------------------------------------------------
|
||||||
|
*
|
||||||
|
* extension_server.c
|
||||||
|
* Request compute_ctl to download extension files.
|
||||||
|
*
|
||||||
|
* IDENTIFICATION
|
||||||
|
* contrib/neon/extension_server.c
|
||||||
|
*
|
||||||
|
*-------------------------------------------------------------------------
|
||||||
|
*/
|
||||||
|
#include "postgres.h"
|
||||||
|
#include "tcop/pquery.h"
|
||||||
|
#include "tcop/utility.h"
|
||||||
|
#include "access/xact.h"
|
||||||
|
#include "utils/hsearch.h"
|
||||||
|
#include "utils/memutils.h"
|
||||||
|
#include "commands/defrem.h"
|
||||||
|
#include "miscadmin.h"
|
||||||
|
#include "utils/acl.h"
|
||||||
|
#include "fmgr.h"
|
||||||
|
#include "utils/guc.h"
|
||||||
|
#include "port.h"
|
||||||
|
#include "fmgr.h"
|
||||||
|
|
||||||
|
#include <curl/curl.h>
|
||||||
|
|
||||||
|
static int extension_server_port = 0;
|
||||||
|
|
||||||
|
static download_extension_file_hook_type prev_download_extension_file_hook = NULL;
|
||||||
|
|
||||||
|
// curl -X POST http://localhost:8080/extension_server/postgis-3.so
|
||||||
|
static bool
|
||||||
|
neon_download_extension_file_http(const char *filename)
|
||||||
|
{
|
||||||
|
CURL *curl;
|
||||||
|
CURLcode res;
|
||||||
|
char *compute_ctl_url;
|
||||||
|
char *postdata;
|
||||||
|
bool ret = false;
|
||||||
|
|
||||||
|
if ((curl = curl_easy_init()) == NULL)
|
||||||
|
{
|
||||||
|
elog(ERROR, "Failed to initialize curl handle");
|
||||||
|
}
|
||||||
|
|
||||||
|
compute_ctl_url = psprintf("http://localhost:%d/extension_server/%s", extension_server_port, filename);
|
||||||
|
|
||||||
|
elog(LOG, "curl_easy_perform() url: %s", compute_ctl_url);
|
||||||
|
|
||||||
|
curl_easy_setopt(curl, CURLOPT_CUSTOMREQUEST, "POST");
|
||||||
|
curl_easy_setopt(curl, CURLOPT_URL, compute_ctl_url);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_TIMEOUT, 3L /* seconds */);
|
||||||
|
|
||||||
|
if (curl)
|
||||||
|
{
|
||||||
|
/* Perform the request, res will get the return code */
|
||||||
|
res = curl_easy_perform(curl);
|
||||||
|
/* Check for errors */
|
||||||
|
if (res == CURLE_OK)
|
||||||
|
{
|
||||||
|
elog(LOG, "curl_easy_perform() succeeded");
|
||||||
|
ret = true;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
elog(WARNING, "curl_easy_perform() failed: %s\n", curl_easy_strerror(res));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* always cleanup */
|
||||||
|
curl_easy_cleanup(curl);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
void pg_init_extension_server()
|
||||||
|
{
|
||||||
|
DefineCustomIntVariable("neon.extension_server_port",
|
||||||
|
"connection string to the compute_ctl",
|
||||||
|
NULL,
|
||||||
|
&extension_server_port,
|
||||||
|
0, 0, INT_MAX,
|
||||||
|
PGC_POSTMASTER,
|
||||||
|
0, /* no flags required */
|
||||||
|
NULL, NULL, NULL);
|
||||||
|
|
||||||
|
// set download_extension_file_hook
|
||||||
|
prev_download_extension_file_hook = download_extension_file_hook;
|
||||||
|
download_extension_file_hook = neon_download_extension_file_http;
|
||||||
|
}
|
||||||
1
pgxn/neon/extension_server.h
Normal file
1
pgxn/neon/extension_server.h
Normal file
@@ -0,0 +1 @@
|
|||||||
|
|
||||||
@@ -190,7 +190,7 @@ lfc_change_limit_hook(int newval, void *extra)
|
|||||||
hash_search(lfc_hash, &victim->key, HASH_REMOVE, NULL);
|
hash_search(lfc_hash, &victim->key, HASH_REMOVE, NULL);
|
||||||
lfc_ctl->used -= 1;
|
lfc_ctl->used -= 1;
|
||||||
}
|
}
|
||||||
elog(DEBUG1, "set local file cache limit to %d", new_size);
|
elog(LOG, "set local file cache limit to %d", new_size);
|
||||||
LWLockRelease(lfc_lock);
|
LWLockRelease(lfc_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -32,7 +32,3 @@ CREATE VIEW local_cache AS
|
|||||||
SELECT P.* FROM local_cache_pages() AS P
|
SELECT P.* FROM local_cache_pages() AS P
|
||||||
(pageoffs int8, relfilenode oid, reltablespace oid, reldatabase oid,
|
(pageoffs int8, relfilenode oid, reltablespace oid, reldatabase oid,
|
||||||
relforknumber int2, relblocknumber int8, accesscount int4);
|
relforknumber int2, relblocknumber int8, accesscount int4);
|
||||||
|
|
||||||
CREATE FUNCTION copy_from(conninfo cstring) RETURNS BIGINT
|
|
||||||
AS 'MODULE_PATHNAME', 'copy_from'
|
|
||||||
LANGUAGE C;
|
|
||||||
|
|||||||
294
pgxn/neon/neon.c
294
pgxn/neon/neon.c
@@ -13,32 +13,20 @@
|
|||||||
|
|
||||||
#include "access/xact.h"
|
#include "access/xact.h"
|
||||||
#include "access/xlog.h"
|
#include "access/xlog.h"
|
||||||
#include "access/relation.h"
|
|
||||||
#include "access/xloginsert.h"
|
|
||||||
#include "storage/buf_internals.h"
|
#include "storage/buf_internals.h"
|
||||||
#include "storage/bufmgr.h"
|
#include "storage/bufmgr.h"
|
||||||
#include "catalog/pg_type.h"
|
#include "catalog/pg_type.h"
|
||||||
#include "catalog/namespace.h"
|
|
||||||
#include "replication/walsender.h"
|
#include "replication/walsender.h"
|
||||||
#include "funcapi.h"
|
#include "funcapi.h"
|
||||||
#include "miscadmin.h"
|
|
||||||
#include "access/htup_details.h"
|
#include "access/htup_details.h"
|
||||||
#include "utils/pg_lsn.h"
|
#include "utils/pg_lsn.h"
|
||||||
#include "utils/guc.h"
|
#include "utils/guc.h"
|
||||||
#include "utils/wait_event.h"
|
|
||||||
#include "utils/rel.h"
|
|
||||||
#include "utils/varlena.h"
|
|
||||||
#include "utils/builtins.h"
|
|
||||||
|
|
||||||
#include "neon.h"
|
#include "neon.h"
|
||||||
#include "walproposer.h"
|
#include "walproposer.h"
|
||||||
#include "pagestore_client.h"
|
#include "pagestore_client.h"
|
||||||
#include "control_plane_connector.h"
|
#include "control_plane_connector.h"
|
||||||
|
|
||||||
#include "libpq-fe.h"
|
|
||||||
#include "libpq/pqformat.h"
|
|
||||||
#include "libpq/libpq.h"
|
|
||||||
|
|
||||||
PG_MODULE_MAGIC;
|
PG_MODULE_MAGIC;
|
||||||
void _PG_init(void);
|
void _PG_init(void);
|
||||||
|
|
||||||
@@ -47,8 +35,11 @@ _PG_init(void)
|
|||||||
{
|
{
|
||||||
pg_init_libpagestore();
|
pg_init_libpagestore();
|
||||||
pg_init_walproposer();
|
pg_init_walproposer();
|
||||||
|
|
||||||
InitControlPlaneConnector();
|
InitControlPlaneConnector();
|
||||||
|
|
||||||
|
pg_init_extension_server();
|
||||||
|
|
||||||
// Important: This must happen after other parts of the extension
|
// Important: This must happen after other parts of the extension
|
||||||
// are loaded, otherwise any settings to GUCs that were set before
|
// are loaded, otherwise any settings to GUCs that were set before
|
||||||
// the extension was loaded will be removed.
|
// the extension was loaded will be removed.
|
||||||
@@ -58,7 +49,6 @@ _PG_init(void)
|
|||||||
PG_FUNCTION_INFO_V1(pg_cluster_size);
|
PG_FUNCTION_INFO_V1(pg_cluster_size);
|
||||||
PG_FUNCTION_INFO_V1(backpressure_lsns);
|
PG_FUNCTION_INFO_V1(backpressure_lsns);
|
||||||
PG_FUNCTION_INFO_V1(backpressure_throttling_time);
|
PG_FUNCTION_INFO_V1(backpressure_throttling_time);
|
||||||
PG_FUNCTION_INFO_V1(copy_from);
|
|
||||||
|
|
||||||
Datum
|
Datum
|
||||||
pg_cluster_size(PG_FUNCTION_ARGS)
|
pg_cluster_size(PG_FUNCTION_ARGS)
|
||||||
@@ -104,281 +94,3 @@ backpressure_throttling_time(PG_FUNCTION_ARGS)
|
|||||||
{
|
{
|
||||||
PG_RETURN_UINT64(BackpressureThrottlingTime());
|
PG_RETURN_UINT64(BackpressureThrottlingTime());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#define N_RAW_PAGE_COLUMNS 4
|
|
||||||
#define COPY_FETCH_COUNT 16
|
|
||||||
|
|
||||||
|
|
||||||
static void
|
|
||||||
report_error(int elevel, PGresult *res, PGconn *conn,
|
|
||||||
bool clear, const char *sql)
|
|
||||||
{
|
|
||||||
/* If requested, PGresult must be released before leaving this function. */
|
|
||||||
PG_TRY();
|
|
||||||
{
|
|
||||||
char *diag_sqlstate = PQresultErrorField(res, PG_DIAG_SQLSTATE);
|
|
||||||
char *message_primary = PQresultErrorField(res, PG_DIAG_MESSAGE_PRIMARY);
|
|
||||||
char *message_detail = PQresultErrorField(res, PG_DIAG_MESSAGE_DETAIL);
|
|
||||||
char *message_hint = PQresultErrorField(res, PG_DIAG_MESSAGE_HINT);
|
|
||||||
char *message_context = PQresultErrorField(res, PG_DIAG_CONTEXT);
|
|
||||||
int sqlstate;
|
|
||||||
|
|
||||||
if (diag_sqlstate)
|
|
||||||
sqlstate = MAKE_SQLSTATE(diag_sqlstate[0],
|
|
||||||
diag_sqlstate[1],
|
|
||||||
diag_sqlstate[2],
|
|
||||||
diag_sqlstate[3],
|
|
||||||
diag_sqlstate[4]);
|
|
||||||
else
|
|
||||||
sqlstate = ERRCODE_CONNECTION_FAILURE;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* If we don't get a message from the PGresult, try the PGconn. This
|
|
||||||
* is needed because for connection-level failures, PQexec may just
|
|
||||||
* return NULL, not a PGresult at all.
|
|
||||||
*/
|
|
||||||
if (message_primary == NULL)
|
|
||||||
message_primary = pchomp(PQerrorMessage(conn));
|
|
||||||
|
|
||||||
ereport(elevel,
|
|
||||||
(errcode(sqlstate),
|
|
||||||
(message_primary != NULL && message_primary[0] != '\0') ?
|
|
||||||
errmsg_internal("%s", message_primary) :
|
|
||||||
errmsg("could not obtain message string for remote error"),
|
|
||||||
message_detail ? errdetail_internal("%s", message_detail) : 0,
|
|
||||||
message_hint ? errhint("%s", message_hint) : 0,
|
|
||||||
message_context ? errcontext("%s", message_context) : 0,
|
|
||||||
sql ? errcontext("remote SQL command: %s", sql) : 0));
|
|
||||||
}
|
|
||||||
PG_FINALLY();
|
|
||||||
{
|
|
||||||
if (clear)
|
|
||||||
PQclear(res);
|
|
||||||
}
|
|
||||||
PG_END_TRY();
|
|
||||||
}
|
|
||||||
|
|
||||||
static PGresult *
|
|
||||||
get_result(PGconn *conn, const char *query)
|
|
||||||
{
|
|
||||||
PGresult *volatile last_res = NULL;
|
|
||||||
|
|
||||||
/* In what follows, do not leak any PGresults on an error. */
|
|
||||||
PG_TRY();
|
|
||||||
{
|
|
||||||
for (;;)
|
|
||||||
{
|
|
||||||
PGresult *res;
|
|
||||||
|
|
||||||
while (PQisBusy(conn))
|
|
||||||
{
|
|
||||||
int wc;
|
|
||||||
|
|
||||||
/* Sleep until there's something to do */
|
|
||||||
wc = WaitLatchOrSocket(MyLatch,
|
|
||||||
WL_LATCH_SET | WL_SOCKET_READABLE |
|
|
||||||
WL_EXIT_ON_PM_DEATH,
|
|
||||||
PQsocket(conn),
|
|
||||||
-1L, PG_WAIT_EXTENSION);
|
|
||||||
ResetLatch(MyLatch);
|
|
||||||
|
|
||||||
CHECK_FOR_INTERRUPTS();
|
|
||||||
|
|
||||||
/* Data available in socket? */
|
|
||||||
if (wc & WL_SOCKET_READABLE)
|
|
||||||
{
|
|
||||||
if (!PQconsumeInput(conn))
|
|
||||||
report_error(ERROR, NULL, conn, false, query);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
res = PQgetResult(conn);
|
|
||||||
if (res == NULL)
|
|
||||||
break; /* query is complete */
|
|
||||||
|
|
||||||
PQclear(last_res);
|
|
||||||
last_res = res;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
PG_CATCH();
|
|
||||||
{
|
|
||||||
PQclear(last_res);
|
|
||||||
PG_RE_THROW();
|
|
||||||
}
|
|
||||||
PG_END_TRY();
|
|
||||||
|
|
||||||
return last_res;
|
|
||||||
}
|
|
||||||
|
|
||||||
#define CREATE_COPYDATA_FUNC "\
|
|
||||||
create or replace function copydata() returns setof record as $$ \
|
|
||||||
declare \
|
|
||||||
relsize integer; \
|
|
||||||
total_relsize integer; \
|
|
||||||
content bytea; \
|
|
||||||
r record; \
|
|
||||||
fork text; \
|
|
||||||
relname text; \
|
|
||||||
pagesize integer; \
|
|
||||||
begin \
|
|
||||||
pagesize = current_setting('block_size'); \
|
|
||||||
for r in select oid,reltoastrelid from pg_class where relnamespace not in (select oid from pg_namespace where nspname in ('pg_catalog','pg_toast','information_schema')) \
|
|
||||||
loop \
|
|
||||||
relname = r.oid::regclass::text; \
|
|
||||||
total_relsize = 0; \
|
|
||||||
foreach fork in array array['main','vm','fsm'] \
|
|
||||||
loop \
|
|
||||||
relsize = pg_relation_size(r.oid, fork)/pagesize; \
|
|
||||||
total_relsize = total_relsize + relsize; \
|
|
||||||
for p in 1..relsize \
|
|
||||||
loop \
|
|
||||||
content = get_raw_page(relname, fork, p-1); \
|
|
||||||
return next row(relname,fork,p-1,content); \
|
|
||||||
end loop; \
|
|
||||||
end loop; \
|
|
||||||
if total_relsize <> 0 and r.reltoastrelid <> 0 then \
|
|
||||||
foreach relname in array array ['pg_toast.pg_toast_'||r.oid, 'pg_toast.pg_toast_'||r.oid||'_index'] \
|
|
||||||
loop \
|
|
||||||
foreach fork in array array['main','vm','fsm'] \
|
|
||||||
loop \
|
|
||||||
relsize = pg_relation_size(relname, fork)/pagesize; \
|
|
||||||
for p in 1..relsize \
|
|
||||||
loop \
|
|
||||||
content = get_raw_page(relname, fork, p-1); \
|
|
||||||
return next row(relname,fork,p-1,content); \
|
|
||||||
end loop; \
|
|
||||||
end loop; \
|
|
||||||
end loop; \
|
|
||||||
end if; \
|
|
||||||
end loop; \
|
|
||||||
end; \
|
|
||||||
$$ language plpgsql"
|
|
||||||
|
|
||||||
Datum
|
|
||||||
copy_from(PG_FUNCTION_ARGS)
|
|
||||||
{
|
|
||||||
char const* conninfo = PG_GETARG_CSTRING(0);
|
|
||||||
PGconn* conn;
|
|
||||||
char const* declare_cursor = "declare copy_data_cursor no scroll cursor for select * from copydata() as raw_page(relid text, fork text, blkno integer, content bytea)";
|
|
||||||
char* fetch_cursor = psprintf("fetch forward %d copy_data_cursor", COPY_FETCH_COUNT);
|
|
||||||
char const* close_cursor = "close copy_data_cursor";
|
|
||||||
char const* vacuum_freeze = "vacuum freeze";
|
|
||||||
char *content;
|
|
||||||
char const* relname;
|
|
||||||
BlockNumber blkno;
|
|
||||||
ForkNumber forknum;
|
|
||||||
BlockNumber prev_blkno = InvalidBlockNumber;
|
|
||||||
RangeVar *relrv;
|
|
||||||
Relation rel = NULL;
|
|
||||||
BlockNumber rel_size;
|
|
||||||
int64_t total = 0;
|
|
||||||
PGresult *res;
|
|
||||||
char blkno_buf[4];
|
|
||||||
int n_tuples;
|
|
||||||
Buffer buf;
|
|
||||||
char* toast_rel_name;
|
|
||||||
Oid relid = InvalidOid;
|
|
||||||
|
|
||||||
/* Connect to the source database */
|
|
||||||
conn = PQconnectdb(conninfo);
|
|
||||||
if (!conn || PQstatus(conn) != CONNECTION_OK)
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_SQLCLIENT_UNABLE_TO_ESTABLISH_SQLCONNECTION),
|
|
||||||
errmsg("could not connect to server \"%s\"",
|
|
||||||
conninfo),
|
|
||||||
errdetail_internal("%s", pchomp(PQerrorMessage(conn)))));
|
|
||||||
|
|
||||||
/* First create store procedure (assumes that pageinspector extension is already installed) */
|
|
||||||
res = PQexec(conn, CREATE_COPYDATA_FUNC);
|
|
||||||
if (res == NULL || PQresultStatus(res) != PGRES_COMMAND_OK)
|
|
||||||
report_error(ERROR, res, conn, true, CREATE_COPYDATA_FUNC);
|
|
||||||
PQclear(res);
|
|
||||||
|
|
||||||
/* Freeze all tables to prevent problems with XID mapping */
|
|
||||||
res = PQexec(conn, vacuum_freeze);
|
|
||||||
if (res == NULL || PQresultStatus(res) != PGRES_COMMAND_OK)
|
|
||||||
report_error(ERROR, res, conn, true, vacuum_freeze);
|
|
||||||
PQclear(res);
|
|
||||||
|
|
||||||
/* Start transaction to use cursor */
|
|
||||||
res = PQexec(conn, "BEGIN");
|
|
||||||
if (res == NULL || PQresultStatus(res) != PGRES_COMMAND_OK)
|
|
||||||
report_error(ERROR, res, conn, true, "BEGIN");
|
|
||||||
PQclear(res);
|
|
||||||
|
|
||||||
/* Declare cursor (we have to use cursor to avoid materializing all database in memory) */
|
|
||||||
res = PQexec(conn, declare_cursor);
|
|
||||||
if (res == NULL || PQresultStatus(res) != PGRES_COMMAND_OK)
|
|
||||||
report_error(ERROR, res, conn, true, declare_cursor);
|
|
||||||
PQclear(res);
|
|
||||||
|
|
||||||
/* Get database data */
|
|
||||||
while ((res = PQexecParams(conn, fetch_cursor, 0, NULL, NULL, NULL, NULL, 1)) != NULL)
|
|
||||||
{
|
|
||||||
if (PQresultStatus(res) != PGRES_TUPLES_OK)
|
|
||||||
report_error(ERROR, res, conn, true, fetch_cursor);
|
|
||||||
|
|
||||||
n_tuples = PQntuples(res);
|
|
||||||
if (PQnfields(res) != 4)
|
|
||||||
elog(ERROR, "unexpected result from copydata()");
|
|
||||||
|
|
||||||
for (int i = 0; i < n_tuples; i++)
|
|
||||||
{
|
|
||||||
relname = PQgetvalue(res, i, 0);
|
|
||||||
forknum = forkname_to_number(PQgetvalue(res, i, 1));
|
|
||||||
memcpy(&blkno, PQgetvalue(res, i, 2), sizeof(BlockNumber));
|
|
||||||
blkno = pg_ntoh32(blkno);
|
|
||||||
content = (char*)PQgetvalue(res, i, 3);
|
|
||||||
|
|
||||||
if (blkno <= prev_blkno)
|
|
||||||
{
|
|
||||||
if (forknum == MAIN_FORKNUM)
|
|
||||||
{
|
|
||||||
char* dst_rel_name = strncmp(relname, "pg_toast.", 9) == 0
|
|
||||||
/* Construct correct TOAST table name */
|
|
||||||
? psprintf("pg_toast.pg_toast_%u%s",
|
|
||||||
relid,
|
|
||||||
strcmp(relname + strlen(relname) - 5, "index") == 0 ? "_index" : "")
|
|
||||||
: (char*)relname;
|
|
||||||
if (rel)
|
|
||||||
relation_close(rel, AccessExclusiveLock);
|
|
||||||
relrv = makeRangeVarFromNameList(textToQualifiedNameList(cstring_to_text(dst_rel_name)));
|
|
||||||
rel = relation_openrv(relrv, AccessExclusiveLock);
|
|
||||||
if (dst_rel_name != relname)
|
|
||||||
pfree(dst_rel_name);
|
|
||||||
else
|
|
||||||
relid = RelationGetRelid(rel);
|
|
||||||
}
|
|
||||||
rel_size = RelationGetNumberOfBlocksInFork(rel, forknum);
|
|
||||||
}
|
|
||||||
buf = ReadBufferExtended(rel, forknum, blkno < rel_size ? blkno : P_NEW, RBM_ZERO_AND_LOCK, NULL);
|
|
||||||
MarkBufferDirty(buf);
|
|
||||||
memcpy(BufferGetPage(buf), content, BLCKSZ);
|
|
||||||
log_newpage_buffer(buf, forknum == MAIN_FORKNUM);
|
|
||||||
UnlockReleaseBuffer(buf);
|
|
||||||
|
|
||||||
total += 1;
|
|
||||||
prev_blkno = blkno;
|
|
||||||
}
|
|
||||||
PQclear(res);
|
|
||||||
if (n_tuples < COPY_FETCH_COUNT)
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
res = PQexec(conn, close_cursor);
|
|
||||||
if (res == NULL || PQresultStatus(res) != PGRES_COMMAND_OK)
|
|
||||||
report_error(ERROR, res, conn, true, close_cursor);
|
|
||||||
PQclear(res);
|
|
||||||
|
|
||||||
if (rel)
|
|
||||||
relation_close(rel, AccessExclusiveLock);
|
|
||||||
|
|
||||||
/* Complete transaction */
|
|
||||||
res = PQexec(conn, "END");
|
|
||||||
if (res == NULL || PQresultStatus(res) != PGRES_COMMAND_OK)
|
|
||||||
report_error(ERROR, res, conn, true, "END");
|
|
||||||
PQclear(res);
|
|
||||||
|
|
||||||
PQfinish(conn);
|
|
||||||
PG_RETURN_INT64(total);
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -21,6 +21,8 @@ extern char *neon_tenant;
|
|||||||
extern void pg_init_libpagestore(void);
|
extern void pg_init_libpagestore(void);
|
||||||
extern void pg_init_walproposer(void);
|
extern void pg_init_walproposer(void);
|
||||||
|
|
||||||
|
extern void pg_init_extension_server(void);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Returns true if we shouldn't do REDO on that block in record indicated by
|
* Returns true if we shouldn't do REDO on that block in record indicated by
|
||||||
* block_id; false otherwise.
|
* block_id; false otherwise.
|
||||||
|
|||||||
@@ -2675,6 +2675,7 @@ bool
|
|||||||
neon_redo_read_buffer_filter(XLogReaderState *record, uint8 block_id)
|
neon_redo_read_buffer_filter(XLogReaderState *record, uint8 block_id)
|
||||||
{
|
{
|
||||||
XLogRecPtr end_recptr = record->EndRecPtr;
|
XLogRecPtr end_recptr = record->EndRecPtr;
|
||||||
|
XLogRecPtr prev_end_recptr = record->ReadRecPtr - 1;
|
||||||
RelFileNode rnode;
|
RelFileNode rnode;
|
||||||
ForkNumber forknum;
|
ForkNumber forknum;
|
||||||
BlockNumber blkno;
|
BlockNumber blkno;
|
||||||
@@ -2718,15 +2719,16 @@ neon_redo_read_buffer_filter(XLogReaderState *record, uint8 block_id)
|
|||||||
|
|
||||||
no_redo_needed = buffer < 0;
|
no_redo_needed = buffer < 0;
|
||||||
|
|
||||||
/* In both cases st lwlsn past this WAL record */
|
/* we don't have the buffer in memory, update lwLsn past this record */
|
||||||
SetLastWrittenLSNForBlock(end_recptr, rnode, forknum, blkno);
|
|
||||||
|
|
||||||
/* we don't have the buffer in memory, update lwLsn past this record,
|
|
||||||
* also evict page fro file cache
|
|
||||||
*/
|
|
||||||
if (no_redo_needed)
|
if (no_redo_needed)
|
||||||
|
{
|
||||||
|
SetLastWrittenLSNForBlock(end_recptr, rnode, forknum, blkno);
|
||||||
lfc_evict(rnode, forknum, blkno);
|
lfc_evict(rnode, forknum, blkno);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
SetLastWrittenLSNForBlock(prev_end_recptr, rnode, forknum, blkno);
|
||||||
|
}
|
||||||
|
|
||||||
LWLockRelease(partitionLock);
|
LWLockRelease(partitionLock);
|
||||||
|
|
||||||
@@ -2734,10 +2736,7 @@ neon_redo_read_buffer_filter(XLogReaderState *record, uint8 block_id)
|
|||||||
if (get_cached_relsize(rnode, forknum, &relsize))
|
if (get_cached_relsize(rnode, forknum, &relsize))
|
||||||
{
|
{
|
||||||
if (relsize < blkno + 1)
|
if (relsize < blkno + 1)
|
||||||
{
|
|
||||||
update_cached_relsize(rnode, forknum, blkno + 1);
|
update_cached_relsize(rnode, forknum, blkno + 1);
|
||||||
SetLastWrittenLSNForRelation(end_recptr, rnode, forknum);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@@ -2769,7 +2768,6 @@ neon_redo_read_buffer_filter(XLogReaderState *record, uint8 block_id)
|
|||||||
Assert(nbresponse->n_blocks > blkno);
|
Assert(nbresponse->n_blocks > blkno);
|
||||||
|
|
||||||
set_cached_relsize(rnode, forknum, nbresponse->n_blocks);
|
set_cached_relsize(rnode, forknum, nbresponse->n_blocks);
|
||||||
SetLastWrittenLSNForRelation(end_recptr, rnode, forknum);
|
|
||||||
|
|
||||||
elog(SmgrTrace, "Set length to %d", nbresponse->n_blocks);
|
elog(SmgrTrace, "Set length to %d", nbresponse->n_blocks);
|
||||||
}
|
}
|
||||||
|
|||||||
123
poetry.lock
generated
123
poetry.lock
generated
@@ -1654,74 +1654,71 @@ test = ["enum34", "ipaddress", "mock", "pywin32", "wmi"]
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "psycopg2-binary"
|
name = "psycopg2-binary"
|
||||||
version = "2.9.6"
|
version = "2.9.3"
|
||||||
description = "psycopg2 - Python-PostgreSQL Database Adapter"
|
description = "psycopg2 - Python-PostgreSQL Database Adapter"
|
||||||
category = "main"
|
category = "main"
|
||||||
optional = false
|
optional = false
|
||||||
python-versions = ">=3.6"
|
python-versions = ">=3.6"
|
||||||
files = [
|
files = [
|
||||||
{file = "psycopg2-binary-2.9.6.tar.gz", hash = "sha256:1f64dcfb8f6e0c014c7f55e51c9759f024f70ea572fbdef123f85318c297947c"},
|
{file = "psycopg2-binary-2.9.3.tar.gz", hash = "sha256:761df5313dc15da1502b21453642d7599d26be88bff659382f8f9747c7ebea4e"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:d26e0342183c762de3276cca7a530d574d4e25121ca7d6e4a98e4f05cb8e4df7"},
|
{file = "psycopg2_binary-2.9.3-cp310-cp310-macosx_10_14_x86_64.macosx_10_9_intel.macosx_10_9_x86_64.macosx_10_10_intel.macosx_10_10_x86_64.whl", hash = "sha256:539b28661b71da7c0e428692438efbcd048ca21ea81af618d845e06ebfd29478"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:c48d8f2db17f27d41fb0e2ecd703ea41984ee19362cbce52c097963b3a1b4365"},
|
{file = "psycopg2_binary-2.9.3-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:2f2534ab7dc7e776a263b463a16e189eb30e85ec9bbe1bff9e78dae802608932"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ffe9dc0a884a8848075e576c1de0290d85a533a9f6e9c4e564f19adf8f6e54a7"},
|
{file = "psycopg2_binary-2.9.3-cp310-cp310-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6e82d38390a03da28c7985b394ec3f56873174e2c88130e6966cb1c946508e65"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp310-cp310-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:8a76e027f87753f9bd1ab5f7c9cb8c7628d1077ef927f5e2446477153a602f2c"},
|
{file = "psycopg2_binary-2.9.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:57804fc02ca3ce0dbfbef35c4b3a4a774da66d66ea20f4bda601294ad2ea6092"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:6460c7a99fc939b849431f1e73e013d54aa54293f30f1109019c56a0b2b2ec2f"},
|
{file = "psycopg2_binary-2.9.3-cp310-cp310-manylinux_2_24_aarch64.whl", hash = "sha256:083a55275f09a62b8ca4902dd11f4b33075b743cf0d360419e2051a8a5d5ff76"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ae102a98c547ee2288637af07393dd33f440c25e5cd79556b04e3fca13325e5f"},
|
{file = "psycopg2_binary-2.9.3-cp310-cp310-manylinux_2_24_ppc64le.whl", hash = "sha256:0a29729145aaaf1ad8bafe663131890e2111f13416b60e460dae0a96af5905c9"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:9972aad21f965599ed0106f65334230ce826e5ae69fda7cbd688d24fa922415e"},
|
{file = "psycopg2_binary-2.9.3-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:3a79d622f5206d695d7824cbf609a4f5b88ea6d6dab5f7c147fc6d333a8787e4"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:7a40c00dbe17c0af5bdd55aafd6ff6679f94a9be9513a4c7e071baf3d7d22a70"},
|
{file = "psycopg2_binary-2.9.3-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:090f3348c0ab2cceb6dfbe6bf721ef61262ddf518cd6cc6ecc7d334996d64efa"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:cacbdc5839bdff804dfebc058fe25684cae322987f7a38b0168bc1b2df703fb1"},
|
{file = "psycopg2_binary-2.9.3-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:a9e1f75f96ea388fbcef36c70640c4efbe4650658f3d6a2967b4cc70e907352e"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:7f0438fa20fb6c7e202863e0d5ab02c246d35efb1d164e052f2f3bfe2b152bd0"},
|
{file = "psycopg2_binary-2.9.3-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:c3ae8e75eb7160851e59adc77b3a19a976e50622e44fd4fd47b8b18208189d42"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp310-cp310-win32.whl", hash = "sha256:b6c8288bb8a84b47e07013bb4850f50538aa913d487579e1921724631d02ea1b"},
|
{file = "psycopg2_binary-2.9.3-cp310-cp310-win32.whl", hash = "sha256:7b1e9b80afca7b7a386ef087db614faebbf8839b7f4db5eb107d0f1a53225029"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp310-cp310-win_amd64.whl", hash = "sha256:61b047a0537bbc3afae10f134dc6393823882eb263088c271331602b672e52e9"},
|
{file = "psycopg2_binary-2.9.3-cp310-cp310-win_amd64.whl", hash = "sha256:8b344adbb9a862de0c635f4f0425b7958bf5a4b927c8594e6e8d261775796d53"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:964b4dfb7c1c1965ac4c1978b0f755cc4bd698e8aa2b7667c575fb5f04ebe06b"},
|
{file = "psycopg2_binary-2.9.3-cp36-cp36m-macosx_10_14_x86_64.macosx_10_9_intel.macosx_10_9_x86_64.macosx_10_10_intel.macosx_10_10_x86_64.whl", hash = "sha256:e847774f8ffd5b398a75bc1c18fbb56564cda3d629fe68fd81971fece2d3c67e"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:afe64e9b8ea66866a771996f6ff14447e8082ea26e675a295ad3bdbffdd72afb"},
|
{file = "psycopg2_binary-2.9.3-cp36-cp36m-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:68641a34023d306be959101b345732360fc2ea4938982309b786f7be1b43a4a1"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:15e2ee79e7cf29582ef770de7dab3d286431b01c3bb598f8e05e09601b890081"},
|
{file = "psycopg2_binary-2.9.3-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3303f8807f342641851578ee7ed1f3efc9802d00a6f83c101d21c608cb864460"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:dfa74c903a3c1f0d9b1c7e7b53ed2d929a4910e272add6700c38f365a6002820"},
|
{file = "psycopg2_binary-2.9.3-cp36-cp36m-manylinux_2_24_aarch64.whl", hash = "sha256:e3699852e22aa68c10de06524a3721ade969abf382da95884e6a10ff798f9281"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b83456c2d4979e08ff56180a76429263ea254c3f6552cd14ada95cff1dec9bb8"},
|
{file = "psycopg2_binary-2.9.3-cp36-cp36m-manylinux_2_24_ppc64le.whl", hash = "sha256:526ea0378246d9b080148f2d6681229f4b5964543c170dd10bf4faaab6e0d27f"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0645376d399bfd64da57148694d78e1f431b1e1ee1054872a5713125681cf1be"},
|
{file = "psycopg2_binary-2.9.3-cp36-cp36m-musllinux_1_1_aarch64.whl", hash = "sha256:b1c8068513f5b158cf7e29c43a77eb34b407db29aca749d3eb9293ee0d3103ca"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:e99e34c82309dd78959ba3c1590975b5d3c862d6f279f843d47d26ff89d7d7e1"},
|
{file = "psycopg2_binary-2.9.3-cp36-cp36m-musllinux_1_1_i686.whl", hash = "sha256:15803fa813ea05bef089fa78835118b5434204f3a17cb9f1e5dbfd0b9deea5af"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:4ea29fc3ad9d91162c52b578f211ff1c931d8a38e1f58e684c45aa470adf19e2"},
|
{file = "psycopg2_binary-2.9.3-cp36-cp36m-musllinux_1_1_ppc64le.whl", hash = "sha256:152f09f57417b831418304c7f30d727dc83a12761627bb826951692cc6491e57"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:4ac30da8b4f57187dbf449294d23b808f8f53cad6b1fc3623fa8a6c11d176dd0"},
|
{file = "psycopg2_binary-2.9.3-cp36-cp36m-musllinux_1_1_x86_64.whl", hash = "sha256:404224e5fef3b193f892abdbf8961ce20e0b6642886cfe1fe1923f41aaa75c9d"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:e78e6e2a00c223e164c417628572a90093c031ed724492c763721c2e0bc2a8df"},
|
{file = "psycopg2_binary-2.9.3-cp36-cp36m-win32.whl", hash = "sha256:1f6b813106a3abdf7b03640d36e24669234120c72e91d5cbaeb87c5f7c36c65b"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp311-cp311-win32.whl", hash = "sha256:1876843d8e31c89c399e31b97d4b9725a3575bb9c2af92038464231ec40f9edb"},
|
{file = "psycopg2_binary-2.9.3-cp36-cp36m-win_amd64.whl", hash = "sha256:2d872e3c9d5d075a2e104540965a1cf898b52274a5923936e5bfddb58c59c7c2"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp311-cp311-win_amd64.whl", hash = "sha256:b4b24f75d16a89cc6b4cdff0eb6a910a966ecd476d1e73f7ce5985ff1328e9a6"},
|
{file = "psycopg2_binary-2.9.3-cp37-cp37m-macosx_10_14_x86_64.macosx_10_9_intel.macosx_10_9_x86_64.macosx_10_10_intel.macosx_10_10_x86_64.whl", hash = "sha256:10bb90fb4d523a2aa67773d4ff2b833ec00857f5912bafcfd5f5414e45280fb1"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp36-cp36m-win32.whl", hash = "sha256:498807b927ca2510baea1b05cc91d7da4718a0f53cb766c154c417a39f1820a0"},
|
{file = "psycopg2_binary-2.9.3-cp37-cp37m-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:874a52ecab70af13e899f7847b3e074eeb16ebac5615665db33bce8a1009cf33"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp36-cp36m-win_amd64.whl", hash = "sha256:0d236c2825fa656a2d98bbb0e52370a2e852e5a0ec45fc4f402977313329174d"},
|
{file = "psycopg2_binary-2.9.3-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a29b3ca4ec9defec6d42bf5feb36bb5817ba3c0230dd83b4edf4bf02684cd0ae"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:34b9ccdf210cbbb1303c7c4db2905fa0319391bd5904d32689e6dd5c963d2ea8"},
|
{file = "psycopg2_binary-2.9.3-cp37-cp37m-manylinux_2_24_aarch64.whl", hash = "sha256:12b11322ea00ad8db8c46f18b7dfc47ae215e4df55b46c67a94b4effbaec7094"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:84d2222e61f313c4848ff05353653bf5f5cf6ce34df540e4274516880d9c3763"},
|
{file = "psycopg2_binary-2.9.3-cp37-cp37m-manylinux_2_24_ppc64le.whl", hash = "sha256:53293533fcbb94c202b7c800a12c873cfe24599656b341f56e71dd2b557be063"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp37-cp37m-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:30637a20623e2a2eacc420059be11527f4458ef54352d870b8181a4c3020ae6b"},
|
{file = "psycopg2_binary-2.9.3-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:c381bda330ddf2fccbafab789d83ebc6c53db126e4383e73794c74eedce855ef"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:8122cfc7cae0da9a3077216528b8bb3629c43b25053284cc868744bfe71eb141"},
|
{file = "psycopg2_binary-2.9.3-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:9d29409b625a143649d03d0fd7b57e4b92e0ecad9726ba682244b73be91d2fdb"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:38601cbbfe600362c43714482f43b7c110b20cb0f8172422c616b09b85a750c5"},
|
{file = "psycopg2_binary-2.9.3-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:183a517a3a63503f70f808b58bfbf962f23d73b6dccddae5aa56152ef2bcb232"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:c7e62ab8b332147a7593a385d4f368874d5fe4ad4e341770d4983442d89603e3"},
|
{file = "psycopg2_binary-2.9.3-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:15c4e4cfa45f5a60599d9cec5f46cd7b1b29d86a6390ec23e8eebaae84e64554"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:2ab652e729ff4ad76d400df2624d223d6e265ef81bb8aa17fbd63607878ecbee"},
|
{file = "psycopg2_binary-2.9.3-cp37-cp37m-win32.whl", hash = "sha256:adf20d9a67e0b6393eac162eb81fb10bc9130a80540f4df7e7355c2dd4af9fba"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:c83a74b68270028dc8ee74d38ecfaf9c90eed23c8959fca95bd703d25b82c88e"},
|
{file = "psycopg2_binary-2.9.3-cp37-cp37m-win_amd64.whl", hash = "sha256:2f9ffd643bc7349eeb664eba8864d9e01f057880f510e4681ba40a6532f93c71"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:d4e6036decf4b72d6425d5b29bbd3e8f0ff1059cda7ac7b96d6ac5ed34ffbacd"},
|
{file = "psycopg2_binary-2.9.3-cp38-cp38-macosx_10_14_x86_64.macosx_10_9_intel.macosx_10_9_x86_64.macosx_10_10_intel.macosx_10_10_x86_64.whl", hash = "sha256:def68d7c21984b0f8218e8a15d514f714d96904265164f75f8d3a70f9c295667"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp37-cp37m-win32.whl", hash = "sha256:a8c28fd40a4226b4a84bdf2d2b5b37d2c7bd49486b5adcc200e8c7ec991dfa7e"},
|
{file = "psycopg2_binary-2.9.3-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:e6aa71ae45f952a2205377773e76f4e3f27951df38e69a4c95440c779e013560"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp37-cp37m-win_amd64.whl", hash = "sha256:51537e3d299be0db9137b321dfb6a5022caaab275775680e0c3d281feefaca6b"},
|
{file = "psycopg2_binary-2.9.3-cp38-cp38-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:dffc08ca91c9ac09008870c9eb77b00a46b3378719584059c034b8945e26b272"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:cf4499e0a83b7b7edcb8dabecbd8501d0d3a5ef66457200f77bde3d210d5debb"},
|
{file = "psycopg2_binary-2.9.3-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:280b0bb5cbfe8039205c7981cceb006156a675362a00fe29b16fbc264e242834"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:7e13a5a2c01151f1208d5207e42f33ba86d561b7a89fca67c700b9486a06d0e2"},
|
{file = "psycopg2_binary-2.9.3-cp38-cp38-manylinux_2_24_aarch64.whl", hash = "sha256:af9813db73395fb1fc211bac696faea4ca9ef53f32dc0cfa27e4e7cf766dcf24"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0e0f754d27fddcfd74006455b6e04e6705d6c31a612ec69ddc040a5468e44b4e"},
|
{file = "psycopg2_binary-2.9.3-cp38-cp38-manylinux_2_24_ppc64le.whl", hash = "sha256:63638d875be8c2784cfc952c9ac34e2b50e43f9f0a0660b65e2a87d656b3116c"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp38-cp38-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d57c3fd55d9058645d26ae37d76e61156a27722097229d32a9e73ed54819982a"},
|
{file = "psycopg2_binary-2.9.3-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:ffb7a888a047696e7f8240d649b43fb3644f14f0ee229077e7f6b9f9081635bd"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:71f14375d6f73b62800530b581aed3ada394039877818b2d5f7fc77e3bb6894d"},
|
{file = "psycopg2_binary-2.9.3-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:0c9d5450c566c80c396b7402895c4369a410cab5a82707b11aee1e624da7d004"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:441cc2f8869a4f0f4bb408475e5ae0ee1f3b55b33f350406150277f7f35384fc"},
|
{file = "psycopg2_binary-2.9.3-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:d1c1b569ecafe3a69380a94e6ae09a4789bbb23666f3d3a08d06bbd2451f5ef1"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:65bee1e49fa6f9cf327ce0e01c4c10f39165ee76d35c846ade7cb0ec6683e303"},
|
{file = "psycopg2_binary-2.9.3-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:8fc53f9af09426a61db9ba357865c77f26076d48669f2e1bb24d85a22fb52307"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:af335bac6b666cc6aea16f11d486c3b794029d9df029967f9938a4bed59b6a19"},
|
{file = "psycopg2_binary-2.9.3-cp38-cp38-win32.whl", hash = "sha256:6472a178e291b59e7f16ab49ec8b4f3bdada0a879c68d3817ff0963e722a82ce"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:cfec476887aa231b8548ece2e06d28edc87c1397ebd83922299af2e051cf2827"},
|
{file = "psycopg2_binary-2.9.3-cp38-cp38-win_amd64.whl", hash = "sha256:35168209c9d51b145e459e05c31a9eaeffa9a6b0fd61689b48e07464ffd1a83e"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:65c07febd1936d63bfde78948b76cd4c2a411572a44ac50719ead41947d0f26b"},
|
{file = "psycopg2_binary-2.9.3-cp39-cp39-macosx_10_14_x86_64.macosx_10_9_intel.macosx_10_9_x86_64.macosx_10_10_intel.macosx_10_10_x86_64.whl", hash = "sha256:47133f3f872faf28c1e87d4357220e809dfd3fa7c64295a4a148bcd1e6e34ec9"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp38-cp38-win32.whl", hash = "sha256:4dfb4be774c4436a4526d0c554af0cc2e02082c38303852a36f6456ece7b3503"},
|
{file = "psycopg2_binary-2.9.3-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:b3a24a1982ae56461cc24f6680604fffa2c1b818e9dc55680da038792e004d18"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp38-cp38-win_amd64.whl", hash = "sha256:02c6e3cf3439e213e4ee930308dc122d6fb4d4bea9aef4a12535fbd605d1a2fe"},
|
{file = "psycopg2_binary-2.9.3-cp39-cp39-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:91920527dea30175cc02a1099f331aa8c1ba39bf8b7762b7b56cbf54bc5cce42"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:e9182eb20f41417ea1dd8e8f7888c4d7c6e805f8a7c98c1081778a3da2bee3e4"},
|
{file = "psycopg2_binary-2.9.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:887dd9aac71765ac0d0bac1d0d4b4f2c99d5f5c1382d8b770404f0f3d0ce8a39"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:8a6979cf527e2603d349a91060f428bcb135aea2be3201dff794813256c274f1"},
|
{file = "psycopg2_binary-2.9.3-cp39-cp39-manylinux_2_24_aarch64.whl", hash = "sha256:1f14c8b0942714eb3c74e1e71700cbbcb415acbc311c730370e70c578a44a25c"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8338a271cb71d8da40b023a35d9c1e919eba6cbd8fa20a54b748a332c355d896"},
|
{file = "psycopg2_binary-2.9.3-cp39-cp39-manylinux_2_24_ppc64le.whl", hash = "sha256:7af0dd86ddb2f8af5da57a976d27cd2cd15510518d582b478fbb2292428710b4"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp39-cp39-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e3ed340d2b858d6e6fb5083f87c09996506af483227735de6964a6100b4e6a54"},
|
{file = "psycopg2_binary-2.9.3-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:93cd1967a18aa0edd4b95b1dfd554cf15af657cb606280996d393dadc88c3c35"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:f81e65376e52f03422e1fb475c9514185669943798ed019ac50410fb4c4df232"},
|
{file = "psycopg2_binary-2.9.3-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:bda845b664bb6c91446ca9609fc69f7db6c334ec5e4adc87571c34e4f47b7ddb"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bfb13af3c5dd3a9588000910178de17010ebcccd37b4f9794b00595e3a8ddad3"},
|
{file = "psycopg2_binary-2.9.3-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:01310cf4cf26db9aea5158c217caa92d291f0500051a6469ac52166e1a16f5b7"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:4c727b597c6444a16e9119386b59388f8a424223302d0c06c676ec8b4bc1f963"},
|
{file = "psycopg2_binary-2.9.3-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:99485cab9ba0fa9b84f1f9e1fef106f44a46ef6afdeec8885e0b88d0772b49e8"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:4d67fbdaf177da06374473ef6f7ed8cc0a9dc640b01abfe9e8a2ccb1b1402c1f"},
|
{file = "psycopg2_binary-2.9.3-cp39-cp39-win32.whl", hash = "sha256:46f0e0a6b5fa5851bbd9ab1bc805eef362d3a230fbdfbc209f4a236d0a7a990d"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:0892ef645c2fabb0c75ec32d79f4252542d0caec1d5d949630e7d242ca4681a3"},
|
{file = "psycopg2_binary-2.9.3-cp39-cp39-win_amd64.whl", hash = "sha256:accfe7e982411da3178ec690baaceaad3c278652998b2c45828aaac66cd8285f"},
|
||||||
{file = "psycopg2_binary-2.9.6-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:02c0f3757a4300cf379eb49f543fb7ac527fb00144d39246ee40e1df684ab514"},
|
|
||||||
{file = "psycopg2_binary-2.9.6-cp39-cp39-win32.whl", hash = "sha256:c3dba7dab16709a33a847e5cd756767271697041fbe3fe97c215b1fc1f5c9848"},
|
|
||||||
{file = "psycopg2_binary-2.9.6-cp39-cp39-win_amd64.whl", hash = "sha256:f6a88f384335bb27812293fdb11ac6aee2ca3f51d3c7820fe03de0a304ab6249"},
|
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ license.workspace = true
|
|||||||
[dependencies]
|
[dependencies]
|
||||||
anyhow.workspace = true
|
anyhow.workspace = true
|
||||||
async-trait.workspace = true
|
async-trait.workspace = true
|
||||||
|
atty.workspace = true
|
||||||
base64.workspace = true
|
base64.workspace = true
|
||||||
bstr.workspace = true
|
bstr.workspace = true
|
||||||
bytes = { workspace = true, features = ["serde"] }
|
bytes = { workspace = true, features = ["serde"] }
|
||||||
@@ -29,7 +30,6 @@ metrics.workspace = true
|
|||||||
once_cell.workspace = true
|
once_cell.workspace = true
|
||||||
opentelemetry.workspace = true
|
opentelemetry.workspace = true
|
||||||
parking_lot.workspace = true
|
parking_lot.workspace = true
|
||||||
pbkdf2.workspace = true
|
|
||||||
pin-project-lite.workspace = true
|
pin-project-lite.workspace = true
|
||||||
postgres_backend.workspace = true
|
postgres_backend.workspace = true
|
||||||
pq_proto.workspace = true
|
pq_proto.workspace = true
|
||||||
@@ -38,7 +38,6 @@ rand.workspace = true
|
|||||||
regex.workspace = true
|
regex.workspace = true
|
||||||
reqwest = { workspace = true, features = ["json"] }
|
reqwest = { workspace = true, features = ["json"] }
|
||||||
reqwest-middleware.workspace = true
|
reqwest-middleware.workspace = true
|
||||||
reqwest-retry.workspace = true
|
|
||||||
reqwest-tracing.workspace = true
|
reqwest-tracing.workspace = true
|
||||||
routerify.workspace = true
|
routerify.workspace = true
|
||||||
rustls-pemfile.workspace = true
|
rustls-pemfile.workspace = true
|
||||||
|
|||||||
@@ -136,17 +136,18 @@ impl Default for ConnCfg {
|
|||||||
|
|
||||||
impl ConnCfg {
|
impl ConnCfg {
|
||||||
/// Establish a raw TCP connection to the compute node.
|
/// Establish a raw TCP connection to the compute node.
|
||||||
async fn connect_raw(&self, timeout: Duration) -> io::Result<(SocketAddr, TcpStream, &str)> {
|
async fn connect_raw(&self) -> io::Result<(SocketAddr, TcpStream, &str)> {
|
||||||
use tokio_postgres::config::Host;
|
use tokio_postgres::config::Host;
|
||||||
|
|
||||||
// wrap TcpStream::connect with timeout
|
// wrap TcpStream::connect with timeout
|
||||||
let connect_with_timeout = |host, port| {
|
let connect_with_timeout = |host, port| {
|
||||||
tokio::time::timeout(timeout, TcpStream::connect((host, port))).map(
|
let connection_timeout = Duration::from_millis(10000);
|
||||||
|
tokio::time::timeout(connection_timeout, TcpStream::connect((host, port))).map(
|
||||||
move |res| match res {
|
move |res| match res {
|
||||||
Ok(tcpstream_connect_res) => tcpstream_connect_res,
|
Ok(tcpstream_connect_res) => tcpstream_connect_res,
|
||||||
Err(_) => Err(io::Error::new(
|
Err(_) => Err(io::Error::new(
|
||||||
io::ErrorKind::TimedOut,
|
io::ErrorKind::TimedOut,
|
||||||
format!("exceeded connection timeout {timeout:?}"),
|
format!("exceeded connection timeout {connection_timeout:?}"),
|
||||||
)),
|
)),
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
@@ -222,9 +223,8 @@ impl ConnCfg {
|
|||||||
async fn do_connect(
|
async fn do_connect(
|
||||||
&self,
|
&self,
|
||||||
allow_self_signed_compute: bool,
|
allow_self_signed_compute: bool,
|
||||||
timeout: Duration,
|
|
||||||
) -> Result<PostgresConnection, ConnectionError> {
|
) -> Result<PostgresConnection, ConnectionError> {
|
||||||
let (socket_addr, stream, host) = self.connect_raw(timeout).await?;
|
let (socket_addr, stream, host) = self.connect_raw().await?;
|
||||||
|
|
||||||
let tls_connector = native_tls::TlsConnector::builder()
|
let tls_connector = native_tls::TlsConnector::builder()
|
||||||
.danger_accept_invalid_certs(allow_self_signed_compute)
|
.danger_accept_invalid_certs(allow_self_signed_compute)
|
||||||
@@ -264,9 +264,8 @@ impl ConnCfg {
|
|||||||
pub async fn connect(
|
pub async fn connect(
|
||||||
&self,
|
&self,
|
||||||
allow_self_signed_compute: bool,
|
allow_self_signed_compute: bool,
|
||||||
timeout: Duration,
|
|
||||||
) -> Result<PostgresConnection, ConnectionError> {
|
) -> Result<PostgresConnection, ConnectionError> {
|
||||||
self.do_connect(allow_self_signed_compute, timeout)
|
self.do_connect(allow_self_signed_compute)
|
||||||
.inspect_err(|err| {
|
.inspect_err(|err| {
|
||||||
// Immediately log the error we have at our disposal.
|
// Immediately log the error we have at our disposal.
|
||||||
error!("couldn't connect to compute node: {err}");
|
error!("couldn't connect to compute node: {err}");
|
||||||
|
|||||||
@@ -212,7 +212,7 @@ pub struct CacheOptions {
|
|||||||
|
|
||||||
impl CacheOptions {
|
impl CacheOptions {
|
||||||
/// Default options for [`crate::auth::caches::NodeInfoCache`].
|
/// Default options for [`crate::auth::caches::NodeInfoCache`].
|
||||||
pub const DEFAULT_OPTIONS_NODE_INFO: &str = "size=4000,ttl=4m";
|
pub const DEFAULT_OPTIONS_NODE_INFO: &str = "size=4000,ttl=5m";
|
||||||
|
|
||||||
/// Parse cache options passed via cmdline.
|
/// Parse cache options passed via cmdline.
|
||||||
/// Example: [`Self::DEFAULT_OPTIONS_NODE_INFO`].
|
/// Example: [`Self::DEFAULT_OPTIONS_NODE_INFO`].
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ use postgres_backend::{self, AuthType, PostgresBackend, PostgresBackendTCP, Quer
|
|||||||
use pq_proto::{BeMessage, SINGLE_COL_ROWDESC};
|
use pq_proto::{BeMessage, SINGLE_COL_ROWDESC};
|
||||||
use std::future;
|
use std::future;
|
||||||
use tokio::net::{TcpListener, TcpStream};
|
use tokio::net::{TcpListener, TcpStream};
|
||||||
use tracing::{error, info, info_span, Instrument};
|
use tracing::{error, info, info_span};
|
||||||
|
|
||||||
static CPLANE_WAITERS: Lazy<Waiters<ComputeReady>> = Lazy::new(Default::default);
|
static CPLANE_WAITERS: Lazy<Waiters<ComputeReady>> = Lazy::new(Default::default);
|
||||||
|
|
||||||
@@ -44,30 +44,19 @@ pub async fn task_main(listener: TcpListener) -> anyhow::Result<()> {
|
|||||||
.set_nodelay(true)
|
.set_nodelay(true)
|
||||||
.context("failed to set client socket option")?;
|
.context("failed to set client socket option")?;
|
||||||
|
|
||||||
let span = info_span!("mgmt", peer = %peer_addr);
|
tokio::task::spawn(async move {
|
||||||
|
let span = info_span!("mgmt", peer = %peer_addr);
|
||||||
|
let _enter = span.enter();
|
||||||
|
|
||||||
tokio::task::spawn(
|
info!("started a new console management API thread");
|
||||||
async move {
|
scopeguard::defer! {
|
||||||
info!("serving a new console management API connection");
|
info!("console management API thread is about to finish");
|
||||||
|
|
||||||
// these might be long running connections, have a separate logging for cancelling
|
|
||||||
// on shutdown and other ways of stopping.
|
|
||||||
let cancelled = scopeguard::guard(tracing::Span::current(), |span| {
|
|
||||||
let _e = span.entered();
|
|
||||||
info!("console management API task cancelled");
|
|
||||||
});
|
|
||||||
|
|
||||||
if let Err(e) = handle_connection(socket).await {
|
|
||||||
error!("serving failed with an error: {e}");
|
|
||||||
} else {
|
|
||||||
info!("serving completed");
|
|
||||||
}
|
|
||||||
|
|
||||||
// we can no longer get dropped
|
|
||||||
scopeguard::ScopeGuard::into_inner(cancelled);
|
|
||||||
}
|
}
|
||||||
.instrument(span),
|
|
||||||
);
|
if let Err(e) = handle_connection(socket).await {
|
||||||
|
error!("thread failed with an error: {e}");
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -88,14 +77,14 @@ impl postgres_backend::Handler<tokio::net::TcpStream> for MgmtHandler {
|
|||||||
pgb: &mut PostgresBackendTCP,
|
pgb: &mut PostgresBackendTCP,
|
||||||
query: &str,
|
query: &str,
|
||||||
) -> Result<(), QueryError> {
|
) -> Result<(), QueryError> {
|
||||||
try_process_query(pgb, query).map_err(|e| {
|
try_process_query(pgb, query).await.map_err(|e| {
|
||||||
error!("failed to process response: {e:?}");
|
error!("failed to process response: {e:?}");
|
||||||
e
|
e
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn try_process_query(pgb: &mut PostgresBackendTCP, query: &str) -> Result<(), QueryError> {
|
async fn try_process_query(pgb: &mut PostgresBackendTCP, query: &str) -> Result<(), QueryError> {
|
||||||
let resp: KickSession = serde_json::from_str(query).context("Failed to parse query as json")?;
|
let resp: KickSession = serde_json::from_str(query).context("Failed to parse query as json")?;
|
||||||
|
|
||||||
let span = info_span!("event", session_id = resp.session_id);
|
let span = info_span!("event", session_id = resp.session_id);
|
||||||
|
|||||||
@@ -2,16 +2,12 @@
|
|||||||
//! Other modules should use stuff from this module instead of
|
//! Other modules should use stuff from this module instead of
|
||||||
//! directly relying on deps like `reqwest` (think loose coupling).
|
//! directly relying on deps like `reqwest` (think loose coupling).
|
||||||
|
|
||||||
pub mod conn_pool;
|
|
||||||
pub mod server;
|
pub mod server;
|
||||||
pub mod sql_over_http;
|
pub mod sql_over_http;
|
||||||
pub mod websocket;
|
pub mod websocket;
|
||||||
|
|
||||||
use std::time::Duration;
|
|
||||||
|
|
||||||
pub use reqwest::{Request, Response, StatusCode};
|
pub use reqwest::{Request, Response, StatusCode};
|
||||||
pub use reqwest_middleware::{ClientWithMiddleware, Error};
|
pub use reqwest_middleware::{ClientWithMiddleware, Error};
|
||||||
pub use reqwest_retry::{policies::ExponentialBackoff, RetryTransientMiddleware};
|
|
||||||
|
|
||||||
use crate::url::ApiUrl;
|
use crate::url::ApiUrl;
|
||||||
use reqwest_middleware::RequestBuilder;
|
use reqwest_middleware::RequestBuilder;
|
||||||
@@ -25,24 +21,6 @@ pub fn new_client() -> ClientWithMiddleware {
|
|||||||
.build()
|
.build()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn new_client_with_timeout(default_timout: Duration) -> ClientWithMiddleware {
|
|
||||||
let timeout_client = reqwest::ClientBuilder::new()
|
|
||||||
.timeout(default_timout)
|
|
||||||
.build()
|
|
||||||
.expect("Failed to create http client with timeout");
|
|
||||||
|
|
||||||
let retry_policy =
|
|
||||||
ExponentialBackoff::builder().build_with_total_retry_duration(default_timout);
|
|
||||||
|
|
||||||
reqwest_middleware::ClientBuilder::new(timeout_client)
|
|
||||||
.with(reqwest_tracing::TracingMiddleware::default())
|
|
||||||
// As per docs, "This middleware always errors when given requests with streaming bodies".
|
|
||||||
// That's all right because we only use this client to send `serde_json::RawValue`, which
|
|
||||||
// is not a stream.
|
|
||||||
.with(RetryTransientMiddleware::new_with_policy(retry_policy))
|
|
||||||
.build()
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Thin convenience wrapper for an API provided by an http endpoint.
|
/// Thin convenience wrapper for an API provided by an http endpoint.
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
pub struct Endpoint {
|
pub struct Endpoint {
|
||||||
|
|||||||
@@ -1,278 +0,0 @@
|
|||||||
use parking_lot::Mutex;
|
|
||||||
use pq_proto::StartupMessageParams;
|
|
||||||
use std::fmt;
|
|
||||||
use std::{collections::HashMap, sync::Arc};
|
|
||||||
|
|
||||||
use futures::TryFutureExt;
|
|
||||||
|
|
||||||
use crate::config;
|
|
||||||
use crate::{auth, console};
|
|
||||||
|
|
||||||
use super::sql_over_http::MAX_RESPONSE_SIZE;
|
|
||||||
|
|
||||||
use crate::proxy::invalidate_cache;
|
|
||||||
use crate::proxy::NUM_RETRIES_WAKE_COMPUTE;
|
|
||||||
|
|
||||||
use tracing::error;
|
|
||||||
use tracing::info;
|
|
||||||
|
|
||||||
pub const APP_NAME: &str = "sql_over_http";
|
|
||||||
const MAX_CONNS_PER_ENDPOINT: usize = 20;
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
|
||||||
pub struct ConnInfo {
|
|
||||||
pub username: String,
|
|
||||||
pub dbname: String,
|
|
||||||
pub hostname: String,
|
|
||||||
pub password: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ConnInfo {
|
|
||||||
// hm, change to hasher to avoid cloning?
|
|
||||||
pub fn db_and_user(&self) -> (String, String) {
|
|
||||||
(self.dbname.clone(), self.username.clone())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl fmt::Display for ConnInfo {
|
|
||||||
// use custom display to avoid logging password
|
|
||||||
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
||||||
write!(f, "{}@{}/{}", self.username, self.hostname, self.dbname)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
struct ConnPoolEntry {
|
|
||||||
conn: tokio_postgres::Client,
|
|
||||||
_last_access: std::time::Instant,
|
|
||||||
}
|
|
||||||
|
|
||||||
// Per-endpoint connection pool, (dbname, username) -> Vec<ConnPoolEntry>
|
|
||||||
// Number of open connections is limited by the `max_conns_per_endpoint`.
|
|
||||||
pub struct EndpointConnPool {
|
|
||||||
pools: HashMap<(String, String), Vec<ConnPoolEntry>>,
|
|
||||||
total_conns: usize,
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct GlobalConnPool {
|
|
||||||
// endpoint -> per-endpoint connection pool
|
|
||||||
//
|
|
||||||
// That should be a fairly conteded map, so return reference to the per-endpoint
|
|
||||||
// pool as early as possible and release the lock.
|
|
||||||
global_pool: Mutex<HashMap<String, Arc<Mutex<EndpointConnPool>>>>,
|
|
||||||
|
|
||||||
// Maximum number of connections per one endpoint.
|
|
||||||
// Can mix different (dbname, username) connections.
|
|
||||||
// When running out of free slots for a particular endpoint,
|
|
||||||
// falls back to opening a new connection for each request.
|
|
||||||
max_conns_per_endpoint: usize,
|
|
||||||
|
|
||||||
proxy_config: &'static crate::config::ProxyConfig,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl GlobalConnPool {
|
|
||||||
pub fn new(config: &'static crate::config::ProxyConfig) -> Arc<Self> {
|
|
||||||
Arc::new(Self {
|
|
||||||
global_pool: Mutex::new(HashMap::new()),
|
|
||||||
max_conns_per_endpoint: MAX_CONNS_PER_ENDPOINT,
|
|
||||||
proxy_config: config,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn get(
|
|
||||||
&self,
|
|
||||||
conn_info: &ConnInfo,
|
|
||||||
force_new: bool,
|
|
||||||
) -> anyhow::Result<tokio_postgres::Client> {
|
|
||||||
let mut client: Option<tokio_postgres::Client> = None;
|
|
||||||
|
|
||||||
if !force_new {
|
|
||||||
let pool = self.get_endpoint_pool(&conn_info.hostname).await;
|
|
||||||
|
|
||||||
// find a pool entry by (dbname, username) if exists
|
|
||||||
let mut pool = pool.lock();
|
|
||||||
let pool_entries = pool.pools.get_mut(&conn_info.db_and_user());
|
|
||||||
if let Some(pool_entries) = pool_entries {
|
|
||||||
if let Some(entry) = pool_entries.pop() {
|
|
||||||
client = Some(entry.conn);
|
|
||||||
pool.total_conns -= 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// ok return cached connection if found and establish a new one otherwise
|
|
||||||
if let Some(client) = client {
|
|
||||||
if client.is_closed() {
|
|
||||||
info!("pool: cached connection '{conn_info}' is closed, opening a new one");
|
|
||||||
connect_to_compute(self.proxy_config, conn_info).await
|
|
||||||
} else {
|
|
||||||
info!("pool: reusing connection '{conn_info}'");
|
|
||||||
Ok(client)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
info!("pool: opening a new connection '{conn_info}'");
|
|
||||||
connect_to_compute(self.proxy_config, conn_info).await
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn put(
|
|
||||||
&self,
|
|
||||||
conn_info: &ConnInfo,
|
|
||||||
client: tokio_postgres::Client,
|
|
||||||
) -> anyhow::Result<()> {
|
|
||||||
let pool = self.get_endpoint_pool(&conn_info.hostname).await;
|
|
||||||
|
|
||||||
// return connection to the pool
|
|
||||||
let mut total_conns;
|
|
||||||
let mut returned = false;
|
|
||||||
let mut per_db_size = 0;
|
|
||||||
{
|
|
||||||
let mut pool = pool.lock();
|
|
||||||
total_conns = pool.total_conns;
|
|
||||||
|
|
||||||
let pool_entries: &mut Vec<ConnPoolEntry> = pool
|
|
||||||
.pools
|
|
||||||
.entry(conn_info.db_and_user())
|
|
||||||
.or_insert_with(|| Vec::with_capacity(1));
|
|
||||||
if total_conns < self.max_conns_per_endpoint {
|
|
||||||
pool_entries.push(ConnPoolEntry {
|
|
||||||
conn: client,
|
|
||||||
_last_access: std::time::Instant::now(),
|
|
||||||
});
|
|
||||||
|
|
||||||
total_conns += 1;
|
|
||||||
returned = true;
|
|
||||||
per_db_size = pool_entries.len();
|
|
||||||
|
|
||||||
pool.total_conns += 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// do logging outside of the mutex
|
|
||||||
if returned {
|
|
||||||
info!("pool: returning connection '{conn_info}' back to the pool, total_conns={total_conns}, for this (db, user)={per_db_size}");
|
|
||||||
} else {
|
|
||||||
info!("pool: throwing away connection '{conn_info}' because pool is full, total_conns={total_conns}");
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn get_endpoint_pool(&self, endpoint: &String) -> Arc<Mutex<EndpointConnPool>> {
|
|
||||||
// find or create a pool for this endpoint
|
|
||||||
let mut created = false;
|
|
||||||
let mut global_pool = self.global_pool.lock();
|
|
||||||
let pool = global_pool
|
|
||||||
.entry(endpoint.clone())
|
|
||||||
.or_insert_with(|| {
|
|
||||||
created = true;
|
|
||||||
Arc::new(Mutex::new(EndpointConnPool {
|
|
||||||
pools: HashMap::new(),
|
|
||||||
total_conns: 0,
|
|
||||||
}))
|
|
||||||
})
|
|
||||||
.clone();
|
|
||||||
let global_pool_size = global_pool.len();
|
|
||||||
drop(global_pool);
|
|
||||||
|
|
||||||
// log new global pool size
|
|
||||||
if created {
|
|
||||||
info!(
|
|
||||||
"pool: created new pool for '{endpoint}', global pool size now {global_pool_size}"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
pool
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
|
||||||
// Wake up the destination if needed. Code here is a bit involved because
|
|
||||||
// we reuse the code from the usual proxy and we need to prepare few structures
|
|
||||||
// that this code expects.
|
|
||||||
//
|
|
||||||
async fn connect_to_compute(
|
|
||||||
config: &config::ProxyConfig,
|
|
||||||
conn_info: &ConnInfo,
|
|
||||||
) -> anyhow::Result<tokio_postgres::Client> {
|
|
||||||
let tls = config.tls_config.as_ref();
|
|
||||||
let common_names = tls.and_then(|tls| tls.common_names.clone());
|
|
||||||
|
|
||||||
let credential_params = StartupMessageParams::new([
|
|
||||||
("user", &conn_info.username),
|
|
||||||
("database", &conn_info.dbname),
|
|
||||||
("application_name", APP_NAME),
|
|
||||||
]);
|
|
||||||
|
|
||||||
let creds = config
|
|
||||||
.auth_backend
|
|
||||||
.as_ref()
|
|
||||||
.map(|_| {
|
|
||||||
auth::ClientCredentials::parse(
|
|
||||||
&credential_params,
|
|
||||||
Some(&conn_info.hostname),
|
|
||||||
common_names,
|
|
||||||
)
|
|
||||||
})
|
|
||||||
.transpose()?;
|
|
||||||
let extra = console::ConsoleReqExtra {
|
|
||||||
session_id: uuid::Uuid::new_v4(),
|
|
||||||
application_name: Some(APP_NAME),
|
|
||||||
};
|
|
||||||
|
|
||||||
let node_info = &mut creds.wake_compute(&extra).await?.expect("msg");
|
|
||||||
|
|
||||||
// This code is a copy of `connect_to_compute` from `src/proxy.rs` with
|
|
||||||
// the difference that it uses `tokio_postgres` for the connection.
|
|
||||||
let mut num_retries: usize = NUM_RETRIES_WAKE_COMPUTE;
|
|
||||||
loop {
|
|
||||||
match connect_to_compute_once(node_info, conn_info).await {
|
|
||||||
Err(e) if num_retries > 0 => {
|
|
||||||
info!("compute node's state has changed; requesting a wake-up");
|
|
||||||
match creds.wake_compute(&extra).await? {
|
|
||||||
// Update `node_info` and try one more time.
|
|
||||||
Some(new) => {
|
|
||||||
*node_info = new;
|
|
||||||
}
|
|
||||||
// Link auth doesn't work that way, so we just exit.
|
|
||||||
None => return Err(e),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
other => return other,
|
|
||||||
}
|
|
||||||
|
|
||||||
num_retries -= 1;
|
|
||||||
info!("retrying after wake-up ({num_retries} attempts left)");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn connect_to_compute_once(
|
|
||||||
node_info: &console::CachedNodeInfo,
|
|
||||||
conn_info: &ConnInfo,
|
|
||||||
) -> anyhow::Result<tokio_postgres::Client> {
|
|
||||||
let mut config = (*node_info.config).clone();
|
|
||||||
|
|
||||||
let (client, connection) = config
|
|
||||||
.user(&conn_info.username)
|
|
||||||
.password(&conn_info.password)
|
|
||||||
.dbname(&conn_info.dbname)
|
|
||||||
.max_backend_message_size(MAX_RESPONSE_SIZE)
|
|
||||||
.connect(tokio_postgres::NoTls)
|
|
||||||
.inspect_err(|e: &tokio_postgres::Error| {
|
|
||||||
error!(
|
|
||||||
"failed to connect to compute node hosts={:?} ports={:?}: {}",
|
|
||||||
node_info.config.get_hosts(),
|
|
||||||
node_info.config.get_ports(),
|
|
||||||
e
|
|
||||||
);
|
|
||||||
invalidate_cache(node_info)
|
|
||||||
})
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
tokio::spawn(async move {
|
|
||||||
if let Err(e) = connection.await {
|
|
||||||
error!("connection error: {}", e);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
Ok(client)
|
|
||||||
}
|
|
||||||
@@ -1,21 +1,25 @@
|
|||||||
use std::sync::Arc;
|
|
||||||
|
|
||||||
use futures::pin_mut;
|
use futures::pin_mut;
|
||||||
use futures::StreamExt;
|
use futures::StreamExt;
|
||||||
|
use futures::TryFutureExt;
|
||||||
use hyper::body::HttpBody;
|
use hyper::body::HttpBody;
|
||||||
use hyper::http::HeaderName;
|
use hyper::http::HeaderName;
|
||||||
use hyper::http::HeaderValue;
|
use hyper::http::HeaderValue;
|
||||||
use hyper::{Body, HeaderMap, Request};
|
use hyper::{Body, HeaderMap, Request};
|
||||||
|
use pq_proto::StartupMessageParams;
|
||||||
use serde_json::json;
|
use serde_json::json;
|
||||||
use serde_json::Map;
|
use serde_json::Map;
|
||||||
use serde_json::Value;
|
use serde_json::Value;
|
||||||
use tokio_postgres::types::Kind;
|
use tokio_postgres::types::Kind;
|
||||||
use tokio_postgres::types::Type;
|
use tokio_postgres::types::Type;
|
||||||
use tokio_postgres::Row;
|
use tokio_postgres::Row;
|
||||||
|
use tracing::error;
|
||||||
|
use tracing::info;
|
||||||
|
use tracing::instrument;
|
||||||
use url::Url;
|
use url::Url;
|
||||||
|
|
||||||
use super::conn_pool::ConnInfo;
|
use crate::proxy::invalidate_cache;
|
||||||
use super::conn_pool::GlobalConnPool;
|
use crate::proxy::NUM_RETRIES_WAKE_COMPUTE;
|
||||||
|
use crate::{auth, config::ProxyConfig, console};
|
||||||
|
|
||||||
#[derive(serde::Deserialize)]
|
#[derive(serde::Deserialize)]
|
||||||
struct QueryData {
|
struct QueryData {
|
||||||
@@ -23,13 +27,12 @@ struct QueryData {
|
|||||||
params: Vec<serde_json::Value>,
|
params: Vec<serde_json::Value>,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub const MAX_RESPONSE_SIZE: usize = 1024 * 1024; // 1 MB
|
const APP_NAME: &str = "sql_over_http";
|
||||||
|
const MAX_RESPONSE_SIZE: usize = 1024 * 1024; // 1 MB
|
||||||
const MAX_REQUEST_SIZE: u64 = 1024 * 1024; // 1 MB
|
const MAX_REQUEST_SIZE: u64 = 1024 * 1024; // 1 MB
|
||||||
|
|
||||||
static RAW_TEXT_OUTPUT: HeaderName = HeaderName::from_static("neon-raw-text-output");
|
static RAW_TEXT_OUTPUT: HeaderName = HeaderName::from_static("neon-raw-text-output");
|
||||||
static ARRAY_MODE: HeaderName = HeaderName::from_static("neon-array-mode");
|
static ARRAY_MODE: HeaderName = HeaderName::from_static("neon-array-mode");
|
||||||
static ALLOW_POOL: HeaderName = HeaderName::from_static("neon-pool-opt-in");
|
|
||||||
|
|
||||||
static HEADER_VALUE_TRUE: HeaderValue = HeaderValue::from_static("true");
|
static HEADER_VALUE_TRUE: HeaderValue = HeaderValue::from_static("true");
|
||||||
|
|
||||||
//
|
//
|
||||||
@@ -93,6 +96,13 @@ fn json_array_to_pg_array(value: &Value) -> Result<Option<String>, serde_json::E
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct ConnInfo {
|
||||||
|
username: String,
|
||||||
|
dbname: String,
|
||||||
|
hostname: String,
|
||||||
|
password: String,
|
||||||
|
}
|
||||||
|
|
||||||
fn get_conn_info(
|
fn get_conn_info(
|
||||||
headers: &HeaderMap,
|
headers: &HeaderMap,
|
||||||
sni_hostname: Option<String>,
|
sni_hostname: Option<String>,
|
||||||
@@ -159,23 +169,50 @@ fn get_conn_info(
|
|||||||
|
|
||||||
// TODO: return different http error codes
|
// TODO: return different http error codes
|
||||||
pub async fn handle(
|
pub async fn handle(
|
||||||
|
config: &'static ProxyConfig,
|
||||||
request: Request<Body>,
|
request: Request<Body>,
|
||||||
sni_hostname: Option<String>,
|
sni_hostname: Option<String>,
|
||||||
conn_pool: Arc<GlobalConnPool>,
|
|
||||||
) -> anyhow::Result<Value> {
|
) -> anyhow::Result<Value> {
|
||||||
//
|
//
|
||||||
// Determine the destination and connection params
|
// Determine the destination and connection params
|
||||||
//
|
//
|
||||||
let headers = request.headers();
|
let headers = request.headers();
|
||||||
let conn_info = get_conn_info(headers, sni_hostname)?;
|
let conn_info = get_conn_info(headers, sni_hostname)?;
|
||||||
|
let credential_params = StartupMessageParams::new([
|
||||||
|
("user", &conn_info.username),
|
||||||
|
("database", &conn_info.dbname),
|
||||||
|
("application_name", APP_NAME),
|
||||||
|
]);
|
||||||
|
|
||||||
// Determine the output options. Default behaviour is 'false'. Anything that is not
|
// Determine the output options. Default behaviour is 'false'. Anything that is not
|
||||||
// strictly 'true' assumed to be false.
|
// strictly 'true' assumed to be false.
|
||||||
let raw_output = headers.get(&RAW_TEXT_OUTPUT) == Some(&HEADER_VALUE_TRUE);
|
let raw_output = headers.get(&RAW_TEXT_OUTPUT) == Some(&HEADER_VALUE_TRUE);
|
||||||
let array_mode = headers.get(&ARRAY_MODE) == Some(&HEADER_VALUE_TRUE);
|
let array_mode = headers.get(&ARRAY_MODE) == Some(&HEADER_VALUE_TRUE);
|
||||||
|
|
||||||
// Allow connection pooling only if explicitly requested
|
//
|
||||||
let allow_pool = headers.get(&ALLOW_POOL) == Some(&HEADER_VALUE_TRUE);
|
// Wake up the destination if needed. Code here is a bit involved because
|
||||||
|
// we reuse the code from the usual proxy and we need to prepare few structures
|
||||||
|
// that this code expects.
|
||||||
|
//
|
||||||
|
let tls = config.tls_config.as_ref();
|
||||||
|
let common_names = tls.and_then(|tls| tls.common_names.clone());
|
||||||
|
let creds = config
|
||||||
|
.auth_backend
|
||||||
|
.as_ref()
|
||||||
|
.map(|_| {
|
||||||
|
auth::ClientCredentials::parse(
|
||||||
|
&credential_params,
|
||||||
|
Some(&conn_info.hostname),
|
||||||
|
common_names,
|
||||||
|
)
|
||||||
|
})
|
||||||
|
.transpose()?;
|
||||||
|
let extra = console::ConsoleReqExtra {
|
||||||
|
session_id: uuid::Uuid::new_v4(),
|
||||||
|
application_name: Some(APP_NAME),
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut node_info = creds.wake_compute(&extra).await?.expect("msg");
|
||||||
|
|
||||||
let request_content_length = match request.body().size_hint().upper() {
|
let request_content_length = match request.body().size_hint().upper() {
|
||||||
Some(v) => v,
|
Some(v) => v,
|
||||||
@@ -198,8 +235,7 @@ pub async fn handle(
|
|||||||
//
|
//
|
||||||
// Now execute the query and return the result
|
// Now execute the query and return the result
|
||||||
//
|
//
|
||||||
let client = conn_pool.get(&conn_info, !allow_pool).await?;
|
let client = connect_to_compute(&mut node_info, &extra, &creds, &conn_info).await?;
|
||||||
|
|
||||||
let row_stream = client.query_raw_txt(query, query_params).await?;
|
let row_stream = client.query_raw_txt(query, query_params).await?;
|
||||||
|
|
||||||
// Manually drain the stream into a vector to leave row_stream hanging
|
// Manually drain the stream into a vector to leave row_stream hanging
|
||||||
@@ -256,13 +292,6 @@ pub async fn handle(
|
|||||||
.map(|row| pg_text_row_to_json(row, raw_output, array_mode))
|
.map(|row| pg_text_row_to_json(row, raw_output, array_mode))
|
||||||
.collect::<Result<Vec<_>, _>>()?;
|
.collect::<Result<Vec<_>, _>>()?;
|
||||||
|
|
||||||
if allow_pool {
|
|
||||||
// return connection to the pool
|
|
||||||
tokio::task::spawn(async move {
|
|
||||||
let _ = conn_pool.put(&conn_info, client).await;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// resulting JSON format is based on the format of node-postgres result
|
// resulting JSON format is based on the format of node-postgres result
|
||||||
Ok(json!({
|
Ok(json!({
|
||||||
"command": command_tag_name,
|
"command": command_tag_name,
|
||||||
@@ -273,6 +302,70 @@ pub async fn handle(
|
|||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// This function is a copy of `connect_to_compute` from `src/proxy.rs` with
|
||||||
|
/// the difference that it uses `tokio_postgres` for the connection.
|
||||||
|
#[instrument(skip_all)]
|
||||||
|
async fn connect_to_compute(
|
||||||
|
node_info: &mut console::CachedNodeInfo,
|
||||||
|
extra: &console::ConsoleReqExtra<'_>,
|
||||||
|
creds: &auth::BackendType<'_, auth::ClientCredentials<'_>>,
|
||||||
|
conn_info: &ConnInfo,
|
||||||
|
) -> anyhow::Result<tokio_postgres::Client> {
|
||||||
|
let mut num_retries: usize = NUM_RETRIES_WAKE_COMPUTE;
|
||||||
|
|
||||||
|
loop {
|
||||||
|
match connect_to_compute_once(node_info, conn_info).await {
|
||||||
|
Err(e) if num_retries > 0 => {
|
||||||
|
info!("compute node's state has changed; requesting a wake-up");
|
||||||
|
match creds.wake_compute(extra).await? {
|
||||||
|
// Update `node_info` and try one more time.
|
||||||
|
Some(new) => {
|
||||||
|
*node_info = new;
|
||||||
|
}
|
||||||
|
// Link auth doesn't work that way, so we just exit.
|
||||||
|
None => return Err(e),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
other => return other,
|
||||||
|
}
|
||||||
|
|
||||||
|
num_retries -= 1;
|
||||||
|
info!("retrying after wake-up ({num_retries} attempts left)");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn connect_to_compute_once(
|
||||||
|
node_info: &console::CachedNodeInfo,
|
||||||
|
conn_info: &ConnInfo,
|
||||||
|
) -> anyhow::Result<tokio_postgres::Client> {
|
||||||
|
let mut config = (*node_info.config).clone();
|
||||||
|
|
||||||
|
let (client, connection) = config
|
||||||
|
.user(&conn_info.username)
|
||||||
|
.password(&conn_info.password)
|
||||||
|
.dbname(&conn_info.dbname)
|
||||||
|
.max_backend_message_size(MAX_RESPONSE_SIZE)
|
||||||
|
.connect(tokio_postgres::NoTls)
|
||||||
|
.inspect_err(|e: &tokio_postgres::Error| {
|
||||||
|
error!(
|
||||||
|
"failed to connect to compute node hosts={:?} ports={:?}: {}",
|
||||||
|
node_info.config.get_hosts(),
|
||||||
|
node_info.config.get_ports(),
|
||||||
|
e
|
||||||
|
);
|
||||||
|
invalidate_cache(node_info)
|
||||||
|
})
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
tokio::spawn(async move {
|
||||||
|
if let Err(e) = connection.await {
|
||||||
|
error!("connection error: {}", e);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
Ok(client)
|
||||||
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
// Convert postgres row with text-encoded values to JSON object
|
// Convert postgres row with text-encoded values to JSON object
|
||||||
//
|
//
|
||||||
|
|||||||
@@ -35,7 +35,7 @@ use utils::http::{error::ApiError, json::json_response};
|
|||||||
// Tracking issue: https://github.com/rust-lang/rust/issues/98407.
|
// Tracking issue: https://github.com/rust-lang/rust/issues/98407.
|
||||||
use sync_wrapper::SyncWrapper;
|
use sync_wrapper::SyncWrapper;
|
||||||
|
|
||||||
use super::{conn_pool::GlobalConnPool, sql_over_http};
|
use super::sql_over_http;
|
||||||
|
|
||||||
pin_project! {
|
pin_project! {
|
||||||
/// This is a wrapper around a [`WebSocketStream`] that
|
/// This is a wrapper around a [`WebSocketStream`] that
|
||||||
@@ -164,7 +164,6 @@ async fn serve_websocket(
|
|||||||
async fn ws_handler(
|
async fn ws_handler(
|
||||||
mut request: Request<Body>,
|
mut request: Request<Body>,
|
||||||
config: &'static ProxyConfig,
|
config: &'static ProxyConfig,
|
||||||
conn_pool: Arc<GlobalConnPool>,
|
|
||||||
cancel_map: Arc<CancelMap>,
|
cancel_map: Arc<CancelMap>,
|
||||||
session_id: uuid::Uuid,
|
session_id: uuid::Uuid,
|
||||||
sni_hostname: Option<String>,
|
sni_hostname: Option<String>,
|
||||||
@@ -193,7 +192,7 @@ async fn ws_handler(
|
|||||||
// TODO: that deserves a refactor as now this function also handles http json client besides websockets.
|
// TODO: that deserves a refactor as now this function also handles http json client besides websockets.
|
||||||
// Right now I don't want to blow up sql-over-http patch with file renames and do that as a follow up instead.
|
// Right now I don't want to blow up sql-over-http patch with file renames and do that as a follow up instead.
|
||||||
} else if request.uri().path() == "/sql" && request.method() == Method::POST {
|
} else if request.uri().path() == "/sql" && request.method() == Method::POST {
|
||||||
let result = sql_over_http::handle(request, sni_hostname, conn_pool)
|
let result = sql_over_http::handle(config, request, sni_hostname)
|
||||||
.instrument(info_span!("sql-over-http"))
|
.instrument(info_span!("sql-over-http"))
|
||||||
.await;
|
.await;
|
||||||
let status_code = match result {
|
let status_code = match result {
|
||||||
@@ -235,8 +234,6 @@ pub async fn task_main(
|
|||||||
info!("websocket server has shut down");
|
info!("websocket server has shut down");
|
||||||
}
|
}
|
||||||
|
|
||||||
let conn_pool: Arc<GlobalConnPool> = GlobalConnPool::new(config);
|
|
||||||
|
|
||||||
let tls_config = config.tls_config.as_ref().map(|cfg| cfg.to_server_config());
|
let tls_config = config.tls_config.as_ref().map(|cfg| cfg.to_server_config());
|
||||||
let tls_acceptor: tokio_rustls::TlsAcceptor = match tls_config {
|
let tls_acceptor: tokio_rustls::TlsAcceptor = match tls_config {
|
||||||
Some(config) => config.into(),
|
Some(config) => config.into(),
|
||||||
@@ -261,18 +258,15 @@ pub async fn task_main(
|
|||||||
let make_svc =
|
let make_svc =
|
||||||
hyper::service::make_service_fn(|stream: &tokio_rustls::server::TlsStream<AddrStream>| {
|
hyper::service::make_service_fn(|stream: &tokio_rustls::server::TlsStream<AddrStream>| {
|
||||||
let sni_name = stream.get_ref().1.sni_hostname().map(|s| s.to_string());
|
let sni_name = stream.get_ref().1.sni_hostname().map(|s| s.to_string());
|
||||||
let conn_pool = conn_pool.clone();
|
|
||||||
|
|
||||||
async move {
|
async move {
|
||||||
Ok::<_, Infallible>(hyper::service::service_fn(move |req: Request<Body>| {
|
Ok::<_, Infallible>(hyper::service::service_fn(move |req: Request<Body>| {
|
||||||
let sni_name = sni_name.clone();
|
let sni_name = sni_name.clone();
|
||||||
let conn_pool = conn_pool.clone();
|
|
||||||
|
|
||||||
async move {
|
async move {
|
||||||
let cancel_map = Arc::new(CancelMap::default());
|
let cancel_map = Arc::new(CancelMap::default());
|
||||||
let session_id = uuid::Uuid::new_v4();
|
let session_id = uuid::Uuid::new_v4();
|
||||||
|
|
||||||
ws_handler(req, config, conn_pool, cancel_map, session_id, sni_name)
|
ws_handler(req, config, cancel_map, session_id, sni_name)
|
||||||
.instrument(info_span!(
|
.instrument(info_span!(
|
||||||
"ws-client",
|
"ws-client",
|
||||||
session = format_args!("{session_id}")
|
session = format_args!("{session_id}")
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ pub async fn init() -> anyhow::Result<LoggingGuard> {
|
|||||||
.from_env_lossy();
|
.from_env_lossy();
|
||||||
|
|
||||||
let fmt_layer = tracing_subscriber::fmt::layer()
|
let fmt_layer = tracing_subscriber::fmt::layer()
|
||||||
.with_ansi(false)
|
.with_ansi(atty::is(atty::Stream::Stderr))
|
||||||
.with_writer(std::io::stderr)
|
.with_writer(std::io::stderr)
|
||||||
.with_target(false);
|
.with_target(false);
|
||||||
|
|
||||||
|
|||||||
@@ -4,13 +4,11 @@ use crate::{config::MetricCollectionConfig, http};
|
|||||||
use chrono::{DateTime, Utc};
|
use chrono::{DateTime, Utc};
|
||||||
use consumption_metrics::{idempotency_key, Event, EventChunk, EventType, CHUNK_SIZE};
|
use consumption_metrics::{idempotency_key, Event, EventChunk, EventType, CHUNK_SIZE};
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use std::{collections::HashMap, time::Duration};
|
use std::collections::HashMap;
|
||||||
use tracing::{error, info, instrument, trace, warn};
|
use tracing::{error, info, instrument, trace, warn};
|
||||||
|
|
||||||
const PROXY_IO_BYTES_PER_CLIENT: &str = "proxy_io_bytes_per_client";
|
const PROXY_IO_BYTES_PER_CLIENT: &str = "proxy_io_bytes_per_client";
|
||||||
|
|
||||||
const DEFAULT_HTTP_REPORTING_TIMEOUT: Duration = Duration::from_secs(60);
|
|
||||||
|
|
||||||
///
|
///
|
||||||
/// Key that uniquely identifies the object, this metric describes.
|
/// Key that uniquely identifies the object, this metric describes.
|
||||||
/// Currently, endpoint_id is enough, but this may change later,
|
/// Currently, endpoint_id is enough, but this may change later,
|
||||||
@@ -32,7 +30,7 @@ pub async fn task_main(config: &MetricCollectionConfig) -> anyhow::Result<()> {
|
|||||||
info!("metrics collector has shut down");
|
info!("metrics collector has shut down");
|
||||||
}
|
}
|
||||||
|
|
||||||
let http_client = http::new_client_with_timeout(DEFAULT_HTTP_REPORTING_TIMEOUT);
|
let http_client = http::new_client();
|
||||||
let mut cached_metrics: HashMap<Ids, (u64, DateTime<Utc>)> = HashMap::new();
|
let mut cached_metrics: HashMap<Ids, (u64, DateTime<Utc>)> = HashMap::new();
|
||||||
let hostname = hostname::get()?.as_os_str().to_string_lossy().into_owned();
|
let hostname = hostname::get()?.as_os_str().to_string_lossy().into_owned();
|
||||||
|
|
||||||
@@ -184,35 +182,35 @@ async fn collect_metrics_iteration(
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
if !res.status().is_success() {
|
if res.status().is_success() {
|
||||||
error!("metrics endpoint refused the sent metrics: {:?}", res);
|
// update cached metrics after they were sent successfully
|
||||||
for metric in chunk.iter().filter(|metric| metric.value > (1u64 << 40)) {
|
for send_metric in chunk {
|
||||||
// Report if the metric value is suspiciously large
|
let stop_time = match send_metric.kind {
|
||||||
error!("potentially abnormal metric value: {:?}", metric);
|
EventType::Incremental { stop_time, .. } => stop_time,
|
||||||
}
|
_ => unreachable!(),
|
||||||
}
|
};
|
||||||
// update cached metrics after they were sent
|
|
||||||
// (to avoid sending the same metrics twice)
|
|
||||||
// see the relevant discussion on why to do so even if the status is not success:
|
|
||||||
// https://github.com/neondatabase/neon/pull/4563#discussion_r1246710956
|
|
||||||
for send_metric in chunk {
|
|
||||||
let stop_time = match send_metric.kind {
|
|
||||||
EventType::Incremental { stop_time, .. } => stop_time,
|
|
||||||
_ => unreachable!(),
|
|
||||||
};
|
|
||||||
|
|
||||||
cached_metrics
|
cached_metrics
|
||||||
.entry(Ids {
|
.entry(Ids {
|
||||||
endpoint_id: send_metric.extra.endpoint_id.clone(),
|
endpoint_id: send_metric.extra.endpoint_id.clone(),
|
||||||
branch_id: send_metric.extra.branch_id.clone(),
|
branch_id: send_metric.extra.branch_id.clone(),
|
||||||
})
|
})
|
||||||
// update cached value (add delta) and time
|
// update cached value (add delta) and time
|
||||||
.and_modify(|e| {
|
.and_modify(|e| {
|
||||||
e.0 = e.0.saturating_add(send_metric.value);
|
e.0 = e.0.saturating_add(send_metric.value);
|
||||||
e.1 = stop_time
|
e.1 = stop_time
|
||||||
})
|
})
|
||||||
// cache new metric
|
// cache new metric
|
||||||
.or_insert((send_metric.value, stop_time));
|
.or_insert((send_metric.value, stop_time));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
error!("metrics endpoint refused the sent metrics: {:?}", res);
|
||||||
|
for metric in chunk.iter() {
|
||||||
|
// Report if the metric value is suspiciously large
|
||||||
|
if metric.value > (1u64 << 40) {
|
||||||
|
error!("potentially abnormal metric value: {:?}", metric);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
|
|||||||
@@ -16,10 +16,7 @@ use metrics::{register_int_counter, register_int_counter_vec, IntCounter, IntCou
|
|||||||
use once_cell::sync::Lazy;
|
use once_cell::sync::Lazy;
|
||||||
use pq_proto::{BeMessage as Be, FeStartupPacket, StartupMessageParams};
|
use pq_proto::{BeMessage as Be, FeStartupPacket, StartupMessageParams};
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use tokio::{
|
use tokio::io::{AsyncRead, AsyncWrite, AsyncWriteExt};
|
||||||
io::{AsyncRead, AsyncWrite, AsyncWriteExt},
|
|
||||||
time,
|
|
||||||
};
|
|
||||||
use tokio_util::sync::CancellationToken;
|
use tokio_util::sync::CancellationToken;
|
||||||
use tracing::{error, info, warn};
|
use tracing::{error, info, warn};
|
||||||
use utils::measured_stream::MeasuredStream;
|
use utils::measured_stream::MeasuredStream;
|
||||||
@@ -308,13 +305,12 @@ pub fn invalidate_cache(node_info: &console::CachedNodeInfo) {
|
|||||||
#[tracing::instrument(name = "connect_once", skip_all)]
|
#[tracing::instrument(name = "connect_once", skip_all)]
|
||||||
async fn connect_to_compute_once(
|
async fn connect_to_compute_once(
|
||||||
node_info: &console::CachedNodeInfo,
|
node_info: &console::CachedNodeInfo,
|
||||||
timeout: time::Duration,
|
|
||||||
) -> Result<PostgresConnection, compute::ConnectionError> {
|
) -> Result<PostgresConnection, compute::ConnectionError> {
|
||||||
let allow_self_signed_compute = node_info.allow_self_signed_compute;
|
let allow_self_signed_compute = node_info.allow_self_signed_compute;
|
||||||
|
|
||||||
node_info
|
node_info
|
||||||
.config
|
.config
|
||||||
.connect(allow_self_signed_compute, timeout)
|
.connect(allow_self_signed_compute)
|
||||||
.inspect_err(|_: &compute::ConnectionError| invalidate_cache(node_info))
|
.inspect_err(|_: &compute::ConnectionError| invalidate_cache(node_info))
|
||||||
.await
|
.await
|
||||||
}
|
}
|
||||||
@@ -332,27 +328,7 @@ async fn connect_to_compute(
|
|||||||
loop {
|
loop {
|
||||||
// Apply startup params to the (possibly, cached) compute node info.
|
// Apply startup params to the (possibly, cached) compute node info.
|
||||||
node_info.config.set_startup_params(params);
|
node_info.config.set_startup_params(params);
|
||||||
|
match connect_to_compute_once(node_info).await {
|
||||||
// Set a shorter timeout for the initial connection attempt.
|
|
||||||
//
|
|
||||||
// In case we try to connect to an outdated address that is no longer valid, the
|
|
||||||
// default behavior of Kubernetes is to drop the packets, causing us to wait for
|
|
||||||
// the entire timeout period. We want to fail fast in such cases.
|
|
||||||
//
|
|
||||||
// A specific case to consider is when we have cached compute node information
|
|
||||||
// with a 4-minute TTL (Time To Live), but the user has executed a `/suspend` API
|
|
||||||
// call, resulting in the nonexistence of the compute node.
|
|
||||||
//
|
|
||||||
// We only use caching in case of scram proxy backed by the console, so reduce
|
|
||||||
// the timeout only in that case.
|
|
||||||
let is_scram_proxy = matches!(creds, auth::BackendType::Console(_, _));
|
|
||||||
let timeout = if is_scram_proxy && num_retries == NUM_RETRIES_WAKE_COMPUTE {
|
|
||||||
time::Duration::from_secs(2)
|
|
||||||
} else {
|
|
||||||
time::Duration::from_secs(10)
|
|
||||||
};
|
|
||||||
|
|
||||||
match connect_to_compute_once(node_info, timeout).await {
|
|
||||||
Err(e) if num_retries > 0 => {
|
Err(e) if num_retries > 0 => {
|
||||||
info!("compute node's state has changed; requesting a wake-up");
|
info!("compute node's state has changed; requesting a wake-up");
|
||||||
match creds.wake_compute(extra).map_err(io_error).await? {
|
match creds.wake_compute(extra).map_err(io_error).await? {
|
||||||
|
|||||||
@@ -45,74 +45,17 @@ fn hmac_sha256<'a>(key: &[u8], parts: impl IntoIterator<Item = &'a [u8]>) -> [u8
|
|||||||
let mut mac = Hmac::<Sha256>::new_from_slice(key).expect("bad key size");
|
let mut mac = Hmac::<Sha256>::new_from_slice(key).expect("bad key size");
|
||||||
parts.into_iter().for_each(|s| mac.update(s));
|
parts.into_iter().for_each(|s| mac.update(s));
|
||||||
|
|
||||||
mac.finalize().into_bytes().into()
|
// TODO: maybe newer `hmac` et al already migrated to regular arrays?
|
||||||
|
let mut result = [0u8; 32];
|
||||||
|
result.copy_from_slice(mac.finalize().into_bytes().as_slice());
|
||||||
|
result
|
||||||
}
|
}
|
||||||
|
|
||||||
fn sha256<'a>(parts: impl IntoIterator<Item = &'a [u8]>) -> [u8; 32] {
|
fn sha256<'a>(parts: impl IntoIterator<Item = &'a [u8]>) -> [u8; 32] {
|
||||||
let mut hasher = Sha256::new();
|
let mut hasher = Sha256::new();
|
||||||
parts.into_iter().for_each(|s| hasher.update(s));
|
parts.into_iter().for_each(|s| hasher.update(s));
|
||||||
|
|
||||||
hasher.finalize().into()
|
let mut result = [0u8; 32];
|
||||||
}
|
result.copy_from_slice(hasher.finalize().as_slice());
|
||||||
|
result
|
||||||
#[cfg(test)]
|
|
||||||
mod tests {
|
|
||||||
use crate::sasl::{Mechanism, Step};
|
|
||||||
|
|
||||||
use super::{password::SaltedPassword, Exchange, ServerSecret};
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn happy_path() {
|
|
||||||
let iterations = 4096;
|
|
||||||
let salt_base64 = "QSXCR+Q6sek8bf92";
|
|
||||||
let pw = SaltedPassword::new(
|
|
||||||
b"pencil",
|
|
||||||
base64::decode(salt_base64).unwrap().as_slice(),
|
|
||||||
iterations,
|
|
||||||
);
|
|
||||||
|
|
||||||
let secret = ServerSecret {
|
|
||||||
iterations,
|
|
||||||
salt_base64: salt_base64.to_owned(),
|
|
||||||
stored_key: pw.client_key().sha256(),
|
|
||||||
server_key: pw.server_key(),
|
|
||||||
doomed: false,
|
|
||||||
};
|
|
||||||
const NONCE: [u8; 18] = [
|
|
||||||
1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18,
|
|
||||||
];
|
|
||||||
let mut exchange = Exchange::new(&secret, || NONCE, None);
|
|
||||||
|
|
||||||
let client_first = "n,,n=user,r=rOprNGfwEbeRWgbNEkqO";
|
|
||||||
let client_final = "c=biws,r=rOprNGfwEbeRWgbNEkqOAQIDBAUGBwgJCgsMDQ4PEBES,p=rw1r5Kph5ThxmaUBC2GAQ6MfXbPnNkFiTIvdb/Rear0=";
|
|
||||||
let server_first =
|
|
||||||
"r=rOprNGfwEbeRWgbNEkqOAQIDBAUGBwgJCgsMDQ4PEBES,s=QSXCR+Q6sek8bf92,i=4096";
|
|
||||||
let server_final = "v=qtUDIofVnIhM7tKn93EQUUt5vgMOldcDVu1HC+OH0o0=";
|
|
||||||
|
|
||||||
exchange = match exchange.exchange(client_first).unwrap() {
|
|
||||||
Step::Continue(exchange, message) => {
|
|
||||||
assert_eq!(message, server_first);
|
|
||||||
exchange
|
|
||||||
}
|
|
||||||
Step::Success(_, _) => panic!("expected continue, got success"),
|
|
||||||
Step::Failure(f) => panic!("{f}"),
|
|
||||||
};
|
|
||||||
|
|
||||||
let key = match exchange.exchange(client_final).unwrap() {
|
|
||||||
Step::Success(key, message) => {
|
|
||||||
assert_eq!(message, server_final);
|
|
||||||
key
|
|
||||||
}
|
|
||||||
Step::Continue(_, _) => panic!("expected success, got continue"),
|
|
||||||
Step::Failure(f) => panic!("{f}"),
|
|
||||||
};
|
|
||||||
|
|
||||||
assert_eq!(
|
|
||||||
key.as_bytes(),
|
|
||||||
[
|
|
||||||
74, 103, 1, 132, 12, 31, 200, 48, 28, 54, 82, 232, 207, 12, 138, 189, 40, 32, 134,
|
|
||||||
27, 125, 170, 232, 35, 171, 167, 166, 41, 70, 228, 182, 112,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,7 +14,19 @@ impl SaltedPassword {
|
|||||||
/// See `scram-common.c : scram_SaltedPassword` for details.
|
/// See `scram-common.c : scram_SaltedPassword` for details.
|
||||||
/// Further reading: <https://datatracker.ietf.org/doc/html/rfc2898> (see `PBKDF2`).
|
/// Further reading: <https://datatracker.ietf.org/doc/html/rfc2898> (see `PBKDF2`).
|
||||||
pub fn new(password: &[u8], salt: &[u8], iterations: u32) -> SaltedPassword {
|
pub fn new(password: &[u8], salt: &[u8], iterations: u32) -> SaltedPassword {
|
||||||
pbkdf2::pbkdf2_hmac_array::<sha2::Sha256, 32>(password, salt, iterations).into()
|
let one = 1_u32.to_be_bytes(); // magic
|
||||||
|
|
||||||
|
let mut current = super::hmac_sha256(password, [salt, &one]);
|
||||||
|
let mut result = current;
|
||||||
|
for _ in 1..iterations {
|
||||||
|
current = super::hmac_sha256(password, [current.as_ref()]);
|
||||||
|
// TODO: result = current.zip(result).map(|(x, y)| x ^ y), issue #80094
|
||||||
|
for (i, x) in current.iter().enumerate() {
|
||||||
|
result[i] ^= x;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
result.into()
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Derive `ClientKey` from a salted hashed password.
|
/// Derive `ClientKey` from a salted hashed password.
|
||||||
@@ -34,41 +46,3 @@ impl From<[u8; SALTED_PASSWORD_LEN]> for SaltedPassword {
|
|||||||
Self { bytes }
|
Self { bytes }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
|
||||||
mod tests {
|
|
||||||
use super::SaltedPassword;
|
|
||||||
|
|
||||||
fn legacy_pbkdf2_impl(password: &[u8], salt: &[u8], iterations: u32) -> SaltedPassword {
|
|
||||||
let one = 1_u32.to_be_bytes(); // magic
|
|
||||||
|
|
||||||
let mut current = super::super::hmac_sha256(password, [salt, &one]);
|
|
||||||
let mut result = current;
|
|
||||||
for _ in 1..iterations {
|
|
||||||
current = super::super::hmac_sha256(password, [current.as_ref()]);
|
|
||||||
// TODO: result = current.zip(result).map(|(x, y)| x ^ y), issue #80094
|
|
||||||
for (i, x) in current.iter().enumerate() {
|
|
||||||
result[i] ^= x;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
result.into()
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn pbkdf2() {
|
|
||||||
let password = "a-very-secure-password";
|
|
||||||
let salt = "such-a-random-salt";
|
|
||||||
let iterations = 4096;
|
|
||||||
let output = [
|
|
||||||
203, 18, 206, 81, 4, 154, 193, 100, 147, 41, 211, 217, 177, 203, 69, 210, 194, 211,
|
|
||||||
101, 1, 248, 156, 96, 0, 8, 223, 30, 87, 158, 41, 20, 42,
|
|
||||||
];
|
|
||||||
|
|
||||||
let actual = SaltedPassword::new(password.as_bytes(), salt.as_bytes(), iterations);
|
|
||||||
let expected = legacy_pbkdf2_impl(password.as_bytes(), salt.as_bytes(), iterations);
|
|
||||||
|
|
||||||
assert_eq!(actual.bytes, output);
|
|
||||||
assert_eq!(actual.bytes, expected.bytes);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ authors = []
|
|||||||
[tool.poetry.dependencies]
|
[tool.poetry.dependencies]
|
||||||
python = "^3.9"
|
python = "^3.9"
|
||||||
pytest = "^7.3.1"
|
pytest = "^7.3.1"
|
||||||
psycopg2-binary = "^2.9.6"
|
psycopg2-binary = "^2.9.1"
|
||||||
typing-extensions = "^4.6.1"
|
typing-extensions = "^4.6.1"
|
||||||
PyJWT = {version = "^2.1.0", extras = ["crypto"]}
|
PyJWT = {version = "^2.1.0", extras = ["crypto"]}
|
||||||
requests = "^2.31.0"
|
requests = "^2.31.0"
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
[toolchain]
|
[toolchain]
|
||||||
channel = "1.70.0"
|
channel = "1.68.2"
|
||||||
profile = "default"
|
profile = "default"
|
||||||
# The default profile includes rustc, rust-std, cargo, rust-docs, rustfmt and clippy.
|
# The default profile includes rustc, rust-std, cargo, rust-docs, rustfmt and clippy.
|
||||||
# https://rust-lang.github.io/rustup/concepts/profiles.html
|
# https://rust-lang.github.io/rustup/concepts/profiles.html
|
||||||
|
|||||||
@@ -144,22 +144,15 @@ const reportSummary = async (params) => {
|
|||||||
}
|
}
|
||||||
summary += `- \`${testName}\`: ${links.join(", ")}\n`
|
summary += `- \`${testName}\`: ${links.join(", ")}\n`
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (failedTestsCount > 0) {
|
const testsToRerun = Object.values(failedTests[pgVersion]).map(x => x[0].name)
|
||||||
const testsToRerun = []
|
const command = `DEFAULT_PG_VERSION=${pgVersion} scripts/pytest -k "${testsToRerun.join(" or ")}"`
|
||||||
for (const pgVersion of Object.keys(failedTests)) {
|
|
||||||
for (const testName of Object.keys(failedTests[pgVersion])) {
|
|
||||||
testsToRerun.push(...failedTests[pgVersion][testName].map(test => test.name))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
const command = `scripts/pytest -vv -n $(nproc) -k "${testsToRerun.join(" or ")}"`
|
|
||||||
|
|
||||||
summary += "```\n"
|
summary += "```\n"
|
||||||
summary += `# Run all failed tests locally:\n`
|
summary += `# Run failed on Postgres ${pgVersion} tests locally:\n`
|
||||||
summary += `${command}\n`
|
summary += `${command}\n`
|
||||||
summary += "```\n"
|
summary += "```\n"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (flakyTestsCount > 0) {
|
if (flakyTestsCount > 0) {
|
||||||
@@ -171,7 +164,8 @@ const reportSummary = async (params) => {
|
|||||||
const links = []
|
const links = []
|
||||||
for (const test of tests) {
|
for (const test of tests) {
|
||||||
const allureLink = `${reportUrl}#suites/${test.parentUid}/${test.uid}/retries`
|
const allureLink = `${reportUrl}#suites/${test.parentUid}/${test.uid}/retries`
|
||||||
links.push(`[${test.buildType}](${allureLink})`)
|
const status = test.status === "passed" ? ":white_check_mark:" : ":x:"
|
||||||
|
links.push(`[${status} ${test.buildType}](${allureLink})`)
|
||||||
}
|
}
|
||||||
summary += `- \`${testName}\`: ${links.join(", ")}\n`
|
summary += `- \`${testName}\`: ${links.join(", ")}\n`
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
pytest_plugins = (
|
pytest_plugins = (
|
||||||
"fixtures.pg_version",
|
"fixtures.pg_version",
|
||||||
"fixtures.parametrize",
|
"fixtures.allure",
|
||||||
"fixtures.neon_fixtures",
|
"fixtures.neon_fixtures",
|
||||||
"fixtures.benchmark_fixture",
|
"fixtures.benchmark_fixture",
|
||||||
"fixtures.pg_stats",
|
"fixtures.pg_stats",
|
||||||
|
|||||||
25
test_runner/fixtures/allure.py
Normal file
25
test_runner/fixtures/allure.py
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
import os
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
from fixtures.pg_version import DEFAULT_VERSION, PgVersion
|
||||||
|
|
||||||
|
"""
|
||||||
|
Set of utilities to make Allure report more informative.
|
||||||
|
|
||||||
|
- It adds BUILD_TYPE and DEFAULT_PG_VERSION to the test names (only in test_runner/regress)
|
||||||
|
to make tests distinguishable in Allure report.
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture(scope="function", autouse=True)
|
||||||
|
def allure_noop():
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
def pytest_generate_tests(metafunc):
|
||||||
|
if "test_runner/regress" in metafunc.definition._nodeid:
|
||||||
|
build_type = os.environ.get("BUILD_TYPE", "DEBUG").lower()
|
||||||
|
pg_version = PgVersion(os.environ.get("DEFAULT_PG_VERSION", DEFAULT_VERSION))
|
||||||
|
|
||||||
|
metafunc.parametrize("allure_noop", [f"{build_type}-pg{pg_version}"])
|
||||||
@@ -59,14 +59,9 @@ PAGESERVER_GLOBAL_METRICS: Tuple[str, ...] = (
|
|||||||
"libmetrics_tracing_event_count_total",
|
"libmetrics_tracing_event_count_total",
|
||||||
"pageserver_materialized_cache_hits_total",
|
"pageserver_materialized_cache_hits_total",
|
||||||
"pageserver_materialized_cache_hits_direct_total",
|
"pageserver_materialized_cache_hits_direct_total",
|
||||||
"pageserver_page_cache_read_hits_total",
|
|
||||||
"pageserver_page_cache_read_accesses_total",
|
|
||||||
"pageserver_page_cache_size_current_bytes",
|
|
||||||
"pageserver_page_cache_size_max_bytes",
|
|
||||||
"pageserver_getpage_reconstruct_seconds_bucket",
|
"pageserver_getpage_reconstruct_seconds_bucket",
|
||||||
"pageserver_getpage_reconstruct_seconds_count",
|
"pageserver_getpage_reconstruct_seconds_count",
|
||||||
"pageserver_getpage_reconstruct_seconds_sum",
|
"pageserver_getpage_reconstruct_seconds_sum",
|
||||||
*[f"pageserver_basebackup_query_seconds_{x}" for x in ["bucket", "count", "sum"]],
|
|
||||||
)
|
)
|
||||||
|
|
||||||
PAGESERVER_PER_TENANT_METRICS: Tuple[str, ...] = (
|
PAGESERVER_PER_TENANT_METRICS: Tuple[str, ...] = (
|
||||||
|
|||||||
@@ -102,8 +102,8 @@ def base_dir() -> Iterator[Path]:
|
|||||||
yield base_dir
|
yield base_dir
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(scope="function")
|
@pytest.fixture(scope="session")
|
||||||
def neon_binpath(base_dir: Path, build_type: str) -> Iterator[Path]:
|
def neon_binpath(base_dir: Path) -> Iterator[Path]:
|
||||||
if os.getenv("REMOTE_ENV"):
|
if os.getenv("REMOTE_ENV"):
|
||||||
# we are in remote env and do not have neon binaries locally
|
# we are in remote env and do not have neon binaries locally
|
||||||
# this is the case for benchmarks run on self-hosted runner
|
# this is the case for benchmarks run on self-hosted runner
|
||||||
@@ -113,6 +113,7 @@ def neon_binpath(base_dir: Path, build_type: str) -> Iterator[Path]:
|
|||||||
if env_neon_bin := os.environ.get("NEON_BIN"):
|
if env_neon_bin := os.environ.get("NEON_BIN"):
|
||||||
binpath = Path(env_neon_bin)
|
binpath = Path(env_neon_bin)
|
||||||
else:
|
else:
|
||||||
|
build_type = os.environ.get("BUILD_TYPE", "debug")
|
||||||
binpath = base_dir / "target" / build_type
|
binpath = base_dir / "target" / build_type
|
||||||
log.info(f"neon_binpath is {binpath}")
|
log.info(f"neon_binpath is {binpath}")
|
||||||
|
|
||||||
@@ -122,7 +123,7 @@ def neon_binpath(base_dir: Path, build_type: str) -> Iterator[Path]:
|
|||||||
yield binpath
|
yield binpath
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(scope="function")
|
@pytest.fixture(scope="session")
|
||||||
def pg_distrib_dir(base_dir: Path) -> Iterator[Path]:
|
def pg_distrib_dir(base_dir: Path) -> Iterator[Path]:
|
||||||
if env_postgres_bin := os.environ.get("POSTGRES_DISTRIB_DIR"):
|
if env_postgres_bin := os.environ.get("POSTGRES_DISTRIB_DIR"):
|
||||||
distrib_dir = Path(env_postgres_bin).resolve()
|
distrib_dir = Path(env_postgres_bin).resolve()
|
||||||
@@ -146,7 +147,7 @@ def top_output_dir(base_dir: Path) -> Iterator[Path]:
|
|||||||
yield output_dir
|
yield output_dir
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(scope="function")
|
@pytest.fixture(scope="session")
|
||||||
def versioned_pg_distrib_dir(pg_distrib_dir: Path, pg_version: PgVersion) -> Iterator[Path]:
|
def versioned_pg_distrib_dir(pg_distrib_dir: Path, pg_version: PgVersion) -> Iterator[Path]:
|
||||||
versioned_dir = pg_distrib_dir / pg_version.v_prefixed
|
versioned_dir = pg_distrib_dir / pg_version.v_prefixed
|
||||||
|
|
||||||
@@ -173,23 +174,7 @@ def shareable_scope(fixture_name: str, config: Config) -> Literal["session", "fu
|
|||||||
def myfixture(...)
|
def myfixture(...)
|
||||||
...
|
...
|
||||||
"""
|
"""
|
||||||
scope: Literal["session", "function"]
|
return "function" if os.environ.get("TEST_SHARED_FIXTURES") is None else "session"
|
||||||
|
|
||||||
if os.environ.get("TEST_SHARED_FIXTURES") is None:
|
|
||||||
# Create the environment in the per-test output directory
|
|
||||||
scope = "function"
|
|
||||||
elif (
|
|
||||||
os.environ.get("BUILD_TYPE") is not None
|
|
||||||
and os.environ.get("DEFAULT_PG_VERSION") is not None
|
|
||||||
):
|
|
||||||
scope = "session"
|
|
||||||
else:
|
|
||||||
pytest.fail(
|
|
||||||
"Shared environment(TEST_SHARED_FIXTURES) requires BUILD_TYPE and DEFAULT_PG_VERSION to be set",
|
|
||||||
pytrace=False,
|
|
||||||
)
|
|
||||||
|
|
||||||
return scope
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(scope="session")
|
@pytest.fixture(scope="session")
|
||||||
@@ -615,6 +600,8 @@ class NeonEnvBuilder:
|
|||||||
self.rust_log_override = rust_log_override
|
self.rust_log_override = rust_log_override
|
||||||
self.port_distributor = port_distributor
|
self.port_distributor = port_distributor
|
||||||
self.remote_storage = remote_storage
|
self.remote_storage = remote_storage
|
||||||
|
self.ext_remote_storage: Optional[Any] = None
|
||||||
|
self.remote_storage_client: Optional[Any] = None
|
||||||
self.remote_storage_users = remote_storage_users
|
self.remote_storage_users = remote_storage_users
|
||||||
self.broker = broker
|
self.broker = broker
|
||||||
self.run_id = run_id
|
self.run_id = run_id
|
||||||
@@ -666,13 +653,18 @@ class NeonEnvBuilder:
|
|||||||
remote_storage_kind: RemoteStorageKind,
|
remote_storage_kind: RemoteStorageKind,
|
||||||
test_name: str,
|
test_name: str,
|
||||||
force_enable: bool = True,
|
force_enable: bool = True,
|
||||||
|
enable_remote_extensions: bool = False,
|
||||||
):
|
):
|
||||||
if remote_storage_kind == RemoteStorageKind.NOOP:
|
if remote_storage_kind == RemoteStorageKind.NOOP:
|
||||||
return
|
return
|
||||||
elif remote_storage_kind == RemoteStorageKind.LOCAL_FS:
|
elif remote_storage_kind == RemoteStorageKind.LOCAL_FS:
|
||||||
self.enable_local_fs_remote_storage(force_enable=force_enable)
|
self.enable_local_fs_remote_storage(force_enable=force_enable)
|
||||||
elif remote_storage_kind == RemoteStorageKind.MOCK_S3:
|
elif remote_storage_kind == RemoteStorageKind.MOCK_S3:
|
||||||
self.enable_mock_s3_remote_storage(bucket_name=test_name, force_enable=force_enable)
|
self.enable_mock_s3_remote_storage(
|
||||||
|
bucket_name=test_name,
|
||||||
|
force_enable=force_enable,
|
||||||
|
enable_remote_extensions=enable_remote_extensions,
|
||||||
|
)
|
||||||
elif remote_storage_kind == RemoteStorageKind.REAL_S3:
|
elif remote_storage_kind == RemoteStorageKind.REAL_S3:
|
||||||
self.enable_real_s3_remote_storage(test_name=test_name, force_enable=force_enable)
|
self.enable_real_s3_remote_storage(test_name=test_name, force_enable=force_enable)
|
||||||
else:
|
else:
|
||||||
@@ -688,11 +680,15 @@ class NeonEnvBuilder:
|
|||||||
assert force_enable or self.remote_storage is None, "remote storage is enabled already"
|
assert force_enable or self.remote_storage is None, "remote storage is enabled already"
|
||||||
self.remote_storage = LocalFsStorage(Path(self.repo_dir / "local_fs_remote_storage"))
|
self.remote_storage = LocalFsStorage(Path(self.repo_dir / "local_fs_remote_storage"))
|
||||||
|
|
||||||
def enable_mock_s3_remote_storage(self, bucket_name: str, force_enable: bool = True):
|
def enable_mock_s3_remote_storage(
|
||||||
|
self, bucket_name: str, force_enable: bool = True, enable_remote_extensions: bool = False
|
||||||
|
):
|
||||||
"""
|
"""
|
||||||
Sets up the pageserver to use the S3 mock server, creates the bucket, if it's not present already.
|
Sets up the pageserver to use the S3 mock server, creates the bucket, if it's not present already.
|
||||||
Starts up the mock server, if that does not run yet.
|
Starts up the mock server, if that does not run yet.
|
||||||
Errors, if the pageserver has some remote storage configuration already, unless `force_enable` is not set to `True`.
|
Errors, if the pageserver has some remote storage configuration already, unless `force_enable` is not set to `True`.
|
||||||
|
|
||||||
|
Also creates the bucket for extensions, self.ext_remote_storage bucket
|
||||||
"""
|
"""
|
||||||
assert force_enable or self.remote_storage is None, "remote storage is enabled already"
|
assert force_enable or self.remote_storage is None, "remote storage is enabled already"
|
||||||
mock_endpoint = self.mock_s3_server.endpoint()
|
mock_endpoint = self.mock_s3_server.endpoint()
|
||||||
@@ -715,6 +711,17 @@ class NeonEnvBuilder:
|
|||||||
secret_key=self.mock_s3_server.secret_key(),
|
secret_key=self.mock_s3_server.secret_key(),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
if enable_remote_extensions:
|
||||||
|
ext_bucket_name = f"ext_{bucket_name}"
|
||||||
|
self.remote_storage_client.create_bucket(Bucket=ext_bucket_name)
|
||||||
|
self.ext_remote_storage = S3Storage(
|
||||||
|
bucket_name=ext_bucket_name,
|
||||||
|
endpoint=mock_endpoint,
|
||||||
|
bucket_region=mock_region,
|
||||||
|
access_key=self.mock_s3_server.access_key(),
|
||||||
|
secret_key=self.mock_s3_server.secret_key(),
|
||||||
|
)
|
||||||
|
|
||||||
def enable_real_s3_remote_storage(self, test_name: str, force_enable: bool = True):
|
def enable_real_s3_remote_storage(self, test_name: str, force_enable: bool = True):
|
||||||
"""
|
"""
|
||||||
Sets up configuration to use real s3 endpoint without mock server
|
Sets up configuration to use real s3 endpoint without mock server
|
||||||
@@ -755,6 +762,17 @@ class NeonEnvBuilder:
|
|||||||
prefix_in_bucket=self.remote_storage_prefix,
|
prefix_in_bucket=self.remote_storage_prefix,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
ext_bucket_name = os.getenv("EXT_REMOTE_STORAGE_S3_BUCKET")
|
||||||
|
if ext_bucket_name is not None:
|
||||||
|
ext_bucket_name = f"ext_{ext_bucket_name}"
|
||||||
|
self.ext_remote_storage = S3Storage(
|
||||||
|
bucket_name=ext_bucket_name,
|
||||||
|
bucket_region=region,
|
||||||
|
access_key=access_key,
|
||||||
|
secret_key=secret_key,
|
||||||
|
prefix_in_bucket=self.remote_storage_prefix,
|
||||||
|
)
|
||||||
|
|
||||||
def cleanup_local_storage(self):
|
def cleanup_local_storage(self):
|
||||||
if self.preserve_database_files:
|
if self.preserve_database_files:
|
||||||
return
|
return
|
||||||
@@ -788,6 +806,7 @@ class NeonEnvBuilder:
|
|||||||
# `self.remote_storage_prefix` is coupled with `S3Storage` storage type,
|
# `self.remote_storage_prefix` is coupled with `S3Storage` storage type,
|
||||||
# so this line effectively a no-op
|
# so this line effectively a no-op
|
||||||
assert isinstance(self.remote_storage, S3Storage)
|
assert isinstance(self.remote_storage, S3Storage)
|
||||||
|
assert self.remote_storage_client is not None
|
||||||
|
|
||||||
if self.keep_remote_storage_contents:
|
if self.keep_remote_storage_contents:
|
||||||
log.info("keep_remote_storage_contents skipping remote storage cleanup")
|
log.info("keep_remote_storage_contents skipping remote storage cleanup")
|
||||||
@@ -917,6 +936,8 @@ class NeonEnv:
|
|||||||
self.neon_binpath = config.neon_binpath
|
self.neon_binpath = config.neon_binpath
|
||||||
self.pg_distrib_dir = config.pg_distrib_dir
|
self.pg_distrib_dir = config.pg_distrib_dir
|
||||||
self.endpoint_counter = 0
|
self.endpoint_counter = 0
|
||||||
|
self.remote_storage_client = config.remote_storage_client
|
||||||
|
self.ext_remote_storage = config.ext_remote_storage
|
||||||
|
|
||||||
# generate initial tenant ID here instead of letting 'neon init' generate it,
|
# generate initial tenant ID here instead of letting 'neon init' generate it,
|
||||||
# so that we don't need to dig it out of the config file afterwards.
|
# so that we don't need to dig it out of the config file afterwards.
|
||||||
@@ -1503,6 +1524,7 @@ class NeonCli(AbstractNeonCli):
|
|||||||
safekeepers: Optional[List[int]] = None,
|
safekeepers: Optional[List[int]] = None,
|
||||||
tenant_id: Optional[TenantId] = None,
|
tenant_id: Optional[TenantId] = None,
|
||||||
lsn: Optional[Lsn] = None,
|
lsn: Optional[Lsn] = None,
|
||||||
|
remote_ext_config: Optional[str] = None,
|
||||||
) -> "subprocess.CompletedProcess[str]":
|
) -> "subprocess.CompletedProcess[str]":
|
||||||
args = [
|
args = [
|
||||||
"endpoint",
|
"endpoint",
|
||||||
@@ -1512,6 +1534,8 @@ class NeonCli(AbstractNeonCli):
|
|||||||
"--pg-version",
|
"--pg-version",
|
||||||
self.env.pg_version,
|
self.env.pg_version,
|
||||||
]
|
]
|
||||||
|
if remote_ext_config is not None:
|
||||||
|
args.extend(["--remote-ext-config", remote_ext_config])
|
||||||
if lsn is not None:
|
if lsn is not None:
|
||||||
args.append(f"--lsn={lsn}")
|
args.append(f"--lsn={lsn}")
|
||||||
args.extend(["--pg-port", str(pg_port)])
|
args.extend(["--pg-port", str(pg_port)])
|
||||||
@@ -2373,7 +2397,7 @@ class Endpoint(PgProtocol):
|
|||||||
|
|
||||||
return self
|
return self
|
||||||
|
|
||||||
def start(self) -> "Endpoint":
|
def start(self, remote_ext_config: Optional[str] = None) -> "Endpoint":
|
||||||
"""
|
"""
|
||||||
Start the Postgres instance.
|
Start the Postgres instance.
|
||||||
Returns self.
|
Returns self.
|
||||||
@@ -2389,6 +2413,7 @@ class Endpoint(PgProtocol):
|
|||||||
http_port=self.http_port,
|
http_port=self.http_port,
|
||||||
tenant_id=self.tenant_id,
|
tenant_id=self.tenant_id,
|
||||||
safekeepers=self.active_safekeepers,
|
safekeepers=self.active_safekeepers,
|
||||||
|
remote_ext_config=remote_ext_config,
|
||||||
)
|
)
|
||||||
self.running = True
|
self.running = True
|
||||||
|
|
||||||
@@ -2478,6 +2503,7 @@ class Endpoint(PgProtocol):
|
|||||||
hot_standby: bool = False,
|
hot_standby: bool = False,
|
||||||
lsn: Optional[Lsn] = None,
|
lsn: Optional[Lsn] = None,
|
||||||
config_lines: Optional[List[str]] = None,
|
config_lines: Optional[List[str]] = None,
|
||||||
|
remote_ext_config: Optional[str] = None,
|
||||||
) -> "Endpoint":
|
) -> "Endpoint":
|
||||||
"""
|
"""
|
||||||
Create an endpoint, apply config, and start Postgres.
|
Create an endpoint, apply config, and start Postgres.
|
||||||
@@ -2492,7 +2518,7 @@ class Endpoint(PgProtocol):
|
|||||||
config_lines=config_lines,
|
config_lines=config_lines,
|
||||||
hot_standby=hot_standby,
|
hot_standby=hot_standby,
|
||||||
lsn=lsn,
|
lsn=lsn,
|
||||||
).start()
|
).start(remote_ext_config=remote_ext_config)
|
||||||
|
|
||||||
log.info(f"Postgres startup took {time.time() - started_at} seconds")
|
log.info(f"Postgres startup took {time.time() - started_at} seconds")
|
||||||
|
|
||||||
@@ -2526,6 +2552,7 @@ class EndpointFactory:
|
|||||||
lsn: Optional[Lsn] = None,
|
lsn: Optional[Lsn] = None,
|
||||||
hot_standby: bool = False,
|
hot_standby: bool = False,
|
||||||
config_lines: Optional[List[str]] = None,
|
config_lines: Optional[List[str]] = None,
|
||||||
|
remote_ext_config: Optional[str] = None,
|
||||||
) -> Endpoint:
|
) -> Endpoint:
|
||||||
ep = Endpoint(
|
ep = Endpoint(
|
||||||
self.env,
|
self.env,
|
||||||
@@ -2542,6 +2569,7 @@ class EndpointFactory:
|
|||||||
hot_standby=hot_standby,
|
hot_standby=hot_standby,
|
||||||
config_lines=config_lines,
|
config_lines=config_lines,
|
||||||
lsn=lsn,
|
lsn=lsn,
|
||||||
|
remote_ext_config=remote_ext_config,
|
||||||
)
|
)
|
||||||
|
|
||||||
def create(
|
def create(
|
||||||
|
|||||||
@@ -21,18 +21,6 @@ class PageserverApiException(Exception):
|
|||||||
self.status_code = status_code
|
self.status_code = status_code
|
||||||
|
|
||||||
|
|
||||||
class TimelineCreate406(PageserverApiException):
|
|
||||||
def __init__(self, res: requests.Response):
|
|
||||||
assert res.status_code == 406
|
|
||||||
super().__init__(res.json()["msg"], res.status_code)
|
|
||||||
|
|
||||||
|
|
||||||
class TimelineCreate409(PageserverApiException):
|
|
||||||
def __init__(self, res: requests.Response):
|
|
||||||
assert res.status_code == 409
|
|
||||||
super().__init__("", res.status_code)
|
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class InMemoryLayerInfo:
|
class InMemoryLayerInfo:
|
||||||
kind: str
|
kind: str
|
||||||
@@ -321,12 +309,9 @@ class PageserverHttpClient(requests.Session):
|
|||||||
res = self.post(
|
res = self.post(
|
||||||
f"http://localhost:{self.port}/v1/tenant/{tenant_id}/timeline", json=body, **kwargs
|
f"http://localhost:{self.port}/v1/tenant/{tenant_id}/timeline", json=body, **kwargs
|
||||||
)
|
)
|
||||||
if res.status_code == 409:
|
|
||||||
raise TimelineCreate409(res)
|
|
||||||
if res.status_code == 406:
|
|
||||||
raise TimelineCreate406(res)
|
|
||||||
|
|
||||||
self.verbose_error(res)
|
self.verbose_error(res)
|
||||||
|
if res.status_code == 409:
|
||||||
|
raise Exception(f"could not create timeline: already exists for id {new_timeline_id}")
|
||||||
|
|
||||||
res_json = res.json()
|
res_json = res.json()
|
||||||
assert isinstance(res_json, dict)
|
assert isinstance(res_json, dict)
|
||||||
|
|||||||
@@ -1,50 +0,0 @@
|
|||||||
import os
|
|
||||||
from typing import Optional
|
|
||||||
|
|
||||||
import pytest
|
|
||||||
from _pytest.fixtures import FixtureRequest
|
|
||||||
from _pytest.python import Metafunc
|
|
||||||
|
|
||||||
from fixtures.pg_version import PgVersion
|
|
||||||
|
|
||||||
"""
|
|
||||||
Dynamically parametrize tests by Postgres version and build type (debug/release/remote)
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(scope="function", autouse=True)
|
|
||||||
def pg_version(request: FixtureRequest) -> Optional[PgVersion]:
|
|
||||||
# Do not parametrize performance tests yet, we need to prepare grafana charts first
|
|
||||||
if "test_runner/performance" in str(request.node.path):
|
|
||||||
v = os.environ.get("DEFAULT_PG_VERSION")
|
|
||||||
return PgVersion(v)
|
|
||||||
|
|
||||||
return None
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(scope="function", autouse=True)
|
|
||||||
def build_type(request: FixtureRequest) -> Optional[str]:
|
|
||||||
# Do not parametrize performance tests yet, we need to prepare grafana charts first
|
|
||||||
if "test_runner/performance" in str(request.node.path):
|
|
||||||
return os.environ.get("BUILD_TYPE", "").lower()
|
|
||||||
|
|
||||||
return None
|
|
||||||
|
|
||||||
|
|
||||||
def pytest_generate_tests(metafunc: Metafunc):
|
|
||||||
# Do not parametrize performance tests yet, we need to prepare grafana charts first
|
|
||||||
if "test_runner/performance" in metafunc.definition._nodeid:
|
|
||||||
return
|
|
||||||
|
|
||||||
if (v := os.environ.get("DEFAULT_PG_VERSION")) is None:
|
|
||||||
pg_versions = [version for version in PgVersion if version != PgVersion.NOT_SET]
|
|
||||||
else:
|
|
||||||
pg_versions = [PgVersion(v)]
|
|
||||||
|
|
||||||
if (bt := os.environ.get("BUILD_TYPE")) is None:
|
|
||||||
build_types = ["debug", "release"]
|
|
||||||
else:
|
|
||||||
build_types = [bt.lower()]
|
|
||||||
|
|
||||||
metafunc.parametrize("build_type", build_types)
|
|
||||||
metafunc.parametrize("pg_version", pg_versions, ids=map(lambda v: f"pg{v}", pg_versions))
|
|
||||||
@@ -1,10 +1,12 @@
|
|||||||
import enum
|
import enum
|
||||||
import os
|
import os
|
||||||
from typing import Optional
|
from typing import Iterator, Optional
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from _pytest.config import Config
|
|
||||||
from _pytest.config.argparsing import Parser
|
from _pytest.config.argparsing import Parser
|
||||||
|
from pytest import FixtureRequest
|
||||||
|
|
||||||
|
from fixtures.log_helper import log
|
||||||
|
|
||||||
"""
|
"""
|
||||||
This fixture is used to determine which version of Postgres to use for tests.
|
This fixture is used to determine which version of Postgres to use for tests.
|
||||||
@@ -73,10 +75,18 @@ def pytest_addoption(parser: Parser):
|
|||||||
"--pg-version",
|
"--pg-version",
|
||||||
action="store",
|
action="store",
|
||||||
type=PgVersion,
|
type=PgVersion,
|
||||||
help="DEPRECATED: Postgres version to use for tests",
|
help="Postgres version to use for tests",
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
def pytest_configure(config: Config):
|
@pytest.fixture(scope="session")
|
||||||
if config.getoption("--pg-version"):
|
def pg_version(request: FixtureRequest) -> Iterator[PgVersion]:
|
||||||
raise Exception("--pg-version is deprecated, use DEFAULT_PG_VERSION env var instead")
|
if v := request.config.getoption("--pg-version"):
|
||||||
|
version, source = v, "from --pg-version command-line argument"
|
||||||
|
elif v := os.environ.get("DEFAULT_PG_VERSION"):
|
||||||
|
version, source = PgVersion(v), "from DEFAULT_PG_VERSION environment variable"
|
||||||
|
else:
|
||||||
|
version, source = DEFAULT_VERSION, "default version"
|
||||||
|
|
||||||
|
log.info(f"pg_version is {version} ({source})")
|
||||||
|
yield version
|
||||||
|
|||||||
@@ -52,7 +52,6 @@ def test_startup_simple(neon_env_builder: NeonEnvBuilder, zenbenchmark: NeonBenc
|
|||||||
"wait_for_spec_ms": f"{i}_wait_for_spec",
|
"wait_for_spec_ms": f"{i}_wait_for_spec",
|
||||||
"sync_safekeepers_ms": f"{i}_sync_safekeepers",
|
"sync_safekeepers_ms": f"{i}_sync_safekeepers",
|
||||||
"basebackup_ms": f"{i}_basebackup",
|
"basebackup_ms": f"{i}_basebackup",
|
||||||
"start_postgres_ms": f"{i}_start_postgres",
|
|
||||||
"config_ms": f"{i}_config",
|
"config_ms": f"{i}_config",
|
||||||
"total_startup_ms": f"{i}_total_startup",
|
"total_startup_ms": f"{i}_total_startup",
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,8 +4,7 @@ import time
|
|||||||
import pytest
|
import pytest
|
||||||
from fixtures.log_helper import log
|
from fixtures.log_helper import log
|
||||||
from fixtures.neon_fixtures import NeonEnv
|
from fixtures.neon_fixtures import NeonEnv
|
||||||
from fixtures.pageserver.http import TimelineCreate406
|
from fixtures.types import Lsn
|
||||||
from fixtures.types import Lsn, TimelineId
|
|
||||||
from fixtures.utils import query_scalar
|
from fixtures.utils import query_scalar
|
||||||
|
|
||||||
|
|
||||||
@@ -174,12 +173,5 @@ def test_branch_creation_before_gc(neon_simple_env: NeonEnv):
|
|||||||
# The starting LSN is invalid as the corresponding record is scheduled to be removed by in-queue GC.
|
# The starting LSN is invalid as the corresponding record is scheduled to be removed by in-queue GC.
|
||||||
with pytest.raises(Exception, match="invalid branch start lsn: .*"):
|
with pytest.raises(Exception, match="invalid branch start lsn: .*"):
|
||||||
env.neon_cli.create_branch("b1", "b0", tenant_id=tenant, ancestor_start_lsn=lsn)
|
env.neon_cli.create_branch("b1", "b0", tenant_id=tenant, ancestor_start_lsn=lsn)
|
||||||
# retry the same with the HTTP API, so that we can inspect the status code
|
|
||||||
with pytest.raises(TimelineCreate406):
|
|
||||||
new_timeline_id = TimelineId.generate()
|
|
||||||
log.info(
|
|
||||||
f"Expecting failure for branch behind gc'ing LSN, new_timeline_id={new_timeline_id}"
|
|
||||||
)
|
|
||||||
pageserver_http_client.timeline_create(env.pg_version, tenant, new_timeline_id, b0, lsn)
|
|
||||||
|
|
||||||
thread.join()
|
thread.join()
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
import pytest
|
import pytest
|
||||||
from fixtures.log_helper import log
|
from fixtures.log_helper import log
|
||||||
from fixtures.neon_fixtures import NeonEnvBuilder
|
from fixtures.neon_fixtures import NeonEnvBuilder
|
||||||
from fixtures.pageserver.http import TimelineCreate406
|
|
||||||
from fixtures.types import Lsn, TimelineId
|
from fixtures.types import Lsn, TimelineId
|
||||||
from fixtures.utils import print_gc_result, query_scalar
|
from fixtures.utils import print_gc_result, query_scalar
|
||||||
|
|
||||||
@@ -18,7 +17,7 @@ def test_branch_behind(neon_env_builder: NeonEnvBuilder):
|
|||||||
env.pageserver.allowed_errors.append(".*invalid start lsn .* for ancestor timeline.*")
|
env.pageserver.allowed_errors.append(".*invalid start lsn .* for ancestor timeline.*")
|
||||||
|
|
||||||
# Branch at the point where only 100 rows were inserted
|
# Branch at the point where only 100 rows were inserted
|
||||||
branch_behind_timeline_id = env.neon_cli.create_branch("test_branch_behind")
|
env.neon_cli.create_branch("test_branch_behind")
|
||||||
endpoint_main = env.endpoints.create_start("test_branch_behind")
|
endpoint_main = env.endpoints.create_start("test_branch_behind")
|
||||||
log.info("postgres is running on 'test_branch_behind' branch")
|
log.info("postgres is running on 'test_branch_behind' branch")
|
||||||
|
|
||||||
@@ -90,7 +89,6 @@ def test_branch_behind(neon_env_builder: NeonEnvBuilder):
|
|||||||
assert query_scalar(main_cur, "SELECT count(*) FROM foo") == 400100
|
assert query_scalar(main_cur, "SELECT count(*) FROM foo") == 400100
|
||||||
|
|
||||||
# Check bad lsn's for branching
|
# Check bad lsn's for branching
|
||||||
pageserver_http = env.pageserver.http_client()
|
|
||||||
|
|
||||||
# branch at segment boundary
|
# branch at segment boundary
|
||||||
env.neon_cli.create_branch(
|
env.neon_cli.create_branch(
|
||||||
@@ -99,52 +97,27 @@ def test_branch_behind(neon_env_builder: NeonEnvBuilder):
|
|||||||
endpoint = env.endpoints.create_start("test_branch_segment_boundary")
|
endpoint = env.endpoints.create_start("test_branch_segment_boundary")
|
||||||
assert endpoint.safe_psql("SELECT 1")[0][0] == 1
|
assert endpoint.safe_psql("SELECT 1")[0][0] == 1
|
||||||
|
|
||||||
# branch at pre-initdb lsn (from main branch)
|
# branch at pre-initdb lsn
|
||||||
with pytest.raises(Exception, match="invalid branch start lsn: .*"):
|
with pytest.raises(Exception, match="invalid branch start lsn: .*"):
|
||||||
env.neon_cli.create_branch("test_branch_preinitdb", ancestor_start_lsn=Lsn("0/42"))
|
env.neon_cli.create_branch("test_branch_preinitdb", ancestor_start_lsn=Lsn("0/42"))
|
||||||
# retry the same with the HTTP API, so that we can inspect the status code
|
|
||||||
with pytest.raises(TimelineCreate406):
|
|
||||||
new_timeline_id = TimelineId.generate()
|
|
||||||
log.info(f"Expecting failure for branch pre-initdb LSN, new_timeline_id={new_timeline_id}")
|
|
||||||
pageserver_http.timeline_create(
|
|
||||||
env.pg_version, env.initial_tenant, new_timeline_id, env.initial_timeline, Lsn("0/42")
|
|
||||||
)
|
|
||||||
|
|
||||||
# branch at pre-ancestor lsn
|
# branch at pre-ancestor lsn
|
||||||
with pytest.raises(Exception, match="less than timeline ancestor lsn"):
|
with pytest.raises(Exception, match="less than timeline ancestor lsn"):
|
||||||
env.neon_cli.create_branch(
|
env.neon_cli.create_branch(
|
||||||
"test_branch_preinitdb", "test_branch_behind", ancestor_start_lsn=Lsn("0/42")
|
"test_branch_preinitdb", "test_branch_behind", ancestor_start_lsn=Lsn("0/42")
|
||||||
)
|
)
|
||||||
# retry the same with the HTTP API, so that we can inspect the status code
|
|
||||||
with pytest.raises(TimelineCreate406):
|
|
||||||
new_timeline_id = TimelineId.generate()
|
|
||||||
log.info(
|
|
||||||
f"Expecting failure for branch pre-ancestor LSN, new_timeline_id={new_timeline_id}"
|
|
||||||
)
|
|
||||||
pageserver_http.timeline_create(
|
|
||||||
env.pg_version,
|
|
||||||
env.initial_tenant,
|
|
||||||
new_timeline_id,
|
|
||||||
branch_behind_timeline_id,
|
|
||||||
Lsn("0/42"),
|
|
||||||
)
|
|
||||||
|
|
||||||
# check that we cannot create branch based on garbage collected data
|
# check that we cannot create branch based on garbage collected data
|
||||||
pageserver_http.timeline_checkpoint(env.initial_tenant, timeline)
|
with env.pageserver.http_client() as pageserver_http:
|
||||||
gc_result = pageserver_http.timeline_gc(env.initial_tenant, timeline, 0)
|
pageserver_http.timeline_checkpoint(env.initial_tenant, timeline)
|
||||||
print_gc_result(gc_result)
|
gc_result = pageserver_http.timeline_gc(env.initial_tenant, timeline, 0)
|
||||||
|
print_gc_result(gc_result)
|
||||||
|
|
||||||
with pytest.raises(Exception, match="invalid branch start lsn: .*"):
|
with pytest.raises(Exception, match="invalid branch start lsn: .*"):
|
||||||
# this gced_lsn is pretty random, so if gc is disabled this woudln't fail
|
# this gced_lsn is pretty random, so if gc is disabled this woudln't fail
|
||||||
env.neon_cli.create_branch(
|
env.neon_cli.create_branch(
|
||||||
"test_branch_create_fail", "test_branch_behind", ancestor_start_lsn=gced_lsn
|
"test_branch_create_fail", "test_branch_behind", ancestor_start_lsn=gced_lsn
|
||||||
)
|
)
|
||||||
# retry the same with the HTTP API, so that we can inspect the status code
|
|
||||||
with pytest.raises(TimelineCreate406):
|
|
||||||
new_timeline_id = TimelineId.generate()
|
|
||||||
log.info(f"Expecting failure for branch behind gc'd LSN, new_timeline_id={new_timeline_id}")
|
|
||||||
pageserver_http.timeline_create(
|
|
||||||
env.pg_version, env.initial_tenant, new_timeline_id, branch_behind_timeline_id, gced_lsn
|
|
||||||
)
|
|
||||||
|
|
||||||
# check that after gc everything is still there
|
# check that after gc everything is still there
|
||||||
assert query_scalar(hundred_cur, "SELECT count(*) FROM foo") == 100
|
assert query_scalar(hundred_cur, "SELECT count(*) FROM foo") == 100
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ import copy
|
|||||||
import os
|
import os
|
||||||
import shutil
|
import shutil
|
||||||
import subprocess
|
import subprocess
|
||||||
import tempfile
|
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from typing import Any, Optional
|
from typing import Any, Optional
|
||||||
|
|
||||||
@@ -449,7 +448,7 @@ def dump_differs(first: Path, second: Path, output: Path) -> bool:
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
with output.open("w") as stdout:
|
with output.open("w") as stdout:
|
||||||
res = subprocess.run(
|
rv = subprocess.run(
|
||||||
[
|
[
|
||||||
"diff",
|
"diff",
|
||||||
"--unified", # Make diff output more readable
|
"--unified", # Make diff output more readable
|
||||||
@@ -461,53 +460,4 @@ def dump_differs(first: Path, second: Path, output: Path) -> bool:
|
|||||||
stdout=stdout,
|
stdout=stdout,
|
||||||
)
|
)
|
||||||
|
|
||||||
differs = res.returncode != 0
|
return rv.returncode != 0
|
||||||
|
|
||||||
# TODO: Remove after https://github.com/neondatabase/neon/pull/4425 is merged, and a couple of releases are made
|
|
||||||
if differs:
|
|
||||||
with tempfile.NamedTemporaryFile(mode="w") as tmp:
|
|
||||||
tmp.write(PR4425_ALLOWED_DIFF)
|
|
||||||
tmp.flush()
|
|
||||||
|
|
||||||
allowed = subprocess.run(
|
|
||||||
[
|
|
||||||
"diff",
|
|
||||||
"--unified", # Make diff output more readable
|
|
||||||
r"--ignore-matching-lines=^---", # Ignore diff headers
|
|
||||||
r"--ignore-matching-lines=^\+\+\+", # Ignore diff headers
|
|
||||||
"--ignore-matching-lines=^@@", # Ignore diff blocks location
|
|
||||||
"--ignore-matching-lines=^ *$", # Ignore lines with only spaces
|
|
||||||
"--ignore-matching-lines=^ --.*", # Ignore the " --" lines for compatibility with PG14
|
|
||||||
"--ignore-blank-lines",
|
|
||||||
str(output),
|
|
||||||
str(tmp.name),
|
|
||||||
],
|
|
||||||
)
|
|
||||||
|
|
||||||
differs = allowed.returncode != 0
|
|
||||||
|
|
||||||
return differs
|
|
||||||
|
|
||||||
|
|
||||||
PR4425_ALLOWED_DIFF = """
|
|
||||||
--- /tmp/test_output/test_backward_compatibility[release-pg15]/compatibility_snapshot/dump.sql 2023-06-08 18:12:45.000000000 +0000
|
|
||||||
+++ /tmp/test_output/test_backward_compatibility[release-pg15]/dump.sql 2023-06-13 07:25:35.211733653 +0000
|
|
||||||
@@ -13,12 +13,20 @@
|
|
||||||
|
|
||||||
CREATE ROLE cloud_admin;
|
|
||||||
ALTER ROLE cloud_admin WITH SUPERUSER INHERIT CREATEROLE CREATEDB LOGIN REPLICATION BYPASSRLS;
|
|
||||||
+CREATE ROLE neon_superuser;
|
|
||||||
+ALTER ROLE neon_superuser WITH NOSUPERUSER INHERIT CREATEROLE CREATEDB NOLOGIN NOREPLICATION NOBYPASSRLS;
|
|
||||||
|
|
||||||
--
|
|
||||||
-- User Configurations
|
|
||||||
--
|
|
||||||
|
|
||||||
|
|
||||||
+--
|
|
||||||
+-- Role memberships
|
|
||||||
+--
|
|
||||||
+
|
|
||||||
+GRANT pg_read_all_data TO neon_superuser GRANTED BY cloud_admin;
|
|
||||||
+GRANT pg_write_all_data TO neon_superuser GRANTED BY cloud_admin;
|
|
||||||
"""
|
|
||||||
|
|||||||
@@ -33,7 +33,6 @@ def handle_role(dbs, roles, operation):
|
|||||||
dbs[db] = operation["name"]
|
dbs[db] = operation["name"]
|
||||||
if "password" in operation:
|
if "password" in operation:
|
||||||
roles[operation["name"]] = operation["password"]
|
roles[operation["name"]] = operation["password"]
|
||||||
assert "encrypted_password" in operation
|
|
||||||
elif operation["op"] == "del":
|
elif operation["op"] == "del":
|
||||||
if "old_name" in operation:
|
if "old_name" in operation:
|
||||||
roles.pop(operation["old_name"])
|
roles.pop(operation["old_name"])
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import shutil
|
import shutil
|
||||||
import time
|
import time
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
|
from pathlib import Path
|
||||||
from typing import Dict, Tuple
|
from typing import Dict, Tuple
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
@@ -427,14 +428,14 @@ def poor_mans_du(
|
|||||||
largest_layer = 0
|
largest_layer = 0
|
||||||
smallest_layer = None
|
smallest_layer = None
|
||||||
for tenant_id, timeline_id in timelines:
|
for tenant_id, timeline_id in timelines:
|
||||||
timeline_dir = env.timeline_dir(tenant_id, timeline_id)
|
dir = Path(env.repo_dir) / "tenants" / str(tenant_id) / "timelines" / str(timeline_id)
|
||||||
assert timeline_dir.exists(), f"timeline dir does not exist: {timeline_dir}"
|
assert dir.exists(), f"timeline dir does not exist: {dir}"
|
||||||
total = 0
|
sum = 0
|
||||||
for file in timeline_dir.iterdir():
|
for file in dir.iterdir():
|
||||||
if "__" not in file.name:
|
if "__" not in file.name:
|
||||||
continue
|
continue
|
||||||
size = file.stat().st_size
|
size = file.stat().st_size
|
||||||
total += size
|
sum += size
|
||||||
largest_layer = max(largest_layer, size)
|
largest_layer = max(largest_layer, size)
|
||||||
if smallest_layer:
|
if smallest_layer:
|
||||||
smallest_layer = min(smallest_layer, size)
|
smallest_layer = min(smallest_layer, size)
|
||||||
@@ -442,8 +443,8 @@ def poor_mans_du(
|
|||||||
smallest_layer = size
|
smallest_layer = size
|
||||||
log.info(f"{tenant_id}/{timeline_id} => {file.name} {size}")
|
log.info(f"{tenant_id}/{timeline_id} => {file.name} {size}")
|
||||||
|
|
||||||
log.info(f"{tenant_id}/{timeline_id}: sum {total}")
|
log.info(f"{tenant_id}/{timeline_id}: sum {sum}")
|
||||||
total_on_disk += total
|
total_on_disk += sum
|
||||||
|
|
||||||
assert smallest_layer is not None or total_on_disk == 0 and largest_layer == 0
|
assert smallest_layer is not None or total_on_disk == 0 and largest_layer == 0
|
||||||
return (total_on_disk, largest_layer, smallest_layer or 0)
|
return (total_on_disk, largest_layer, smallest_layer or 0)
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user