Compare commits
396 Commits
v6.0.0-bet
...
v7.6.0
Author | SHA1 | Date | |
---|---|---|---|
d55c62c073 | |||
4833c34800 | |||
fc70e657f0 | |||
ee23f629f6 | |||
44402c9571 | |||
ffefb38161 | |||
6d667f653e | |||
1c75fed727 | |||
e7837aea88 | |||
9c133be779 | |||
71eb953fd3 | |||
f49ef21fed | |||
6a6fa04ba0 | |||
83b0838c94 | |||
4ebc1e671f | |||
08c7e38587 | |||
b863d9feb3 | |||
e527f043b0 | |||
58bb403787 | |||
e4725c23eb | |||
b0db8caf65 | |||
3bcc6bdf93 | |||
eafb75a958 | |||
31ca0939aa | |||
7784fdcd6a | |||
8247eef735 | |||
cb6629f301 | |||
3a6fe1b374 | |||
0ba2f37004 | |||
e052dee753 | |||
9c2ec32d12 | |||
1669c38bc9 | |||
c6ce6d1b49 | |||
bc242b0aa7 | |||
41b67f6af4 | |||
bef21e1cb9 | |||
8c73630f5a | |||
724953d5b7 | |||
a22b231982 | |||
910bfe2318 | |||
70a524da46 | |||
bf6c846fac | |||
b83e4bef3f | |||
9f7fe0d8f7 | |||
741dee57e4 | |||
fff4dba708 | |||
f4f7ab3e49 | |||
88fe99b1b8 | |||
92c1486f6a | |||
caea64cef3 | |||
90783d8ee8 | |||
be57801e21 | |||
ff84786b4e | |||
1e863672cb | |||
fb98a9c383 | |||
05163f22cb | |||
160f12d7d3 | |||
49e4e36184 | |||
c4f8879cda | |||
8f54166653 | |||
b9f020c447 | |||
c357f3eb4d | |||
7ebbb0417a | |||
6e4b4173b5 | |||
87ebad7efb | |||
3294aaed3b | |||
0e21f3eab6 | |||
9fcf692cb8 | |||
da577ea3cc | |||
6ae1d8938a | |||
3e18a7390c | |||
5f43f1afc6 | |||
2fc9c03430 | |||
d951a9ba02 | |||
93385af675 | |||
dd75d0ece7 | |||
dcd37ed916 | |||
2e4d722d7f | |||
09543400ca | |||
8b101e5043 | |||
b31fff9c2b | |||
0c5b100dd9 | |||
253825a35e | |||
8937d19891 | |||
0fdd9e75a6 | |||
77da00c2c5 | |||
3744080d11 | |||
c9e546a8fd | |||
6691992a79 | |||
1ea0f4c339 | |||
8bfa117be2 | |||
b3acecdcea | |||
ec479c7e91 | |||
fd7760d9ed | |||
c9fcec6889 | |||
fd901ef2cf | |||
8afdaa8f0e | |||
254bfccc62 | |||
5b4aeca63c | |||
17871daf0c | |||
cdd4460968 | |||
fa6a37880b | |||
d4e1dabe12 | |||
a3fd376b24 | |||
aaac1f54e8 | |||
41c0329822 | |||
74d48fd7e1 | |||
9c3c953129 | |||
f5cae18da7 | |||
59d47592d9 | |||
2b6c991190 | |||
26020ba8bb | |||
b573bc20b5 | |||
210dbfa265 | |||
b37cac93ff | |||
eea953efb6 | |||
7ad9d7b291 | |||
20937c4f72 | |||
dbbfa07639 | |||
9e1a4cad5c | |||
02bbedcfca | |||
cd70d90914 | |||
819f297de8 | |||
0608adde89 | |||
ad7bcf4669 | |||
2eccc86e83 | |||
16d18f23a1 | |||
5631ae1b6c | |||
5fb29992f6 | |||
910d633ac2 | |||
32f8380e56 | |||
43e4dd6bb6 | |||
4f0b1688db | |||
9e75ee09bb | |||
9ae8822e00 | |||
babffd1fe6 | |||
5615d62032 | |||
4b89d15c1e | |||
815f510d5f | |||
199ba193be | |||
4ae9bd3f9a | |||
1c9cf639ea | |||
0040464ca1 | |||
79997efbb6 | |||
0e42009798 | |||
93fdcb8739 | |||
aca926e202 | |||
9941027b10 | |||
9104de4290 | |||
5dc692809c | |||
8dc1d1bd21 | |||
fe588485a9 | |||
19ef1d7025 | |||
62523a8662 | |||
6e97665e2e | |||
4988680353 | |||
c5de5c20c7 | |||
1a0fee1aa2 | |||
06a44603cd | |||
e48459762e | |||
235ebeae97 | |||
dfe909606e | |||
6fd0c7726c | |||
819e045811 | |||
1ba780598d | |||
aeb0cb3110 | |||
88923838c5 | |||
df9f6fd7fd | |||
98e46d6ac9 | |||
daff614fb4 | |||
5ea324c7f2 | |||
23fedbf94a | |||
593d66d8d6 | |||
851dcd5bf7 | |||
2e919681ae | |||
5da68cd48c | |||
27fdaeff46 | |||
53c0079656 | |||
93780b77e0 | |||
b712ed0421 | |||
ee96f1b345 | |||
d13464df3d | |||
6bde2e4d96 | |||
0a4953c020 | |||
96c488880c | |||
7e0adf3f66 | |||
09f716440a | |||
2251c84171 | |||
5cfe78dcd1 | |||
6a48325132 | |||
294be0a79a | |||
c94b264b44 | |||
7220c4e3e3 | |||
5aadeba2ec | |||
0f47a5b6ba | |||
36f32d28f2 | |||
6d69ccf229 | |||
37073b42be | |||
837501c948 | |||
b300966fa8 | |||
730eb06c84 | |||
aca8d3f4b7 | |||
b5b3af4659 | |||
6cd231426d | |||
0c7cd1cd75 | |||
2425704ead | |||
4e22159206 | |||
52cf1ba02c | |||
516e84182f | |||
a3a9853e18 | |||
08e26600fd | |||
7793c2c6ba | |||
9e826d16dd | |||
80618bbd9c | |||
38ad47ea75 | |||
45ed359bef | |||
fcc26c3e7a | |||
d4ff6b1f40 | |||
557de34eea | |||
e034dc4d90 | |||
cfbd1e5e4b | |||
0df661819f | |||
1a9f6d10d4 | |||
a787215c95 | |||
64ab400af5 | |||
a463878bf2 | |||
9f72024c61 | |||
243fbd4dc9 | |||
4e6a290693 | |||
ac05d529ca | |||
b38d79004a | |||
f4a547df11 | |||
2b87c35058 | |||
b11833e450 | |||
fa8e119514 | |||
677cb5c330 | |||
6e74c79bfe | |||
54474f5908 | |||
99cc0f519b | |||
92a01f89ef | |||
fd83a0c743 | |||
988e46c875 | |||
f081c2fc20 | |||
b4b376a1a5 | |||
0db4179d47 | |||
795b7c6234 | |||
091b9a57f5 | |||
564e1422ac | |||
8ed4ed3fc4 | |||
29fe4566a7 | |||
ae3bfb28ed | |||
14aab97d8a | |||
52d7a47cd7 | |||
f338dcbeed | |||
dcec058a22 | |||
2bdc6b156b | |||
84ca9e6b81 | |||
11cb0fd2db | |||
3f620ffb6f | |||
1a0e05d073 | |||
a4d2de23a1 | |||
85cecc9811 | |||
9899f742a8 | |||
b5484740b7 | |||
016b15b437 | |||
6fb936798e | |||
a692b87843 | |||
19663885a4 | |||
49b87777f9 | |||
d4523bb1e6 | |||
e3200899e2 | |||
36c7a1ab9e | |||
c54fbd5eca | |||
bbe828071e | |||
23f6c7db00 | |||
b1ea9e7a71 | |||
fb71d0e272 | |||
fa72a29999 | |||
af77b31d54 | |||
8280dace26 | |||
ecaf1c7b7c | |||
8702ec29a8 | |||
d8206434bc | |||
c71c2a8710 | |||
e55b881017 | |||
ab906ec417 | |||
0b1ff529f7 | |||
85a6835748 | |||
259271bc0f | |||
b7bc0f178b | |||
688455d0aa | |||
3c96d2ea42 | |||
ab81481e5a | |||
a429ad5dcf | |||
5e1c5b510b | |||
9e63183f4b | |||
b1e740f026 | |||
ce4ea55438 | |||
18ab7cd22f | |||
8807743daf | |||
aad50377ff | |||
4b3ae58ea7 | |||
ce2c68ecc9 | |||
0c155a7a2e | |||
afddfe8b58 | |||
5fa0915271 | |||
6a0a170b17 | |||
4dde5b6e45 | |||
4b93a1cb9e | |||
e3a0639a0c | |||
4d3220820b | |||
a4ac9fb0f3 | |||
1ff79ecf07 | |||
1166b16420 | |||
213224942f | |||
ff16e66275 | |||
3c338e983f | |||
2c11ba6520 | |||
9a21656706 | |||
e96ee5ba53 | |||
b421633a8a | |||
e2e0d62560 | |||
c71fb06940 | |||
e2171af99c | |||
8cebf049d4 | |||
ef139ed1cc | |||
d717de006a | |||
a44f091878 | |||
1b37ba5339 | |||
bbaa90e997 | |||
86e6c4a419 | |||
4159883791 | |||
d8b00da3a1 | |||
a24945bc1b | |||
158759493f | |||
36e32d6ddc | |||
84908e2ec0 | |||
a2dc51d811 | |||
fb3b0e2c29 | |||
1a3e4c68bb | |||
11b2342da0 | |||
80d4a808d3 | |||
da4146eb59 | |||
a0be35db8b | |||
14db9cd523 | |||
0c315385dd | |||
c0a0eb02fb | |||
ee407c32ad | |||
9262d21829 | |||
a13f710325 | |||
eac1a6036f | |||
987f3d7586 | |||
875322669c | |||
33a264b3d0 | |||
c059eff170 | |||
b4a22fc9dd | |||
6d1cbdc463 | |||
2bfbba4daf | |||
21ffe82bde | |||
8e6f597027 | |||
16c5065560 | |||
c4b985f1a4 | |||
042747c7d2 | |||
e4a46f31de | |||
6d9e62d2b4 | |||
9caaa507b1 | |||
5c7d3c5b44 | |||
8bac57d87a | |||
b8d759cd63 | |||
da72e3e5ac | |||
2afd36fee0 | |||
b7e75d8828 | |||
30e20f4e7d | |||
ce0ab8dccf | |||
5b20ab2f7c | |||
daaaed43df | |||
3a4bd791ad | |||
eecddd7f6b | |||
a34eaa136e | |||
53be8b5e96 | |||
f0ae5ea908 | |||
9910556a8b | |||
5997416e1b | |||
9a9fc56f85 | |||
2a5e919f29 | |||
8031d51e15 | |||
56ce9c0d0d | |||
8cd584cbd5 | |||
f5b87f4669 | |||
a1a65c5529 | |||
832434095e | |||
b85f1ef351 | |||
8bee5d788e | |||
0752d857e2 | |||
07e4056694 | |||
0eb4ab85b3 |
.gitattributesindex.tsindex.tsxindex.tskeycloakJsAdapter.ts
.github
.gitignore.prettierignore.prettierrc.jsonREADME.mdpackage.jsonrenovate.jsonscripts
src
account
bin
build-keycloak-theme
build-keycloak-theme.ts
create-keycloak-email-directory.tsdownload-builtin-keycloak-theme.tseject-keycloak-page.tsgenerate-i18n-messages.tsinitialize-email-theme.tsgenerateFtl
generateKeycloakThemeResources.tsgenerateStartKeycloakTestingContainer.tsindex.tsreplaceImportFromStatic.tskeycloakify
BuildOptions.tsbuild-paths.tsftlValuesGlobalName.ts
link_in_test_app.tspromptKeycloakVersion.tsgenerateFtl
generateJavaStackFiles.tsgenerateKeycloakThemeResources.tsgenerateStartKeycloakTestingContainer.tsindex.tskeycloakify.tsparsedPackageJson.tsreplacers
tools
lib
components
Error.tsxInfo.tsxKcApp.tsxKcProps.tsLogin.tsxLoginConfigTotp.tsxLoginIdpLinkConfirm.tsxLoginIdpLinkEmail.tsxLoginOtp.tsxLoginPageExpired.tsxLoginResetPassword.tsxLoginUpdatePassword.tsxLoginUpdateProfile.tsxLoginVerifyEmail.tsxLogoutConfirm.tsxRegister.tsxRegisterUserProfile.tsxTerms.tsx
getKcContext
i18n
generated_messages
11.0.3
account
admin
email
login
15.0.2
account
ca.tscs.tsda.tsde.tsen.tses.tsfr.tshu.tsit.tsja.tslt.tsnl.tsno.tspl.tspt-BR.tsru.tssk.tssv.tstr.tszh-CN.ts
admin
email
ca.tscs.tsda.tsde.tsen.tses.tsfr.tshu.tsit.tsja.tslt.tsnl.tsno.tspl.tspt-BR.tsru.tssk.tssv.tstr.tszh-CN.ts
login
18.0.1
tools
useGetClassName.tsusePrepareTemplate.tslogin
Fallback.tsxTemplate.tsxTemplateProps.ts
i18n
index.tskcContext
lib
pages
Error.tsxIdpReviewUserProfile.tsxInfo.tsxLogin.tsxLoginConfigTotp.tsxLoginIdpLinkConfirm.tsxLoginIdpLinkEmail.tsxLoginOtp.tsxLoginPageExpired.tsxLoginPassword.tsxLoginResetPassword.tsxLoginUpdatePassword.tsxLoginUpdateProfile.tsxLoginUsername.tsxLoginVerifyEmail.tsxLogoutConfirm.tsxPageProps.tsRegister.tsxRegisterUserProfile.tsxSelectAuthenticator.tsxTerms.tsxUpdateEmail.tsxUpdateUserProfile.tsxWebauthnAuthenticate.tsx
shared
test
bin
generateKeycloakThemeResources.tsindex.tsmain.tsreplaceImportFromStatic.tssetupSampleReactProject.ts
lib
tools
tools
AndByDiscriminatingKey.tsArray.prototype.every.tsDeepPartial.tsHTMLElement.prototype.prepend.tsMarkdown.tsassert.tsclsx.tsdeepAssign.tsdeepClone.tsemailRegExp.tsheadInsert.tsmemoize.tspathBasename.tsuseConst.tsuseConstCallback.ts
tsconfig.jsontest
tsproject.jsonvitest.config.tsyarn.lock
2
.gitattributes
vendored
2
.gitattributes
vendored
@ -1,3 +1,3 @@
|
|||||||
src/lib/i18n/generated_kcMessages/* linguist-documentation
|
src/lib/i18n/generated_kcMessages/* linguist-documentation
|
||||||
src/bin/build-keycloak-theme/index.ts -linguist-detectable
|
src/bin/keycloakify/index.ts -linguist-detectable
|
||||||
src/bin/install-builtin-keycloak-themes.ts -linguist-detectable
|
src/bin/install-builtin-keycloak-themes.ts -linguist-detectable
|
||||||
|
4
.github/FUNDING.yaml
vendored
Normal file
4
.github/FUNDING.yaml
vendored
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
# These are supported funding model platforms
|
||||||
|
|
||||||
|
github: [garronej]
|
||||||
|
custom: ['https://www.ringerhq.com/experts/garronej']
|
46
.github/workflows/ci.yaml
vendored
46
.github/workflows/ci.yaml
vendored
@ -13,10 +13,10 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
if: ${{ !github.event.created && github.repository != 'garronej/ts-ci' }}
|
if: ${{ !github.event.created && github.repository != 'garronej/ts-ci' }}
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2.3.4
|
- uses: actions/checkout@v3
|
||||||
- uses: actions/setup-node@v2.1.3
|
- uses: actions/setup-node@v3
|
||||||
- uses: bahmutov/npm-install@v1
|
- uses: bahmutov/npm-install@v1
|
||||||
- name: If this step fails run 'npm run lint' and 'npm run format' then commit again.
|
- name: If this step fails run 'yarn format' then commit again.
|
||||||
run: |
|
run: |
|
||||||
PACKAGE_MANAGER=npm
|
PACKAGE_MANAGER=npm
|
||||||
if [ -f "./yarn.lock" ]; then
|
if [ -f "./yarn.lock" ]; then
|
||||||
@ -24,22 +24,21 @@ jobs:
|
|||||||
fi
|
fi
|
||||||
$PACKAGE_MANAGER run format:check
|
$PACKAGE_MANAGER run format:check
|
||||||
test:
|
test:
|
||||||
runs-on: macos-10.15
|
runs-on: ${{ matrix.os }}
|
||||||
needs: test_lint
|
needs: test_lint
|
||||||
env:
|
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
node: [ '15', '14' ]
|
node: [ '16' ]
|
||||||
name: Test with Node v${{ matrix.node }}
|
os: [ ubuntu-latest ]
|
||||||
|
name: Test with Node v${{ matrix.node }} on ${{ matrix.os }}
|
||||||
steps:
|
steps:
|
||||||
- name: Tell if project is using npm or yarn
|
- name: Tell if project is using npm or yarn
|
||||||
id: step1
|
id: step1
|
||||||
uses: garronej/ts-ci@v1.1.4
|
uses: garronej/ts-ci@v2.0.2
|
||||||
with:
|
with:
|
||||||
action_name: tell_if_project_uses_npm_or_yarn
|
action_name: tell_if_project_uses_npm_or_yarn
|
||||||
- uses: actions/checkout@v2.3.4
|
- uses: actions/checkout@v3
|
||||||
- uses: actions/setup-node@v2.1.3
|
- uses: actions/setup-node@v3
|
||||||
with:
|
with:
|
||||||
node-version: ${{ matrix.node }}
|
node-version: ${{ matrix.node }}
|
||||||
- uses: bahmutov/npm-install@v1
|
- uses: bahmutov/npm-install@v1
|
||||||
@ -65,9 +64,9 @@ jobs:
|
|||||||
from_version: ${{ steps.step1.outputs.from_version }}
|
from_version: ${{ steps.step1.outputs.from_version }}
|
||||||
to_version: ${{ steps.step1.outputs.to_version }}
|
to_version: ${{ steps.step1.outputs.to_version }}
|
||||||
is_upgraded_version: ${{ steps.step1.outputs.is_upgraded_version }}
|
is_upgraded_version: ${{ steps.step1.outputs.is_upgraded_version }}
|
||||||
is_release_beta: ${{steps.step1.outputs.is_release_beta }}
|
is_pre_release: ${{steps.step1.outputs.is_pre_release }}
|
||||||
steps:
|
steps:
|
||||||
- uses: garronej/ts-ci@v1.1.4
|
- uses: garronej/ts-ci@v2.0.2
|
||||||
id: step1
|
id: step1
|
||||||
with:
|
with:
|
||||||
action_name: is_package_json_version_upgraded
|
action_name: is_package_json_version_upgraded
|
||||||
@ -75,13 +74,13 @@ jobs:
|
|||||||
|
|
||||||
create_github_release:
|
create_github_release:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
# We create a release only if the version have been upgraded and we are on a default branch
|
# We create a release only if the version have been upgraded and we are on the main branch
|
||||||
# PR on the default branch can release beta but not real release
|
# or if we are on a branch of the repo that has an PR open on main.
|
||||||
if: |
|
if: |
|
||||||
needs.check_if_version_upgraded.outputs.is_upgraded_version == 'true' &&
|
needs.check_if_version_upgraded.outputs.is_upgraded_version == 'true' &&
|
||||||
(
|
(
|
||||||
github.event_name == 'push' ||
|
github.event_name == 'push' ||
|
||||||
needs.check_if_version_upgraded.outputs.is_release_beta == 'true'
|
needs.check_if_version_upgraded.outputs.is_pre_release == 'true'
|
||||||
)
|
)
|
||||||
needs:
|
needs:
|
||||||
- check_if_version_upgraded
|
- check_if_version_upgraded
|
||||||
@ -93,7 +92,7 @@ jobs:
|
|||||||
target_commitish: ${{ github.head_ref || github.ref }}
|
target_commitish: ${{ github.head_ref || github.ref }}
|
||||||
generate_release_notes: true
|
generate_release_notes: true
|
||||||
draft: false
|
draft: false
|
||||||
prerelease: ${{ needs.check_if_version_upgraded.outputs.is_release_beta == 'true' }}
|
prerelease: ${{ needs.check_if_version_upgraded.outputs.is_pre_release == 'true' }}
|
||||||
env:
|
env:
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
@ -103,12 +102,11 @@ jobs:
|
|||||||
- create_github_release
|
- create_github_release
|
||||||
- check_if_version_upgraded
|
- check_if_version_upgraded
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2.3.4
|
- uses: actions/checkout@v3
|
||||||
with:
|
with:
|
||||||
ref: ${{ github.ref }}
|
ref: ${{ github.ref }}
|
||||||
- uses: actions/setup-node@v2.1.3
|
- uses: actions/setup-node@v3
|
||||||
with:
|
with:
|
||||||
node-version: '15'
|
|
||||||
registry-url: https://registry.npmjs.org/
|
registry-url: https://registry.npmjs.org/
|
||||||
- uses: bahmutov/npm-install@v1
|
- uses: bahmutov/npm-install@v1
|
||||||
- run: |
|
- run: |
|
||||||
@ -117,7 +115,7 @@ jobs:
|
|||||||
PACKAGE_MANAGER=yarn
|
PACKAGE_MANAGER=yarn
|
||||||
fi
|
fi
|
||||||
$PACKAGE_MANAGER run build
|
$PACKAGE_MANAGER run build
|
||||||
- run: npx -y -p denoify@0.11.7 enable_short_npm_import_path
|
- run: npx -y -p denoify@1.2.2 enable_short_npm_import_path
|
||||||
env:
|
env:
|
||||||
DRY_RUN: "0"
|
DRY_RUN: "0"
|
||||||
- name: Publishing on NPM
|
- name: Publishing on NPM
|
||||||
@ -131,11 +129,11 @@ jobs:
|
|||||||
false
|
false
|
||||||
fi
|
fi
|
||||||
EXTRA_ARGS=""
|
EXTRA_ARGS=""
|
||||||
if [ "$IS_BETA" = "true" ]; then
|
if [ "$IS_PRE_RELEASE" = "true" ]; then
|
||||||
EXTRA_ARGS="--tag beta"
|
EXTRA_ARGS="--tag next"
|
||||||
fi
|
fi
|
||||||
npm publish $EXTRA_ARGS
|
npm publish $EXTRA_ARGS
|
||||||
env:
|
env:
|
||||||
NODE_AUTH_TOKEN: ${{secrets.NPM_TOKEN}}
|
NODE_AUTH_TOKEN: ${{secrets.NPM_TOKEN}}
|
||||||
VERSION: ${{ needs.check_if_version_upgraded.outputs.to_version }}
|
VERSION: ${{ needs.check_if_version_upgraded.outputs.to_version }}
|
||||||
IS_BETA: ${{ needs.check_if_version_upgraded.outputs.is_release_beta }}
|
IS_PRE_RELEASE: ${{ needs.check_if_version_upgraded.outputs.is_pre_release }}
|
||||||
|
11
.gitignore
vendored
11
.gitignore
vendored
@ -41,12 +41,15 @@ jspm_packages
|
|||||||
.DS_Store
|
.DS_Store
|
||||||
|
|
||||||
/dist
|
/dist
|
||||||
/dist_test
|
/keycloakify_starter_test/
|
||||||
|
/sample_custom_react_project/
|
||||||
/sample_react_project/
|
/sample_react_project/
|
||||||
/.yarn_home/
|
/.yarn_home/
|
||||||
|
|
||||||
.idea
|
.idea
|
||||||
|
|
||||||
/keycloak_email
|
/src/login/i18n/baseMessages/
|
||||||
/build_keycloak
|
/src/account/i18n/baseMessages/
|
||||||
|
|
||||||
|
# VS Code devcontainers
|
||||||
|
.devcontainer
|
@ -1,9 +1,15 @@
|
|||||||
node_modules/
|
node_modules/
|
||||||
/dist/
|
/dist/
|
||||||
/dist_test/
|
|
||||||
/CHANGELOG.md
|
/CHANGELOG.md
|
||||||
/.yarn_home/
|
/.yarn_home/
|
||||||
/src/test/apps/
|
/src/test/apps/
|
||||||
/src/tools/types/
|
/src/tools/types/
|
||||||
/sample_react_project
|
/build_keycloak/
|
||||||
/build_keycloak/
|
/.vscode/
|
||||||
|
/src/login/i18n/baseMessages/
|
||||||
|
/src/account/i18n/baseMessages/
|
||||||
|
# Test Build Directories
|
||||||
|
/dist_test
|
||||||
|
/sample_react_project/
|
||||||
|
/sample_custom_react_project/
|
||||||
|
/keycloakify_starter_test/
|
@ -5,7 +5,7 @@
|
|||||||
"semi": true,
|
"semi": true,
|
||||||
"singleQuote": false,
|
"singleQuote": false,
|
||||||
"quoteProps": "preserve",
|
"quoteProps": "preserve",
|
||||||
"trailingComma": "all",
|
"trailingComma": "none",
|
||||||
"bracketSpacing": true,
|
"bracketSpacing": true,
|
||||||
"arrowParens": "avoid"
|
"arrowParens": "avoid"
|
||||||
}
|
}
|
||||||
|
104
README.md
104
README.md
@ -2,22 +2,19 @@
|
|||||||
<img src="https://user-images.githubusercontent.com/6702424/109387840-eba11f80-7903-11eb-9050-db1dad883f78.png">
|
<img src="https://user-images.githubusercontent.com/6702424/109387840-eba11f80-7903-11eb-9050-db1dad883f78.png">
|
||||||
</p>
|
</p>
|
||||||
<p align="center">
|
<p align="center">
|
||||||
<i>🔏 Create Keycloak themes using React 🔏</i>
|
<i>🔏 Create Keycloak themes using React 🔏</i>
|
||||||
<br>
|
<br>
|
||||||
<br>
|
<br>
|
||||||
<a href="https://github.com/garronej/keycloakify/actions">
|
<a href="https://github.com/garronej/keycloakify/actions">
|
||||||
<img src="https://github.com/garronej/keycloakify/workflows/ci/badge.svg?branch=main">
|
<img src="https://github.com/garronej/keycloakify/workflows/ci/badge.svg?branch=main">
|
||||||
</a>
|
</a>
|
||||||
<a href="https://bundlephobia.com/package/keycloakify">
|
|
||||||
<img src="https://img.shields.io/bundlephobia/minzip/keycloakify">
|
|
||||||
</a>
|
|
||||||
<a href="https://www.npmjs.com/package/keycloakify">
|
<a href="https://www.npmjs.com/package/keycloakify">
|
||||||
<img src="https://img.shields.io/npm/dm/keycloakify">
|
<img src="https://img.shields.io/npm/dm/keycloakify">
|
||||||
</a>
|
</a>
|
||||||
<a href="https://github.com/garronej/keycloakify/blob/main/LICENSE">
|
<a href="https://github.com/garronej/keycloakify/blob/main/LICENSE">
|
||||||
<img src="https://img.shields.io/npm/l/keycloakify">
|
<img src="https://img.shields.io/npm/l/keycloakify">
|
||||||
</a>
|
</a>
|
||||||
<a href="https://github.com/InseeFrLab/keycloakify/blob/729503fe31a155a823f46dd66ad4ff34ca274e0a/tsconfig.json#L14">
|
<a href="https://github.com/keycloakify/keycloakify/blob/729503fe31a155a823f46dd66ad4ff34ca274e0a/tsconfig.json#L14">
|
||||||
<img src="https://camo.githubusercontent.com/0f9fcc0ac1b8617ad4989364f60f78b2d6b32985ad6a508f215f14d8f897b8d3/68747470733a2f2f62616467656e2e6e65742f62616467652f547970655363726970742f7374726963742532302546302539462539322541412f626c7565">
|
<img src="https://camo.githubusercontent.com/0f9fcc0ac1b8617ad4989364f60f78b2d6b32985ad6a508f215f14d8f897b8d3/68747470733a2f2f62616467656e2e6e65742f62616467652f547970655363726970742f7374726963742532302546302539462539322541412f626c7565">
|
||||||
</a>
|
</a>
|
||||||
<a href="https://github.com/thomasdarimont/awesome-keycloak">
|
<a href="https://github.com/thomasdarimont/awesome-keycloak">
|
||||||
@ -27,8 +24,11 @@
|
|||||||
<a href="https://www.keycloakify.dev">Home</a>
|
<a href="https://www.keycloakify.dev">Home</a>
|
||||||
-
|
-
|
||||||
<a href="https://docs.keycloakify.dev">Documentation</a>
|
<a href="https://docs.keycloakify.dev">Documentation</a>
|
||||||
</p>
|
-
|
||||||
|
<a href="https://storybook.keycloakify.dev/storybook">Storybook</a>
|
||||||
|
-
|
||||||
|
<a href="https://github.com/codegouvfr/keycloakify-starter">Starter project</a>
|
||||||
|
</p>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<p align="center">
|
<p align="center">
|
||||||
@ -36,15 +36,89 @@
|
|||||||
<img src="https://user-images.githubusercontent.com/6702424/110260457-a1c3d380-7fac-11eb-853a-80459b65626b.png">
|
<img src="https://user-images.githubusercontent.com/6702424/110260457-a1c3d380-7fac-11eb-853a-80459b65626b.png">
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
|
The more ⭐️ the project gets, the more time I spend improving and maintaining it. Thank you for your support 😊
|
||||||
|
|
||||||
|
> 🗣 V7 have been released 🎉
|
||||||
|
> [It features major improvements](https://github.com/keycloakify/keycloakify#70-).
|
||||||
|
> Checkout [the migration guide](https://docs.keycloakify.dev/migration-guides/v6-greater-than-v7).
|
||||||
|
|
||||||
# Changelog highlights
|
# Changelog highlights
|
||||||
|
|
||||||
|
## 7.0 🍾
|
||||||
|
|
||||||
|
- Account theme support 🚀
|
||||||
|
- It's much easier to customize pages at the CSS level, you can now see in the browser dev tool the customizable classes.
|
||||||
|
- New interactive CLI tool `npx eject-keycloak-page`, that enables to select the page you want to customize at the component level.
|
||||||
|
- There is [a Storybook](https://storybook.keycloakify.dev)
|
||||||
|
- [Remember me is fixed](https://github.com/keycloakify/keycloakify/pull/272)
|
||||||
|
|
||||||
|
## 6.13
|
||||||
|
|
||||||
|
- Build work behind corporate proxies, [see issue](https://github.com/keycloakify/keycloakify/issues/257).
|
||||||
|
|
||||||
|
## 6.12
|
||||||
|
|
||||||
|
Massive improvement in the developer experience:
|
||||||
|
|
||||||
|
- There is now only one starter repo: https://github.com/codegouvfr/keycloakify-starter
|
||||||
|
- A lot of comments have been added in the code of the starter to make it easier to get started.
|
||||||
|
- The doc has been updated: https://docs.keycloakify.dev
|
||||||
|
- A lot of improvements in the type system.
|
||||||
|
|
||||||
|
## 6.11.4
|
||||||
|
|
||||||
|
- You no longer need to have Maven installed to build the theme. Thanks to @lordvlad, [see PR](https://github.com/keycloakify/keycloakify/pull/239).
|
||||||
|
- Feature new build options: [`bundler`](https://docs.keycloakify.dev/build-options#keycloakify.bundler), [`groupId`](https://docs.keycloakify.dev/build-options#keycloakify.groupid), [`artifactId`](https://docs.keycloakify.dev/build-options#keycloakify.artifactid), [`version`](https://docs.keycloakify.dev/build-options#version).
|
||||||
|
Theses options can be user to customize the output name of the .jar. You can use environnement variables to overrides the values read in the package.json. Thanks to @lordvlad.
|
||||||
|
|
||||||
|
## 6.10.0
|
||||||
|
|
||||||
|
- Widows compat (thanks to @lordvlad, [see PR](https://github.com/keycloakify/keycloakify/pull/226)). WSL is no longer required 🎉
|
||||||
|
|
||||||
|
## 6.8.4
|
||||||
|
|
||||||
|
- `@emotion/react` is no longer a peer dependency of Keycloakify.
|
||||||
|
|
||||||
|
## 6.8.0
|
||||||
|
|
||||||
|
- It is now possible to pass a custom `<Template />` component as a prop to `<KcApp />` and every
|
||||||
|
individual page (`<Login />`, `<RegisterUserProfile />`, ...) it enables to customize only the header and footer for
|
||||||
|
example without having to switch to a full-component level customization. [See issue](https://github.com/keycloakify/keycloakify/issues/191).
|
||||||
|
|
||||||
|
## 6.7.0
|
||||||
|
|
||||||
|
- Add support for `webauthn-authenticate.ftl` thanks to [@mstrodl](https://github.com/Mstrodl)'s hacktoberfest [PR](https://github.com/keycloakify/keycloakify/pull/185).
|
||||||
|
|
||||||
|
## 6.6.0
|
||||||
|
|
||||||
|
- Add support for `login-password.ftl` thanks to [@mstrodl](https://github.com/Mstrodl)'s hacktoberfest [PR](https://github.com/keycloakify/keycloakify/pull/184).
|
||||||
|
|
||||||
|
## 6.5.0
|
||||||
|
|
||||||
|
- Add support for `login-username.ftl` thanks to [@mstrodl](https://github.com/Mstrodl)'s hacktoberfest [PR](https://github.com/keycloakify/keycloakify/pull/183).
|
||||||
|
|
||||||
|
## 6.4.0
|
||||||
|
|
||||||
|
- You can now optionally pass a `doFetchDefaultThemeResources: boolean` prop to every page component and the default `<KcApp />`
|
||||||
|
This enables you to prevent the default CSS and JS that comes with the builtin Keycloak theme to be downloaded.
|
||||||
|
You'll get [a black slate](https://user-images.githubusercontent.com/6702424/192619083-4baa5df4-4a21-4ec7-8e28-d200d1208299.png).
|
||||||
|
|
||||||
|
## 6.0.0
|
||||||
|
|
||||||
|
- Bundle size drastically reduced, locals and component dynamically loaded.
|
||||||
|
- First print much quicker, use of React.lazy() everywhere.
|
||||||
|
- Real i18n API.
|
||||||
|
- Actual documentation for build options.
|
||||||
|
|
||||||
|
Checkout [the migration guide](https://docs.keycloakify.dev/v5-to-v6)
|
||||||
|
|
||||||
## 5.8.0
|
## 5.8.0
|
||||||
|
|
||||||
- [React.lazy()](https://reactjs.org/docs/code-splitting.html#reactlazy) support 🎉. [#141](https://github.com/InseeFrLab/keycloakify/issues/141)
|
- [React.lazy()](https://reactjs.org/docs/code-splitting.html#reactlazy) support 🎉. [#141](https://github.com/keycloakify/keycloakify/issues/141)
|
||||||
|
|
||||||
## 5.7.0
|
## 5.7.0
|
||||||
|
|
||||||
- Feat `logout-confirm.ftl`. [PR](https://github.com/InseeFrLab/keycloakify/pull/120)
|
- Feat `logout-confirm.ftl`. [PR](https://github.com/keycloakify/keycloakify/pull/120)
|
||||||
|
|
||||||
## 5.6.4
|
## 5.6.4
|
||||||
|
|
||||||
@ -52,7 +126,7 @@ Fix `login-verify-email.ftl` page. [Before](https://user-images.githubuserconten
|
|||||||
|
|
||||||
## v5.6.0
|
## v5.6.0
|
||||||
|
|
||||||
Add support for `login-config-totp.ftl` page [#127](https://github.com/InseeFrLab/keycloakify/pull/127).
|
Add support for `login-config-totp.ftl` page [#127](https://github.com/keycloakify/keycloakify/pull/127).
|
||||||
|
|
||||||
## v5.3.0
|
## v5.3.0
|
||||||
|
|
||||||
@ -67,7 +141,7 @@ Import of terms and services have changed. [See example](https://github.com/garr
|
|||||||
|
|
||||||
## v4.10.0
|
## v4.10.0
|
||||||
|
|
||||||
Add `login-idp-link-email.ftl` page [See PR](https://github.com/InseeFrLab/keycloakify/pull/92).
|
Add `login-idp-link-email.ftl` page [See PR](https://github.com/keycloakify/keycloakify/pull/92).
|
||||||
|
|
||||||
## v4.8.0
|
## v4.8.0
|
||||||
|
|
||||||
@ -80,7 +154,7 @@ Add `login-idp-link-email.ftl` page [See PR](https://github.com/InseeFrLab/keycl
|
|||||||
## v4.7.2
|
## v4.7.2
|
||||||
|
|
||||||
> WARNING: This is broken.
|
> WARNING: This is broken.
|
||||||
> Testing with local Keycloak container working with M1 Mac. Thanks to [@eduardosanzb](https://github.com/InseeFrLab/keycloakify/issues/43#issuecomment-975699658).
|
> Testing with local Keycloak container working with M1 Mac. Thanks to [@eduardosanzb](https://github.com/keycloakify/keycloakify/issues/43#issuecomment-975699658).
|
||||||
> Be aware: When running M1s you are testing with Keycloak v15 else the local container spun will be a Keycloak v16.1.0.
|
> Be aware: When running M1s you are testing with Keycloak v15 else the local container spun will be a Keycloak v16.1.0.
|
||||||
|
|
||||||
## v4.7.0
|
## v4.7.0
|
||||||
@ -114,12 +188,12 @@ Change [this](https://github.com/garronej/keycloakify-demo-app/blob/df664c13c77c
|
|||||||
|
|
||||||
No breaking changes except that `@emotion/react`, [`tss-react`](https://www.npmjs.com/package/tss-react) and [`powerhooks`](https://www.npmjs.com/package/powerhooks) are now `peerDependencies` instead of being just dependencies.
|
No breaking changes except that `@emotion/react`, [`tss-react`](https://www.npmjs.com/package/tss-react) and [`powerhooks`](https://www.npmjs.com/package/powerhooks) are now `peerDependencies` instead of being just dependencies.
|
||||||
It's important to avoid problem when using `keycloakify` alongside [`mui`](https://mui.com) and
|
It's important to avoid problem when using `keycloakify` alongside [`mui`](https://mui.com) and
|
||||||
[when passing params from the app to the login page](https://github.com/InseeFrLab/keycloakify#implement-context-persistence-optional).
|
[when passing params from the app to the login page](https://github.com/keycloakify/keycloakify#implement-context-persistence-optional).
|
||||||
|
|
||||||
## v2.5
|
## v2.5
|
||||||
|
|
||||||
- Feature [Use advanced message](https://github.com/InseeFrLab/keycloakify/blob/59f106bf9e210b63b190826da2bf5f75fc8b7644/src/lib/i18n/useKcMessage.tsx#L53-L66)
|
- Feature [Use advanced message](https://github.com/keycloakify/keycloakify/blob/59f106bf9e210b63b190826da2bf5f75fc8b7644/src/lib/i18n/useKcMessage.tsx#L53-L66)
|
||||||
and [`messagesPerFields`](https://github.com/InseeFrLab/keycloakify/blob/59f106bf9e210b63b190826da2bf5f75fc8b7644/src/lib/getKcContext/KcContextBase.ts#L70-L75) (implementation [here](https://github.com/InseeFrLab/keycloakify/blob/59f106bf9e210b63b190826da2bf5f75fc8b7644/src/bin/build-keycloak-theme/generateFtl/common.ftl#L130-L189))
|
and [`messagesPerFields`](https://github.com/keycloakify/keycloakify/blob/59f106bf9e210b63b190826da2bf5f75fc8b7644/src/lib/getKcContext/KcContextBase.ts#L70-L75) (implementation [here](https://github.com/keycloakify/keycloakify/blob/59f106bf9e210b63b190826da2bf5f75fc8b7644/src/bin/build-keycloak-theme/generateFtl/common.ftl#L130-L189))
|
||||||
- Test container now uses Keycloak version `15.0.2`.
|
- Test container now uses Keycloak version `15.0.2`.
|
||||||
|
|
||||||
## v2
|
## v2
|
||||||
|
70
package.json
Executable file → Normal file
70
package.json
Executable file → Normal file
@ -1,29 +1,35 @@
|
|||||||
{
|
{
|
||||||
"name": "keycloakify",
|
"name": "keycloakify",
|
||||||
"version": "6.0.0-beta.4",
|
"version": "7.6.0",
|
||||||
"description": "Keycloak theme generator for Reacts app",
|
"description": "Create Keycloak themes using React",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "git://github.com/garronej/keycloakify.git"
|
"url": "git://github.com/keycloakify/keycloakify.git"
|
||||||
},
|
},
|
||||||
"main": "dist/lib/index.js",
|
"main": "dist/index.js",
|
||||||
"types": "dist/lib/index.d.ts",
|
"types": "dist/index.d.ts",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "rimraf dist/ && tsc -p src/bin && tsc -p src/lib && yarn grant-exec-perms && yarn copy-files dist/",
|
"prepare": "yarn generate-i18n-messages",
|
||||||
"build:test": "rimraf dist_test/ && tsc -p src/test && yarn copy-files dist_test/",
|
"build": "rimraf dist/ && tsc -p src/bin && tsc -p src/tsconfig.json && tsc-alias -p src/tsconfig.json && yarn grant-exec-perms && yarn copy-files dist/",
|
||||||
|
"build:watch": "tsc -p src/tsconfig.json && (concurrently \"tsc -p src/tsconfig.json -w\" \"tsc-alias -p src/tsconfig.json\")",
|
||||||
"grant-exec-perms": "node dist/bin/tools/grant-exec-perms.js",
|
"grant-exec-perms": "node dist/bin/tools/grant-exec-perms.js",
|
||||||
"copy-files": "copyfiles -u 1 src/**/*.ftl",
|
"copy-files": "copyfiles -u 1 src/**/*.ftl",
|
||||||
"test": "yarn build:test && node dist_test/test/bin && node dist_test/test/lib",
|
"test": "yarn test:types && vitest run",
|
||||||
"generate-messages": "node dist/bin/generate-i18n-messages.js",
|
"test:keycloakify-starter": "ts-node scripts/test-keycloakify-starter",
|
||||||
"link_in_test_app": "node dist/bin/link_in_test_app.js",
|
"test:types": "tsc -p test/tsconfig.json --noEmit",
|
||||||
"_format": "prettier '**/*.{ts,tsx,json,md}'",
|
"_format": "prettier '**/*.{ts,tsx,json,md}'",
|
||||||
"format": "yarn _format --write",
|
"format": "yarn _format --write",
|
||||||
"format:check": "yarn _format --list-different"
|
"format:check": "yarn _format --list-different",
|
||||||
|
"generate-i18n-messages": "ts-node --skipProject scripts/generate-i18n-messages.ts",
|
||||||
|
"link-in-app": "ts-node --skipProject scripts/link-in-app.ts",
|
||||||
|
"link-in-starter": "yarn link-in-app keycloakify-starter",
|
||||||
|
"tsc-watch": "tsc -p src/bin -w & tsc -p src/lib -w "
|
||||||
},
|
},
|
||||||
"bin": {
|
"bin": {
|
||||||
"build-keycloak-theme": "dist/bin/build-keycloak-theme/index.js",
|
"keycloakify": "dist/bin/keycloakify/index.js",
|
||||||
"create-keycloak-email-directory": "dist/bin/create-keycloak-email-directory.js",
|
"initialize-email-theme": "dist/bin/initialize-email-theme.js",
|
||||||
"download-builtin-keycloak-theme": "dist/bin/download-builtin-keycloak-theme.js"
|
"download-builtin-keycloak-theme": "dist/bin/download-builtin-keycloak-theme.js",
|
||||||
|
"eject-keycloak-page": "dist/bin/eject-keycloak-page.js"
|
||||||
},
|
},
|
||||||
"lint-staged": {
|
"lint-staged": {
|
||||||
"*.{ts,tsx,json,md}": [
|
"*.{ts,tsx,json,md}": [
|
||||||
@ -40,7 +46,8 @@
|
|||||||
"files": [
|
"files": [
|
||||||
"src/",
|
"src/",
|
||||||
"dist/",
|
"dist/",
|
||||||
"!dist/tsconfig.tsbuildinfo"
|
"!dist/tsconfig.tsbuildinfo",
|
||||||
|
"!dist/bin/tsconfig.tsbuildinfo"
|
||||||
],
|
],
|
||||||
"keywords": [
|
"keywords": [
|
||||||
"bluehats",
|
"bluehats",
|
||||||
@ -52,16 +59,18 @@
|
|||||||
"login",
|
"login",
|
||||||
"register"
|
"register"
|
||||||
],
|
],
|
||||||
"homepage": "https://github.com/garronej/keycloakify",
|
"homepage": "https://www.keycloakify.dev",
|
||||||
"peerDependencies": {
|
"peerDependencies": {
|
||||||
"@emotion/react": "^11.4.1",
|
|
||||||
"react": "^16.8.0 || ^17.0.0 || ^18.0.0"
|
"react": "^16.8.0 || ^17.0.0 || ^18.0.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@emotion/react": "^11.4.1",
|
"@babel/core": "^7.0.0",
|
||||||
"@types/memoizee": "^0.4.7",
|
"@types/make-fetch-happen": "^10.0.1",
|
||||||
"@types/node": "^17.0.25",
|
"@types/minimist": "^1.2.2",
|
||||||
|
"@types/node": "^18.15.3",
|
||||||
"@types/react": "18.0.9",
|
"@types/react": "18.0.9",
|
||||||
|
"@types/yauzl": "^2.10.0",
|
||||||
|
"concurrently": "^7.6.0",
|
||||||
"copyfiles": "^2.4.1",
|
"copyfiles": "^2.4.1",
|
||||||
"husky": "^4.3.8",
|
"husky": "^4.3.8",
|
||||||
"lint-staged": "^11.0.0",
|
"lint-staged": "^11.0.0",
|
||||||
@ -69,20 +78,25 @@
|
|||||||
"properties-parser": "^0.3.1",
|
"properties-parser": "^0.3.1",
|
||||||
"react": "18.1.0",
|
"react": "18.1.0",
|
||||||
"rimraf": "^3.0.2",
|
"rimraf": "^3.0.2",
|
||||||
"typescript": "^4.2.3"
|
"scripting-tools": "^0.19.13",
|
||||||
|
"ts-node": "^10.9.1",
|
||||||
|
"tsc-alias": "^1.8.3",
|
||||||
|
"typescript": "^5.0.1-rc",
|
||||||
|
"vitest": "^0.29.8"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@octokit/rest": "^18.12.0",
|
"@octokit/rest": "^18.12.0",
|
||||||
"cheerio": "^1.0.0-rc.5",
|
"cheerio": "^1.0.0-rc.5",
|
||||||
"cli-select": "^1.1.2",
|
"cli-select": "^1.1.2",
|
||||||
"evt": "^2.3.1",
|
"evt": "^2.4.18",
|
||||||
"memoizee": "^0.4.15",
|
"make-fetch-happen": "^11.0.3",
|
||||||
"minimal-polyfills": "^2.2.1",
|
"minimal-polyfills": "^2.2.2",
|
||||||
|
"minimist": "^1.2.6",
|
||||||
"path-browserify": "^1.0.1",
|
"path-browserify": "^1.0.1",
|
||||||
"powerhooks": "^0.20.10",
|
|
||||||
"react-markdown": "^5.0.3",
|
"react-markdown": "^5.0.3",
|
||||||
"scripting-tools": "^0.19.13",
|
"rfc4648": "^1.5.2",
|
||||||
"tsafe": "^0.10.1",
|
"tsafe": "^1.6.0",
|
||||||
"tss-react": "^3.7.1"
|
"yauzl": "^2.10.0",
|
||||||
|
"zod": "^3.17.10"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -13,11 +13,11 @@
|
|||||||
"packageRules": [
|
"packageRules": [
|
||||||
{
|
{
|
||||||
"packagePatterns": ["*"],
|
"packagePatterns": ["*"],
|
||||||
"excludePackagePatterns": ["tss-react", "powerhooks", "tsafe", "evt"],
|
"excludePackagePatterns": ["tsafe", "evt"],
|
||||||
"enabled": false
|
"enabled": false
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"packagePatterns": ["tss-react", "powerhooks", "tsafe", "evt"],
|
"packagePatterns": ["tsafe", "evt"],
|
||||||
"matchUpdateTypes": ["minor", "patch"],
|
"matchUpdateTypes": ["minor", "patch"],
|
||||||
"automerge": true,
|
"automerge": true,
|
||||||
"automergeType": "branch",
|
"automergeType": "branch",
|
||||||
|
123
scripts/generate-i18n-messages.ts
Normal file
123
scripts/generate-i18n-messages.ts
Normal file
@ -0,0 +1,123 @@
|
|||||||
|
import "minimal-polyfills/Object.fromEntries";
|
||||||
|
import * as fs from "fs";
|
||||||
|
import { join as pathJoin, relative as pathRelative, dirname as pathDirname } from "path";
|
||||||
|
import { crawl } from "../src/bin/tools/crawl";
|
||||||
|
import { downloadBuiltinKeycloakTheme } from "../src/bin/download-builtin-keycloak-theme";
|
||||||
|
import { getProjectRoot } from "../src/bin/tools/getProjectRoot";
|
||||||
|
import { getCliOptions } from "../src/bin/tools/cliOptions";
|
||||||
|
import { getLogger } from "../src/bin/tools/logger";
|
||||||
|
|
||||||
|
// NOTE: To run without argument when we want to generate src/i18n/generated_kcMessages files,
|
||||||
|
// update the version array for generating for newer version.
|
||||||
|
|
||||||
|
//@ts-ignore
|
||||||
|
const propertiesParser = require("properties-parser");
|
||||||
|
|
||||||
|
const { isSilent } = getCliOptions(process.argv.slice(2));
|
||||||
|
const logger = getLogger({ isSilent });
|
||||||
|
|
||||||
|
async function main() {
|
||||||
|
const keycloakVersion = "21.0.1";
|
||||||
|
|
||||||
|
const tmpDirPath = pathJoin(getProjectRoot(), "tmp_xImOef9dOd44");
|
||||||
|
|
||||||
|
fs.rmSync(tmpDirPath, { "recursive": true, "force": true });
|
||||||
|
|
||||||
|
await downloadBuiltinKeycloakTheme({
|
||||||
|
keycloakVersion,
|
||||||
|
"destDirPath": tmpDirPath,
|
||||||
|
isSilent
|
||||||
|
});
|
||||||
|
|
||||||
|
type Dictionary = { [idiomId: string]: string };
|
||||||
|
|
||||||
|
const record: { [typeOfPage: string]: { [language: string]: Dictionary } } = {};
|
||||||
|
|
||||||
|
{
|
||||||
|
const baseThemeDirPath = pathJoin(tmpDirPath, "base");
|
||||||
|
|
||||||
|
crawl(baseThemeDirPath).forEach(filePath => {
|
||||||
|
const match =
|
||||||
|
filePath.match(/^([^/]+)\/messages\/messages_([^.]+)\.properties$/) ||
|
||||||
|
filePath.match(/^([^\\]+)\\messages\\messages_([^.]+)\.properties$/);
|
||||||
|
|
||||||
|
if (match === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const [, typeOfPage, language] = match;
|
||||||
|
|
||||||
|
(record[typeOfPage] ??= {})[language.replace(/_/g, "-")] = Object.fromEntries(
|
||||||
|
Object.entries(propertiesParser.parse(fs.readFileSync(pathJoin(baseThemeDirPath, filePath)).toString("utf8"))).map(
|
||||||
|
([key, value]: any) => [key, value.replace(/''/g, "'")]
|
||||||
|
)
|
||||||
|
);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
fs.rmSync(tmpDirPath, { recursive: true, force: true });
|
||||||
|
|
||||||
|
Object.keys(record).forEach(themeType => {
|
||||||
|
const recordForPageType = record[themeType];
|
||||||
|
|
||||||
|
if (themeType !== "login" && themeType !== "account") {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const baseMessagesDirPath = pathJoin(getProjectRoot(), "src", themeType, "i18n", "baseMessages");
|
||||||
|
|
||||||
|
const languages = Object.keys(recordForPageType);
|
||||||
|
|
||||||
|
const generatedFileHeader = [
|
||||||
|
`//This code was automatically generated by running ${pathRelative(getProjectRoot(), __filename)}`,
|
||||||
|
"//PLEASE DO NOT EDIT MANUALLY",
|
||||||
|
""
|
||||||
|
].join("\n");
|
||||||
|
|
||||||
|
languages.forEach(language => {
|
||||||
|
const filePath = pathJoin(baseMessagesDirPath, `${language}.ts`);
|
||||||
|
|
||||||
|
fs.mkdirSync(pathDirname(filePath), { "recursive": true });
|
||||||
|
|
||||||
|
fs.writeFileSync(
|
||||||
|
filePath,
|
||||||
|
Buffer.from(
|
||||||
|
[
|
||||||
|
generatedFileHeader,
|
||||||
|
"/* spell-checker: disable */",
|
||||||
|
`const messages= ${JSON.stringify(recordForPageType[language], null, 2)};`,
|
||||||
|
"",
|
||||||
|
"export default messages;",
|
||||||
|
"/* spell-checker: enable */"
|
||||||
|
].join("\n"),
|
||||||
|
"utf8"
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
logger.log(`${filePath} wrote`);
|
||||||
|
});
|
||||||
|
|
||||||
|
fs.writeFileSync(
|
||||||
|
pathJoin(baseMessagesDirPath, "index.ts"),
|
||||||
|
Buffer.from(
|
||||||
|
[
|
||||||
|
generatedFileHeader,
|
||||||
|
"export async function getMessages(currentLanguageTag: string) {",
|
||||||
|
" const { default: messages } = await (() => {",
|
||||||
|
" switch (currentLanguageTag) {",
|
||||||
|
...languages.map(language => ` case "${language}": return import("./${language}");`),
|
||||||
|
' default: return { "default": {} };',
|
||||||
|
" }",
|
||||||
|
" })();",
|
||||||
|
" return messages;",
|
||||||
|
"}"
|
||||||
|
].join("\n"),
|
||||||
|
"utf8"
|
||||||
|
)
|
||||||
|
);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (require.main === module) {
|
||||||
|
main().catch(e => console.error(e));
|
||||||
|
}
|
143
scripts/link-in-app.ts
Normal file
143
scripts/link-in-app.ts
Normal file
@ -0,0 +1,143 @@
|
|||||||
|
import { execSync } from "child_process";
|
||||||
|
import { join as pathJoin, relative as pathRelative } from "path";
|
||||||
|
import { getProjectRoot } from "../src/bin/tools/getProjectRoot";
|
||||||
|
import * as fs from "fs";
|
||||||
|
|
||||||
|
const singletonDependencies: string[] = ["react", "@types/react"];
|
||||||
|
|
||||||
|
const rootDirPath = getProjectRoot();
|
||||||
|
|
||||||
|
//NOTE: This is only required because of: https://github.com/garronej/ts-ci/blob/c0e207b9677523d4ec97fe672ddd72ccbb3c1cc4/README.md?plain=1#L54-L58
|
||||||
|
fs.writeFileSync(
|
||||||
|
pathJoin(rootDirPath, "dist", "package.json"),
|
||||||
|
Buffer.from(
|
||||||
|
JSON.stringify(
|
||||||
|
(() => {
|
||||||
|
const packageJsonParsed = JSON.parse(fs.readFileSync(pathJoin(rootDirPath, "package.json")).toString("utf8"));
|
||||||
|
|
||||||
|
return {
|
||||||
|
...packageJsonParsed,
|
||||||
|
"main": packageJsonParsed["main"]?.replace(/^dist\//, ""),
|
||||||
|
"types": packageJsonParsed["types"]?.replace(/^dist\//, ""),
|
||||||
|
"module": packageJsonParsed["module"]?.replace(/^dist\//, ""),
|
||||||
|
"exports": !("exports" in packageJsonParsed)
|
||||||
|
? undefined
|
||||||
|
: Object.fromEntries(
|
||||||
|
Object.entries(packageJsonParsed["exports"]).map(([key, value]) => [
|
||||||
|
key,
|
||||||
|
(value as string).replace(/^\.\/dist\//, "./")
|
||||||
|
])
|
||||||
|
)
|
||||||
|
};
|
||||||
|
})(),
|
||||||
|
null,
|
||||||
|
2
|
||||||
|
),
|
||||||
|
"utf8"
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
fs.cpSync(pathJoin(rootDirPath, "src"), pathJoin(rootDirPath, "dist", "src"), { "recursive": true });
|
||||||
|
|
||||||
|
const commonThirdPartyDeps = (() => {
|
||||||
|
// For example [ "@emotion" ] it's more convenient than
|
||||||
|
// having to list every sub emotion packages (@emotion/css @emotion/utils ...)
|
||||||
|
// in singletonDependencies
|
||||||
|
const namespaceSingletonDependencies: string[] = [];
|
||||||
|
|
||||||
|
return [
|
||||||
|
...namespaceSingletonDependencies
|
||||||
|
.map(namespaceModuleName =>
|
||||||
|
fs
|
||||||
|
.readdirSync(pathJoin(rootDirPath, "node_modules", namespaceModuleName))
|
||||||
|
.map(submoduleName => `${namespaceModuleName}/${submoduleName}`)
|
||||||
|
)
|
||||||
|
.reduce((prev, curr) => [...prev, ...curr], []),
|
||||||
|
...singletonDependencies
|
||||||
|
];
|
||||||
|
})();
|
||||||
|
|
||||||
|
const yarnGlobalDirPath = pathJoin(rootDirPath, ".yarn_home");
|
||||||
|
|
||||||
|
fs.rmSync(yarnGlobalDirPath, { "recursive": true, "force": true });
|
||||||
|
fs.mkdirSync(yarnGlobalDirPath);
|
||||||
|
|
||||||
|
const execYarnLink = (params: { targetModuleName?: string; cwd: string }) => {
|
||||||
|
const { targetModuleName, cwd } = params;
|
||||||
|
|
||||||
|
const cmd = ["yarn", "link", ...(targetModuleName !== undefined ? [targetModuleName] : ["--no-bin-links"])].join(" ");
|
||||||
|
|
||||||
|
console.log(`$ cd ${pathRelative(rootDirPath, cwd) || "."} && ${cmd}`);
|
||||||
|
|
||||||
|
execSync(cmd, {
|
||||||
|
cwd,
|
||||||
|
"env": {
|
||||||
|
...process.env,
|
||||||
|
"HOME": yarnGlobalDirPath
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const testAppPaths = (() => {
|
||||||
|
const [, , ...testAppNames] = process.argv;
|
||||||
|
|
||||||
|
return testAppNames
|
||||||
|
.map(testAppName => {
|
||||||
|
const testAppPath = pathJoin(rootDirPath, "..", testAppName);
|
||||||
|
|
||||||
|
if (fs.existsSync(testAppPath)) {
|
||||||
|
return testAppPath;
|
||||||
|
}
|
||||||
|
|
||||||
|
console.warn(`Skipping ${testAppName} since it cant be found here: ${testAppPath}`);
|
||||||
|
|
||||||
|
return undefined;
|
||||||
|
})
|
||||||
|
.filter((path): path is string => path !== undefined);
|
||||||
|
})();
|
||||||
|
|
||||||
|
if (testAppPaths.length === 0) {
|
||||||
|
console.error("No test app to link into!");
|
||||||
|
process.exit(-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
testAppPaths.forEach(testAppPath => execSync("yarn install", { "cwd": testAppPath }));
|
||||||
|
|
||||||
|
console.log("=== Linking common dependencies ===");
|
||||||
|
|
||||||
|
const total = commonThirdPartyDeps.length;
|
||||||
|
let current = 0;
|
||||||
|
|
||||||
|
commonThirdPartyDeps.forEach(commonThirdPartyDep => {
|
||||||
|
current++;
|
||||||
|
|
||||||
|
console.log(`${current}/${total} ${commonThirdPartyDep}`);
|
||||||
|
|
||||||
|
const localInstallPath = pathJoin(
|
||||||
|
...[rootDirPath, "node_modules", ...(commonThirdPartyDep.startsWith("@") ? commonThirdPartyDep.split("/") : [commonThirdPartyDep])]
|
||||||
|
);
|
||||||
|
|
||||||
|
execYarnLink({ "cwd": localInstallPath });
|
||||||
|
});
|
||||||
|
|
||||||
|
commonThirdPartyDeps.forEach(commonThirdPartyDep =>
|
||||||
|
testAppPaths.forEach(testAppPath =>
|
||||||
|
execYarnLink({
|
||||||
|
"cwd": testAppPath,
|
||||||
|
"targetModuleName": commonThirdPartyDep
|
||||||
|
})
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
console.log("=== Linking in house dependencies ===");
|
||||||
|
|
||||||
|
execYarnLink({ "cwd": pathJoin(rootDirPath, "dist") });
|
||||||
|
|
||||||
|
testAppPaths.forEach(testAppPath =>
|
||||||
|
execYarnLink({
|
||||||
|
"cwd": testAppPath,
|
||||||
|
"targetModuleName": JSON.parse(fs.readFileSync(pathJoin(rootDirPath, "package.json")).toString("utf8"))["name"]
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
export {};
|
29
scripts/test-keycloakify-starter.ts
Normal file
29
scripts/test-keycloakify-starter.ts
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
import { execSync } from "child_process";
|
||||||
|
import { existsSync, readFileSync, rmSync, writeFileSync } from "fs";
|
||||||
|
import path from "path";
|
||||||
|
|
||||||
|
const testDir = "keycloakify_starter_test";
|
||||||
|
|
||||||
|
if (existsSync(path.join(process.cwd(), testDir))) {
|
||||||
|
rmSync(path.join(process.cwd(), testDir), { recursive: true });
|
||||||
|
}
|
||||||
|
// Build and link package
|
||||||
|
execSync("yarn build");
|
||||||
|
const pkgJSON = JSON.parse(readFileSync(path.join(process.cwd(), "package.json")).toString("utf8"));
|
||||||
|
pkgJSON.main = "./index.js";
|
||||||
|
pkgJSON.types = "./index.d.ts";
|
||||||
|
pkgJSON.scripts.prepare = undefined;
|
||||||
|
writeFileSync(path.join(process.cwd(), "dist", "package.json"), JSON.stringify(pkgJSON));
|
||||||
|
// Wrapped in a try/catch because unlink errors if the package isn't linked
|
||||||
|
try {
|
||||||
|
execSync("yarn unlink");
|
||||||
|
} catch {}
|
||||||
|
execSync("yarn link", { "cwd": path.join(process.cwd(), "dist") });
|
||||||
|
|
||||||
|
// Clone latest keycloakify-starter and link to keycloakify output
|
||||||
|
execSync(`git clone https://github.com/keycloakify/keycloakify-starter.git ${testDir}`);
|
||||||
|
execSync("yarn install", { "cwd": path.join(process.cwd(), testDir) });
|
||||||
|
execSync("yarn link keycloakify", { "cwd": path.join(process.cwd(), testDir) });
|
||||||
|
|
||||||
|
//Ensure keycloak theme can be built
|
||||||
|
execSync("yarn build-keycloak-theme", { "cwd": path.join(process.cwd(), testDir) });
|
26
src/account/Fallback.tsx
Normal file
26
src/account/Fallback.tsx
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
import { lazy, Suspense } from "react";
|
||||||
|
import type { PageProps } from "keycloakify/account/pages/PageProps";
|
||||||
|
import type { I18n } from "keycloakify/account/i18n";
|
||||||
|
import type { KcContext } from "./kcContext";
|
||||||
|
import { assert, type Equals } from "tsafe/assert";
|
||||||
|
|
||||||
|
const Password = lazy(() => import("keycloakify/account/pages/Password"));
|
||||||
|
const Account = lazy(() => import("keycloakify/account/pages/Account"));
|
||||||
|
|
||||||
|
export default function Fallback(props: PageProps<KcContext, I18n>) {
|
||||||
|
const { kcContext, ...rest } = props;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Suspense>
|
||||||
|
{(() => {
|
||||||
|
switch (kcContext.pageId) {
|
||||||
|
case "password.ftl":
|
||||||
|
return <Password kcContext={kcContext} {...rest} />;
|
||||||
|
case "account.ftl":
|
||||||
|
return <Account kcContext={kcContext} {...rest} />;
|
||||||
|
}
|
||||||
|
assert<Equals<typeof kcContext, never>>(false);
|
||||||
|
})()}
|
||||||
|
</Suspense>
|
||||||
|
);
|
||||||
|
}
|
131
src/account/Template.tsx
Normal file
131
src/account/Template.tsx
Normal file
@ -0,0 +1,131 @@
|
|||||||
|
import { clsx } from "keycloakify/tools/clsx";
|
||||||
|
import { usePrepareTemplate } from "keycloakify/lib/usePrepareTemplate";
|
||||||
|
import { type TemplateProps } from "keycloakify/account/TemplateProps";
|
||||||
|
import { useGetClassName } from "keycloakify/account/lib/useGetClassName";
|
||||||
|
import type { KcContext } from "./kcContext";
|
||||||
|
import type { I18n } from "./i18n";
|
||||||
|
import { assert } from "keycloakify/tools/assert";
|
||||||
|
|
||||||
|
export default function Template(props: TemplateProps<KcContext, I18n>) {
|
||||||
|
const { kcContext, i18n, doUseDefaultCss, active, classes, children } = props;
|
||||||
|
|
||||||
|
const { getClassName } = useGetClassName({ doUseDefaultCss, classes });
|
||||||
|
|
||||||
|
const { msg, changeLocale, labelBySupportedLanguageTag, currentLanguageTag } = i18n;
|
||||||
|
|
||||||
|
const { locale, url, features, realm, message, referrer } = kcContext;
|
||||||
|
|
||||||
|
const { isReady } = usePrepareTemplate({
|
||||||
|
"doFetchDefaultThemeResources": doUseDefaultCss,
|
||||||
|
url,
|
||||||
|
"stylesCommon": ["node_modules/patternfly/dist/css/patternfly.min.css", "node_modules/patternfly/dist/css/patternfly-additions.min.css"],
|
||||||
|
"styles": ["css/account.css"],
|
||||||
|
"htmlClassName": undefined,
|
||||||
|
"bodyClassName": clsx("admin-console", "user", getClassName("kcBodyClass"))
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!isReady) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<header className="navbar navbar-default navbar-pf navbar-main header">
|
||||||
|
<nav className="navbar" role="navigation">
|
||||||
|
<div className="navbar-header">
|
||||||
|
<div className="container">
|
||||||
|
<h1 className="navbar-title">Keycloak</h1>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="navbar-collapse navbar-collapse-1">
|
||||||
|
<div className="container">
|
||||||
|
<ul className="nav navbar-nav navbar-utility">
|
||||||
|
{realm.internationalizationEnabled && (assert(locale !== undefined), true) && locale.supported.length > 1 && (
|
||||||
|
<li>
|
||||||
|
<div className="kc-dropdown" id="kc-locale-dropdown">
|
||||||
|
{/* eslint-disable-next-line jsx-a11y/anchor-is-valid */}
|
||||||
|
<a href="#" id="kc-current-locale-link">
|
||||||
|
{labelBySupportedLanguageTag[currentLanguageTag]}
|
||||||
|
</a>
|
||||||
|
<ul>
|
||||||
|
{locale.supported.map(({ languageTag }) => (
|
||||||
|
<li key={languageTag} className="kc-dropdown-item">
|
||||||
|
{/* eslint-disable-next-line jsx-a11y/anchor-is-valid */}
|
||||||
|
<a href="#" onClick={() => changeLocale(languageTag)}>
|
||||||
|
{labelBySupportedLanguageTag[languageTag]}
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</li>
|
||||||
|
)}
|
||||||
|
{referrer?.url !== undefined && (
|
||||||
|
<li>
|
||||||
|
<a href={referrer.url} id="referrer">
|
||||||
|
{msg("backTo", referrer.name)}
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
)}
|
||||||
|
<li>
|
||||||
|
<a href={url.getLogoutUrl()}>{msg("doSignOut")}</a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</nav>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<div className="container">
|
||||||
|
<div className="bs-sidebar col-sm-3">
|
||||||
|
<ul>
|
||||||
|
<li className={clsx(active === "account" && "active")}>
|
||||||
|
<a href={url.accountUrl}>{msg("account")}</a>
|
||||||
|
</li>
|
||||||
|
{features.passwordUpdateSupported && (
|
||||||
|
<li className={clsx(active === "password" && "active")}>
|
||||||
|
<a href={url.passwordUrl}>{msg("password")}</a>
|
||||||
|
</li>
|
||||||
|
)}
|
||||||
|
<li className={clsx(active === "totp" && "active")}>
|
||||||
|
<a href={url.totpUrl}>{msg("authenticator")}</a>
|
||||||
|
</li>
|
||||||
|
{features.identityFederation && (
|
||||||
|
<li className={clsx(active === "social" && "active")}>
|
||||||
|
<a href={url.socialUrl}>{msg("federatedIdentity")}</a>
|
||||||
|
</li>
|
||||||
|
)}
|
||||||
|
<li className={clsx(active === "sessions" && "active")}>
|
||||||
|
<a href={url.sessionsUrl}>{msg("sessions")}</a>
|
||||||
|
</li>
|
||||||
|
<li className={clsx(active === "applications" && "active")}>
|
||||||
|
<a href={url.applicationsUrl}>{msg("applications")}</a>
|
||||||
|
</li>
|
||||||
|
{features.log && (
|
||||||
|
<li className={clsx(active === "log" && "active")}>
|
||||||
|
<a href={url.logUrl}>{msg("log")}</a>
|
||||||
|
</li>
|
||||||
|
)}
|
||||||
|
{realm.userManagedAccessAllowed && features.authorization && (
|
||||||
|
<li className={clsx(active === "authorization" && "active")}>
|
||||||
|
<a href={url.resourceUrl}>{msg("myResources")}</a>
|
||||||
|
</li>
|
||||||
|
)}
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="col-sm-9 content-area">
|
||||||
|
{message !== undefined && (
|
||||||
|
<div className={clsx("alert", `alert-${message.type}`)}>
|
||||||
|
{message.type === "success" && <span className="pficon pficon-ok"></span>}
|
||||||
|
{message.type === "error" && <span className="pficon pficon-error-circle-o"></span>}
|
||||||
|
<span className="kc-feedback-text">{message.summary}</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
14
src/account/TemplateProps.ts
Normal file
14
src/account/TemplateProps.ts
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
import type { ReactNode } from "react";
|
||||||
|
import type { KcContext } from "./kcContext";
|
||||||
|
import type { I18n } from "./i18n";
|
||||||
|
|
||||||
|
export type TemplateProps<KcContext extends KcContext.Common, I18nExtended extends I18n> = {
|
||||||
|
kcContext: KcContext;
|
||||||
|
i18n: I18nExtended;
|
||||||
|
doUseDefaultCss: boolean;
|
||||||
|
active: string;
|
||||||
|
classes?: Partial<Record<ClassKey, string>>;
|
||||||
|
children: ReactNode;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type ClassKey = "kcBodyClass" | "kcButtonClass" | "kcButtonPrimaryClass" | "kcButtonLargeClass" | "kcButtonDefaultClass";
|
229
src/account/i18n/i18n.tsx
Normal file
229
src/account/i18n/i18n.tsx
Normal file
@ -0,0 +1,229 @@
|
|||||||
|
import "minimal-polyfills/Object.fromEntries";
|
||||||
|
//NOTE for later: https://github.com/remarkjs/react-markdown/blob/236182ecf30bd89c1e5a7652acaf8d0bf81e6170/src/renderers.js#L7-L35
|
||||||
|
import { useEffect, useState, useRef } from "react";
|
||||||
|
import fallbackMessages from "./baseMessages/en";
|
||||||
|
import { getMessages } from "./baseMessages";
|
||||||
|
import { assert } from "tsafe/assert";
|
||||||
|
import type { KcContext } from "../kcContext/KcContext";
|
||||||
|
import { Markdown } from "keycloakify/tools/Markdown";
|
||||||
|
|
||||||
|
export const fallbackLanguageTag = "en";
|
||||||
|
|
||||||
|
export type KcContextLike = {
|
||||||
|
locale?: {
|
||||||
|
currentLanguageTag: string;
|
||||||
|
supported: { languageTag: string; url: string; label: string }[];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
assert<KcContext extends KcContextLike ? true : false>();
|
||||||
|
|
||||||
|
export type MessageKey = keyof typeof fallbackMessages | keyof (typeof keycloakifyExtraMessages)[typeof fallbackLanguageTag];
|
||||||
|
|
||||||
|
export type GenericI18n<MessageKey extends string> = {
|
||||||
|
/**
|
||||||
|
* e.g: "en", "fr", "zh-CN"
|
||||||
|
*
|
||||||
|
* The current language
|
||||||
|
*/
|
||||||
|
currentLanguageTag: string;
|
||||||
|
/**
|
||||||
|
* To call when the user switch language.
|
||||||
|
* This will cause the page to be reloaded,
|
||||||
|
* on next load currentLanguageTag === newLanguageTag
|
||||||
|
*/
|
||||||
|
changeLocale: (newLanguageTag: string) => never;
|
||||||
|
/**
|
||||||
|
* e.g. "en" => "English", "fr" => "Français", ...
|
||||||
|
*
|
||||||
|
* Used to render a select that enable user to switch language.
|
||||||
|
* ex: https://user-images.githubusercontent.com/6702424/186044799-38801eec-4e89-483b-81dd-8e9233e8c0eb.png
|
||||||
|
* */
|
||||||
|
labelBySupportedLanguageTag: Record<string, string>;
|
||||||
|
/**
|
||||||
|
* Examples assuming currentLanguageTag === "en"
|
||||||
|
*
|
||||||
|
* msg("access-denied") === <span>Access denied</span>
|
||||||
|
* msg("impersonateTitleHtml", "Foo") === <span><strong>Foo</strong> Impersonate User</span>
|
||||||
|
*/
|
||||||
|
msg: (key: MessageKey, ...args: (string | undefined)[]) => JSX.Element;
|
||||||
|
/**
|
||||||
|
* It's the same thing as msg() but instead of returning a JSX.Element it returns a string.
|
||||||
|
* It can be more convenient to manipulate strings but if there are HTML tags it wont render.
|
||||||
|
* msgStr("impersonateTitleHtml", "Foo") === "<strong>Foo</strong> Impersonate User"
|
||||||
|
*/
|
||||||
|
msgStr: (key: MessageKey, ...args: (string | undefined)[]) => string;
|
||||||
|
/**
|
||||||
|
* Examples assuming currentLanguageTag === "en"
|
||||||
|
* advancedMsg("${access-denied} foo bar") === <span>${msgStr("access-denied")} foo bar<span> === <span>Access denied foo bar</span>
|
||||||
|
* advancedMsg("${access-denied}") === advancedMsg("access-denied") === msg("access-denied") === <span>Access denied</span>
|
||||||
|
* advancedMsg("${not-a-message-key}") === advancedMsg(not-a-message-key") === <span>not-a-message-key</span>
|
||||||
|
*/
|
||||||
|
advancedMsg: (key: string, ...args: (string | undefined)[]) => JSX.Element;
|
||||||
|
/**
|
||||||
|
* Examples assuming currentLanguageTag === "en"
|
||||||
|
* advancedMsg("${access-denied} foo bar") === msg("access-denied") + " foo bar" === "Access denied foo bar"
|
||||||
|
* advancedMsg("${not-a-message-key}") === advancedMsg(not-a-message-key") === "not-a-message-key"
|
||||||
|
*/
|
||||||
|
advancedMsgStr: (key: string, ...args: (string | undefined)[]) => string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type I18n = GenericI18n<MessageKey>;
|
||||||
|
|
||||||
|
export function createUseI18n<ExtraMessageKey extends string = never>(extraMessages: {
|
||||||
|
[languageTag: string]: { [key in ExtraMessageKey]: string };
|
||||||
|
}) {
|
||||||
|
function useI18n(params: { kcContext: KcContextLike }): GenericI18n<MessageKey | ExtraMessageKey> | null {
|
||||||
|
const { kcContext } = params;
|
||||||
|
|
||||||
|
const [i18n, setI18n] = useState<GenericI18n<ExtraMessageKey | MessageKey> | undefined>(undefined);
|
||||||
|
|
||||||
|
const refHasStartedFetching = useRef(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (refHasStartedFetching.current) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
refHasStartedFetching.current = true;
|
||||||
|
|
||||||
|
(async () => {
|
||||||
|
const { currentLanguageTag = fallbackLanguageTag } = kcContext.locale ?? {};
|
||||||
|
|
||||||
|
setI18n({
|
||||||
|
...createI18nTranslationFunctions({
|
||||||
|
"fallbackMessages": {
|
||||||
|
...fallbackMessages,
|
||||||
|
...(keycloakifyExtraMessages[fallbackLanguageTag] ?? {}),
|
||||||
|
...(extraMessages[fallbackLanguageTag] ?? {})
|
||||||
|
} as any,
|
||||||
|
"messages": {
|
||||||
|
...(await getMessages(currentLanguageTag)),
|
||||||
|
...((keycloakifyExtraMessages as any)[currentLanguageTag] ?? {}),
|
||||||
|
...(extraMessages[currentLanguageTag] ?? {})
|
||||||
|
} as any
|
||||||
|
}),
|
||||||
|
currentLanguageTag,
|
||||||
|
"changeLocale": newLanguageTag => {
|
||||||
|
const { locale } = kcContext;
|
||||||
|
|
||||||
|
assert(locale !== undefined, "Internationalization not enabled");
|
||||||
|
|
||||||
|
const targetSupportedLocale = locale.supported.find(({ languageTag }) => languageTag === newLanguageTag);
|
||||||
|
|
||||||
|
assert(targetSupportedLocale !== undefined, `${newLanguageTag} need to be enabled in Keycloak admin`);
|
||||||
|
|
||||||
|
window.location.href = targetSupportedLocale.url;
|
||||||
|
|
||||||
|
assert(false, "never");
|
||||||
|
},
|
||||||
|
"labelBySupportedLanguageTag": Object.fromEntries(
|
||||||
|
(kcContext.locale?.supported ?? []).map(({ languageTag, label }) => [languageTag, label])
|
||||||
|
)
|
||||||
|
});
|
||||||
|
})();
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return i18n ?? null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return { useI18n };
|
||||||
|
}
|
||||||
|
|
||||||
|
function createI18nTranslationFunctions<MessageKey extends string>(params: {
|
||||||
|
fallbackMessages: Record<MessageKey, string>;
|
||||||
|
messages: Record<MessageKey, string>;
|
||||||
|
}): Pick<GenericI18n<MessageKey>, "msg" | "msgStr" | "advancedMsg" | "advancedMsgStr"> {
|
||||||
|
const { fallbackMessages, messages } = params;
|
||||||
|
|
||||||
|
function resolveMsg(props: { key: string; args: (string | undefined)[]; doRenderMarkdown: boolean }): string | JSX.Element | undefined {
|
||||||
|
const { key, args, doRenderMarkdown } = props;
|
||||||
|
|
||||||
|
const messageOrUndefined: string | undefined = (messages as any)[key] ?? (fallbackMessages as any)[key];
|
||||||
|
|
||||||
|
if (messageOrUndefined === undefined) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
const message = messageOrUndefined;
|
||||||
|
|
||||||
|
const messageWithArgsInjectedIfAny = (() => {
|
||||||
|
const startIndex = message
|
||||||
|
.match(/{[0-9]+}/g)
|
||||||
|
?.map(g => g.match(/{([0-9]+)}/)![1])
|
||||||
|
.map(indexStr => parseInt(indexStr))
|
||||||
|
.sort((a, b) => a - b)[0];
|
||||||
|
|
||||||
|
if (startIndex === undefined) {
|
||||||
|
// No {0} in message (no arguments expected)
|
||||||
|
return message;
|
||||||
|
}
|
||||||
|
|
||||||
|
let messageWithArgsInjected = message;
|
||||||
|
|
||||||
|
args.forEach((arg, i) => {
|
||||||
|
if (arg === undefined) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
messageWithArgsInjected = messageWithArgsInjected.replace(new RegExp(`\\{${i + startIndex}\\}`, "g"), arg);
|
||||||
|
});
|
||||||
|
|
||||||
|
return messageWithArgsInjected;
|
||||||
|
})();
|
||||||
|
|
||||||
|
return doRenderMarkdown ? (
|
||||||
|
<Markdown allowDangerousHtml renderers={{ "paragraph": "span" }}>
|
||||||
|
{messageWithArgsInjectedIfAny}
|
||||||
|
</Markdown>
|
||||||
|
) : (
|
||||||
|
messageWithArgsInjectedIfAny
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function resolveMsgAdvanced(props: { key: string; args: (string | undefined)[]; doRenderMarkdown: boolean }): JSX.Element | string {
|
||||||
|
const { key, args, doRenderMarkdown } = props;
|
||||||
|
|
||||||
|
const match = key.match(/^\$\{([^{]+)\}$/);
|
||||||
|
|
||||||
|
const keyUnwrappedFromCurlyBraces = match === null ? key : match[1];
|
||||||
|
|
||||||
|
const out = resolveMsg({
|
||||||
|
"key": keyUnwrappedFromCurlyBraces,
|
||||||
|
args,
|
||||||
|
doRenderMarkdown
|
||||||
|
});
|
||||||
|
|
||||||
|
return (out !== undefined ? out : doRenderMarkdown ? <span>{keyUnwrappedFromCurlyBraces}</span> : keyUnwrappedFromCurlyBraces) as any;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
"msgStr": (key, ...args) => resolveMsg({ key, args, "doRenderMarkdown": false }) as string,
|
||||||
|
"msg": (key, ...args) => resolveMsg({ key, args, "doRenderMarkdown": true }) as JSX.Element,
|
||||||
|
"advancedMsg": (key, ...args) => resolveMsgAdvanced({ key, args, "doRenderMarkdown": true }) as JSX.Element,
|
||||||
|
"advancedMsgStr": (key, ...args) => resolveMsgAdvanced({ key, args, "doRenderMarkdown": false }) as string
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const keycloakifyExtraMessages = {
|
||||||
|
"en": {
|
||||||
|
"shouldBeEqual": "{0} should be equal to {1}",
|
||||||
|
"shouldBeDifferent": "{0} should be different to {1}",
|
||||||
|
"shouldMatchPattern": "Pattern should match: `/{0}/`",
|
||||||
|
"mustBeAnInteger": "Must be an integer",
|
||||||
|
"notAValidOption": "Not a valid option"
|
||||||
|
},
|
||||||
|
"fr": {
|
||||||
|
/* spell-checker: disable */
|
||||||
|
"shouldBeEqual": "{0} doit être égal à {1}",
|
||||||
|
"shouldBeDifferent": "{0} doit être différent de {1}",
|
||||||
|
"shouldMatchPattern": "Dois respecter le schéma: `/{0}/`",
|
||||||
|
"mustBeAnInteger": "Doit être un nombre entier",
|
||||||
|
"notAValidOption": "N'est pas une option valide",
|
||||||
|
|
||||||
|
"logoutConfirmTitle": "Déconnexion",
|
||||||
|
"logoutConfirmHeader": "Êtes-vous sûr(e) de vouloir vous déconnecter ?",
|
||||||
|
"doLogout": "Se déconnecter"
|
||||||
|
/* spell-checker: enable */
|
||||||
|
}
|
||||||
|
};
|
1
src/account/i18n/index.ts
Normal file
1
src/account/i18n/index.ts
Normal file
@ -0,0 +1 @@
|
|||||||
|
export type { I18n } from "./i18n";
|
8
src/account/index.ts
Normal file
8
src/account/index.ts
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
import Fallback from "keycloakify/account/Fallback";
|
||||||
|
|
||||||
|
export default Fallback;
|
||||||
|
|
||||||
|
export { getKcContext } from "keycloakify/account/kcContext/getKcContext";
|
||||||
|
export { createUseI18n } from "keycloakify/account/i18n/i18n";
|
||||||
|
|
||||||
|
export type { PageProps } from "keycloakify/account/pages/PageProps";
|
84
src/account/kcContext/KcContext.ts
Normal file
84
src/account/kcContext/KcContext.ts
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
import type { AccountThemePageId } from "keycloakify/bin/keycloakify/generateFtl";
|
||||||
|
import { assert } from "tsafe/assert";
|
||||||
|
import type { Equals } from "tsafe";
|
||||||
|
|
||||||
|
export type KcContext = KcContext.Password | KcContext.Account;
|
||||||
|
|
||||||
|
export declare namespace KcContext {
|
||||||
|
export type Common = {
|
||||||
|
locale?: {
|
||||||
|
supported: {
|
||||||
|
url: string;
|
||||||
|
label: string;
|
||||||
|
languageTag: string;
|
||||||
|
}[];
|
||||||
|
currentLanguageTag: string;
|
||||||
|
};
|
||||||
|
url: {
|
||||||
|
accountUrl: string;
|
||||||
|
passwordUrl: string;
|
||||||
|
totpUrl: string;
|
||||||
|
socialUrl: string;
|
||||||
|
sessionsUrl: string;
|
||||||
|
applicationsUrl: string;
|
||||||
|
logUrl: string;
|
||||||
|
resourceUrl: string;
|
||||||
|
resourcesCommonPath: string;
|
||||||
|
resourcesPath: string;
|
||||||
|
getLogoutUrl: () => string;
|
||||||
|
};
|
||||||
|
features: {
|
||||||
|
passwordUpdateSupported: boolean;
|
||||||
|
identityFederation: boolean;
|
||||||
|
log: boolean;
|
||||||
|
authorization: boolean;
|
||||||
|
};
|
||||||
|
realm: {
|
||||||
|
internationalizationEnabled: boolean;
|
||||||
|
userManagedAccessAllowed: boolean;
|
||||||
|
};
|
||||||
|
message?: {
|
||||||
|
type: "success" | "warning" | "error" | "info";
|
||||||
|
summary: string;
|
||||||
|
};
|
||||||
|
referrer?: {
|
||||||
|
url?: string;
|
||||||
|
name: string;
|
||||||
|
};
|
||||||
|
messagesPerField: {
|
||||||
|
printIfExists: <T>(fieldName: string, x: T) => T | undefined;
|
||||||
|
existsError: (fieldName: string) => boolean;
|
||||||
|
get: (fieldName: string) => string;
|
||||||
|
exists: (fieldName: string) => boolean;
|
||||||
|
};
|
||||||
|
account: {
|
||||||
|
email?: string;
|
||||||
|
firstName: string;
|
||||||
|
lastName?: string;
|
||||||
|
username?: string;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Password = Common & {
|
||||||
|
pageId: "password.ftl";
|
||||||
|
password: {
|
||||||
|
passwordSet: boolean;
|
||||||
|
};
|
||||||
|
stateChecker: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Account = Common & {
|
||||||
|
pageId: "account.ftl";
|
||||||
|
url: {
|
||||||
|
referrerURI: string;
|
||||||
|
accountUrl: string;
|
||||||
|
};
|
||||||
|
realm: {
|
||||||
|
registrationEmailAsUsername: boolean;
|
||||||
|
editUsernameAllowed: boolean;
|
||||||
|
};
|
||||||
|
stateChecker: string;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
assert<Equals<KcContext["pageId"], AccountThemePageId>>();
|
76
src/account/kcContext/getKcContext.ts
Normal file
76
src/account/kcContext/getKcContext.ts
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
import type { DeepPartial } from "keycloakify/tools/DeepPartial";
|
||||||
|
import { deepAssign } from "keycloakify/tools/deepAssign";
|
||||||
|
import type { ExtendKcContext } from "./getKcContextFromWindow";
|
||||||
|
import { getKcContextFromWindow } from "./getKcContextFromWindow";
|
||||||
|
import { pathJoin } from "keycloakify/bin/tools/pathJoin";
|
||||||
|
import { pathBasename } from "keycloakify/tools/pathBasename";
|
||||||
|
import { mockTestingResourcesCommonPath } from "keycloakify/bin/mockTestingResourcesPath";
|
||||||
|
import { symToStr } from "tsafe/symToStr";
|
||||||
|
import { kcContextMocks, kcContextCommonMock } from "keycloakify/account/kcContext/kcContextMocks";
|
||||||
|
|
||||||
|
export function getKcContext<KcContextExtension extends { pageId: string } = never>(params?: {
|
||||||
|
mockPageId?: ExtendKcContext<KcContextExtension>["pageId"];
|
||||||
|
mockData?: readonly DeepPartial<ExtendKcContext<KcContextExtension>>[];
|
||||||
|
}): { kcContext: ExtendKcContext<KcContextExtension> | undefined } {
|
||||||
|
const { mockPageId, mockData } = params ?? {};
|
||||||
|
|
||||||
|
const realKcContext = getKcContextFromWindow<KcContextExtension>();
|
||||||
|
|
||||||
|
if (mockPageId !== undefined && realKcContext === undefined) {
|
||||||
|
//TODO maybe trow if no mock fo custom page
|
||||||
|
|
||||||
|
console.log(
|
||||||
|
[
|
||||||
|
`%cKeycloakify: ${symToStr({ mockPageId })} set to ${mockPageId}.`,
|
||||||
|
`If assets are missing make sure you have built your Keycloak theme at least once.`
|
||||||
|
].join(" "),
|
||||||
|
"background: red; color: yellow; font-size: medium"
|
||||||
|
);
|
||||||
|
|
||||||
|
const kcContextDefaultMock = kcContextMocks.find(({ pageId }) => pageId === mockPageId);
|
||||||
|
|
||||||
|
const partialKcContextCustomMock = mockData?.find(({ pageId }) => pageId === mockPageId);
|
||||||
|
|
||||||
|
if (kcContextDefaultMock === undefined && partialKcContextCustomMock === undefined) {
|
||||||
|
console.warn(
|
||||||
|
[
|
||||||
|
`WARNING: You declared the non build in page ${mockPageId} but you didn't `,
|
||||||
|
`provide mock data needed to debug the page outside of Keycloak as you are trying to do now.`,
|
||||||
|
`Please check the documentation of the getKcContext function`
|
||||||
|
].join("\n")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const kcContext: any = {};
|
||||||
|
|
||||||
|
deepAssign({
|
||||||
|
"target": kcContext,
|
||||||
|
"source": kcContextDefaultMock !== undefined ? kcContextDefaultMock : { "pageId": mockPageId, ...kcContextCommonMock }
|
||||||
|
});
|
||||||
|
|
||||||
|
if (partialKcContextCustomMock !== undefined) {
|
||||||
|
deepAssign({
|
||||||
|
"target": kcContext,
|
||||||
|
"source": partialKcContextCustomMock
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return { kcContext };
|
||||||
|
}
|
||||||
|
|
||||||
|
if (realKcContext === undefined) {
|
||||||
|
return { "kcContext": undefined };
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!("account" in realKcContext)) {
|
||||||
|
return { "kcContext": undefined };
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
const { url } = realKcContext;
|
||||||
|
|
||||||
|
url.resourcesCommonPath = pathJoin(url.resourcesPath, pathBasename(mockTestingResourcesCommonPath));
|
||||||
|
}
|
||||||
|
|
||||||
|
return { "kcContext": realKcContext };
|
||||||
|
}
|
11
src/account/kcContext/getKcContextFromWindow.ts
Normal file
11
src/account/kcContext/getKcContextFromWindow.ts
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
import type { AndByDiscriminatingKey } from "keycloakify/tools/AndByDiscriminatingKey";
|
||||||
|
import { ftlValuesGlobalName } from "keycloakify/bin/keycloakify/ftlValuesGlobalName";
|
||||||
|
import type { KcContext } from "./KcContext";
|
||||||
|
|
||||||
|
export type ExtendKcContext<KcContextExtension extends { pageId: string }> = [KcContextExtension] extends [never]
|
||||||
|
? KcContext
|
||||||
|
: AndByDiscriminatingKey<"pageId", KcContextExtension & KcContext.Common, KcContext>;
|
||||||
|
|
||||||
|
export function getKcContextFromWindow<KcContextExtension extends { pageId: string } = never>(): ExtendKcContext<KcContextExtension> | undefined {
|
||||||
|
return typeof window === "undefined" ? undefined : (window as any)[ftlValuesGlobalName];
|
||||||
|
}
|
1
src/account/kcContext/index.ts
Normal file
1
src/account/kcContext/index.ts
Normal file
@ -0,0 +1 @@
|
|||||||
|
export type { KcContext } from "./KcContext";
|
175
src/account/kcContext/kcContextMocks.ts
Normal file
175
src/account/kcContext/kcContextMocks.ts
Normal file
@ -0,0 +1,175 @@
|
|||||||
|
import "minimal-polyfills/Object.fromEntries";
|
||||||
|
import { mockTestingResourcesCommonPath, mockTestingResourcesPath } from "keycloakify/bin/mockTestingResourcesPath";
|
||||||
|
import { pathJoin } from "keycloakify/bin/tools/pathJoin";
|
||||||
|
import { id } from "tsafe/id";
|
||||||
|
import type { KcContext } from "./KcContext";
|
||||||
|
|
||||||
|
const PUBLIC_URL = process.env["PUBLIC_URL"] ?? "/";
|
||||||
|
|
||||||
|
export const kcContextCommonMock: KcContext.Common = {
|
||||||
|
"url": {
|
||||||
|
"resourcesPath": pathJoin(PUBLIC_URL, mockTestingResourcesPath),
|
||||||
|
"resourcesCommonPath": pathJoin(PUBLIC_URL, mockTestingResourcesCommonPath),
|
||||||
|
"resourceUrl": "#",
|
||||||
|
"accountUrl": "#",
|
||||||
|
"applicationsUrl": "#",
|
||||||
|
"getLogoutUrl": () => "#",
|
||||||
|
"logUrl": "#",
|
||||||
|
"passwordUrl": "#",
|
||||||
|
"sessionsUrl": "#",
|
||||||
|
"socialUrl": "#",
|
||||||
|
"totpUrl": "#"
|
||||||
|
},
|
||||||
|
"realm": {
|
||||||
|
"internationalizationEnabled": true,
|
||||||
|
"userManagedAccessAllowed": true
|
||||||
|
},
|
||||||
|
"messagesPerField": {
|
||||||
|
"printIfExists": () => {
|
||||||
|
return undefined;
|
||||||
|
},
|
||||||
|
"existsError": () => false,
|
||||||
|
"get": key => `Fake error for ${key}`,
|
||||||
|
"exists": () => false
|
||||||
|
},
|
||||||
|
"locale": {
|
||||||
|
"supported": [
|
||||||
|
/* spell-checker: disable */
|
||||||
|
{
|
||||||
|
"url": "/auth/realms/myrealm/login-actions/authenticate?client_id=account&tab_id=HoAx28ja4xg&execution=ee6c2834-46a4-4a20-a1b6-f6d6f6451b36&kc_locale=de",
|
||||||
|
"label": "Deutsch",
|
||||||
|
"languageTag": "de"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "/auth/realms/myrealm/login-actions/authenticate?client_id=account&tab_id=HoAx28ja4xg&execution=ee6c2834-46a4-4a20-a1b6-f6d6f6451b36&kc_locale=no",
|
||||||
|
"label": "Norsk",
|
||||||
|
"languageTag": "no"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "/auth/realms/myrealm/login-actions/authenticate?client_id=account&tab_id=HoAx28ja4xg&execution=ee6c2834-46a4-4a20-a1b6-f6d6f6451b36&kc_locale=ru",
|
||||||
|
"label": "Русский",
|
||||||
|
"languageTag": "ru"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "/auth/realms/myrealm/login-actions/authenticate?client_id=account&tab_id=HoAx28ja4xg&execution=ee6c2834-46a4-4a20-a1b6-f6d6f6451b36&kc_locale=sv",
|
||||||
|
"label": "Svenska",
|
||||||
|
"languageTag": "sv"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "/auth/realms/myrealm/login-actions/authenticate?client_id=account&tab_id=HoAx28ja4xg&execution=ee6c2834-46a4-4a20-a1b6-f6d6f6451b36&kc_locale=pt-BR",
|
||||||
|
"label": "Português (Brasil)",
|
||||||
|
"languageTag": "pt-BR"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "/auth/realms/myrealm/login-actions/authenticate?client_id=account&tab_id=HoAx28ja4xg&execution=ee6c2834-46a4-4a20-a1b6-f6d6f6451b36&kc_locale=lt",
|
||||||
|
"label": "Lietuvių",
|
||||||
|
"languageTag": "lt"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "/auth/realms/myrealm/login-actions/authenticate?client_id=account&tab_id=HoAx28ja4xg&execution=ee6c2834-46a4-4a20-a1b6-f6d6f6451b36&kc_locale=en",
|
||||||
|
"label": "English",
|
||||||
|
"languageTag": "en"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "/auth/realms/myrealm/login-actions/authenticate?client_id=account&tab_id=HoAx28ja4xg&execution=ee6c2834-46a4-4a20-a1b6-f6d6f6451b36&kc_locale=it",
|
||||||
|
"label": "Italiano",
|
||||||
|
"languageTag": "it"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "/auth/realms/myrealm/login-actions/authenticate?client_id=account&tab_id=HoAx28ja4xg&execution=ee6c2834-46a4-4a20-a1b6-f6d6f6451b36&kc_locale=fr",
|
||||||
|
"label": "Français",
|
||||||
|
"languageTag": "fr"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "/auth/realms/myrealm/login-actions/authenticate?client_id=account&tab_id=HoAx28ja4xg&execution=ee6c2834-46a4-4a20-a1b6-f6d6f6451b36&kc_locale=zh-CN",
|
||||||
|
"label": "中文简体",
|
||||||
|
"languageTag": "zh-CN"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "/auth/realms/myrealm/login-actions/authenticate?client_id=account&tab_id=HoAx28ja4xg&execution=ee6c2834-46a4-4a20-a1b6-f6d6f6451b36&kc_locale=es",
|
||||||
|
"label": "Español",
|
||||||
|
"languageTag": "es"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "/auth/realms/myrealm/login-actions/authenticate?client_id=account&tab_id=HoAx28ja4xg&execution=ee6c2834-46a4-4a20-a1b6-f6d6f6451b36&kc_locale=cs",
|
||||||
|
"label": "Čeština",
|
||||||
|
"languageTag": "cs"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "/auth/realms/myrealm/login-actions/authenticate?client_id=account&tab_id=HoAx28ja4xg&execution=ee6c2834-46a4-4a20-a1b6-f6d6f6451b36&kc_locale=ja",
|
||||||
|
"label": "日本語",
|
||||||
|
"languageTag": "ja"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "/auth/realms/myrealm/login-actions/authenticate?client_id=account&tab_id=HoAx28ja4xg&execution=ee6c2834-46a4-4a20-a1b6-f6d6f6451b36&kc_locale=sk",
|
||||||
|
"label": "Slovenčina",
|
||||||
|
"languageTag": "sk"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "/auth/realms/myrealm/login-actions/authenticate?client_id=account&tab_id=HoAx28ja4xg&execution=ee6c2834-46a4-4a20-a1b6-f6d6f6451b36&kc_locale=pl",
|
||||||
|
"label": "Polski",
|
||||||
|
"languageTag": "pl"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "/auth/realms/myrealm/login-actions/authenticate?client_id=account&tab_id=HoAx28ja4xg&execution=ee6c2834-46a4-4a20-a1b6-f6d6f6451b36&kc_locale=ca",
|
||||||
|
"label": "Català",
|
||||||
|
"languageTag": "ca"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "/auth/realms/myrealm/login-actions/authenticate?client_id=account&tab_id=HoAx28ja4xg&execution=ee6c2834-46a4-4a20-a1b6-f6d6f6451b36&kc_locale=nl",
|
||||||
|
"label": "Nederlands",
|
||||||
|
"languageTag": "nl"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"url": "/auth/realms/myrealm/login-actions/authenticate?client_id=account&tab_id=HoAx28ja4xg&execution=ee6c2834-46a4-4a20-a1b6-f6d6f6451b36&kc_locale=tr",
|
||||||
|
"label": "Türkçe",
|
||||||
|
"languageTag": "tr"
|
||||||
|
}
|
||||||
|
/* spell-checker: enable */
|
||||||
|
],
|
||||||
|
"currentLanguageTag": "en"
|
||||||
|
},
|
||||||
|
"message": {
|
||||||
|
"type": "success",
|
||||||
|
"summary": "This is a test message"
|
||||||
|
},
|
||||||
|
"features": {
|
||||||
|
"authorization": true,
|
||||||
|
"identityFederation": true,
|
||||||
|
"log": true,
|
||||||
|
"passwordUpdateSupported": true
|
||||||
|
},
|
||||||
|
"referrer": undefined,
|
||||||
|
"account": {
|
||||||
|
"firstName": "john",
|
||||||
|
"lastName": "doe",
|
||||||
|
"email": "john.doe@code.gouv.fr",
|
||||||
|
"username": "doe_j"
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const kcContextMocks: KcContext[] = [
|
||||||
|
id<KcContext.Password>({
|
||||||
|
...kcContextCommonMock,
|
||||||
|
"pageId": "password.ftl",
|
||||||
|
"password": {
|
||||||
|
"passwordSet": true
|
||||||
|
},
|
||||||
|
"stateChecker": "state checker"
|
||||||
|
}),
|
||||||
|
id<KcContext.Account>({
|
||||||
|
...kcContextCommonMock,
|
||||||
|
"pageId": "account.ftl",
|
||||||
|
"url": {
|
||||||
|
...kcContextCommonMock.url,
|
||||||
|
"referrerURI": "#",
|
||||||
|
"accountUrl": "#"
|
||||||
|
},
|
||||||
|
"realm": {
|
||||||
|
...kcContextCommonMock.realm,
|
||||||
|
"registrationEmailAsUsername": true,
|
||||||
|
"editUsernameAllowed": true
|
||||||
|
},
|
||||||
|
"stateChecker": ""
|
||||||
|
})
|
||||||
|
];
|
12
src/account/lib/useGetClassName.ts
Normal file
12
src/account/lib/useGetClassName.ts
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
import { createUseClassName } from "keycloakify/lib/useGetClassName";
|
||||||
|
import type { ClassKey } from "keycloakify/account/TemplateProps";
|
||||||
|
|
||||||
|
export const { useGetClassName } = createUseClassName<ClassKey>({
|
||||||
|
"defaultClasses": {
|
||||||
|
"kcBodyClass": undefined,
|
||||||
|
"kcButtonClass": "btn",
|
||||||
|
"kcButtonPrimaryClass": "btn-primary",
|
||||||
|
"kcButtonLargeClass": "btn-lg",
|
||||||
|
"kcButtonDefaultClass": "btn-default"
|
||||||
|
}
|
||||||
|
});
|
134
src/account/pages/Account.tsx
Normal file
134
src/account/pages/Account.tsx
Normal file
@ -0,0 +1,134 @@
|
|||||||
|
import { clsx } from "keycloakify/tools/clsx";
|
||||||
|
import type { PageProps } from "keycloakify/account/pages/PageProps";
|
||||||
|
import { useGetClassName } from "keycloakify/account/lib/useGetClassName";
|
||||||
|
import type { KcContext } from "../kcContext";
|
||||||
|
import type { I18n } from "../i18n";
|
||||||
|
|
||||||
|
export default function LogoutConfirm(props: PageProps<Extract<KcContext, { pageId: "account.ftl" }>, I18n>) {
|
||||||
|
const { kcContext, i18n, doUseDefaultCss, Template, classes } = props;
|
||||||
|
|
||||||
|
const { getClassName } = useGetClassName({
|
||||||
|
doUseDefaultCss,
|
||||||
|
"classes": {
|
||||||
|
...classes,
|
||||||
|
"kcBodyClass": clsx(classes?.kcBodyClass, "user")
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const { url, realm, messagesPerField, stateChecker, account } = kcContext;
|
||||||
|
|
||||||
|
const { msg } = i18n;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Template {...{ kcContext, i18n, doUseDefaultCss, classes }} active="account">
|
||||||
|
<div className="row">
|
||||||
|
<div className="col-md-10">
|
||||||
|
<h2>{msg("editAccountHtmlTitle")}</h2>
|
||||||
|
</div>
|
||||||
|
<div className="col-md-2 subtitle">
|
||||||
|
<span className="subtitle">
|
||||||
|
<span className="required">*</span> {msg("requiredFields")}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<form action={url.accountUrl} className="form-horizontal" method="post">
|
||||||
|
<input type="hidden" id="stateChecker" name="stateChecker" value={stateChecker} />
|
||||||
|
|
||||||
|
{!realm.registrationEmailAsUsername && (
|
||||||
|
<div className={clsx("form-group", messagesPerField.printIfExists("username", "has-error"))}>
|
||||||
|
<div className="col-sm-2 col-md-2">
|
||||||
|
<label htmlFor="username" className="control-label">
|
||||||
|
{msg("username")}
|
||||||
|
</label>
|
||||||
|
{realm.editUsernameAllowed && <span className="required">*</span>}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="col-sm-10 col-md-10">
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
className="form-control"
|
||||||
|
id="username"
|
||||||
|
name="username"
|
||||||
|
disabled={!realm.editUsernameAllowed}
|
||||||
|
value={account.username ?? ""}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<div className={clsx("form-group", messagesPerField.printIfExists("email", "has-error"))}>
|
||||||
|
<div className="col-sm-2 col-md-2">
|
||||||
|
<label htmlFor="email" className="control-label">
|
||||||
|
{msg("email")}
|
||||||
|
</label>{" "}
|
||||||
|
<span className="required">*</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="col-sm-10 col-md-10">
|
||||||
|
<input type="text" className="form-control" id="email" name="email" autoFocus value={account.email ?? ""} />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className={clsx("form-group", messagesPerField.printIfExists("firstName", "has-error"))}>
|
||||||
|
<div className="col-sm-2 col-md-2">
|
||||||
|
<label htmlFor="firstName" className="control-label">
|
||||||
|
{msg("firstName")}
|
||||||
|
</label>{" "}
|
||||||
|
<span className="required">*</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="col-sm-10 col-md-10">
|
||||||
|
<input type="text" className="form-control" id="firstName" name="firstName" value={account.firstName ?? ""} />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className={clsx("form-group", messagesPerField.printIfExists("lastName", "has-error"))}>
|
||||||
|
<div className="col-sm-2 col-md-2">
|
||||||
|
<label htmlFor="lastName" className="control-label">
|
||||||
|
{msg("lastName")}
|
||||||
|
</label>{" "}
|
||||||
|
<span className="required">*</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="col-sm-10 col-md-10">
|
||||||
|
<input type="text" className="form-control" id="lastName" name="lastName" value={account.lastName ?? ""} />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="form-group">
|
||||||
|
<div id="kc-form-buttons" className="col-md-offset-2 col-md-10 submit">
|
||||||
|
<div>
|
||||||
|
{url.referrerURI !== undefined && <a href={url.referrerURI}>${msg("backToApplication")}</a>}
|
||||||
|
<button
|
||||||
|
type="submit"
|
||||||
|
className={clsx(
|
||||||
|
getClassName("kcButtonClass"),
|
||||||
|
getClassName("kcButtonPrimaryClass"),
|
||||||
|
getClassName("kcButtonLargeClass")
|
||||||
|
)}
|
||||||
|
name="submitAction"
|
||||||
|
value="Save"
|
||||||
|
>
|
||||||
|
{msg("doSave")}
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
type="submit"
|
||||||
|
className={clsx(
|
||||||
|
getClassName("kcButtonClass"),
|
||||||
|
getClassName("kcButtonDefaultClass"),
|
||||||
|
getClassName("kcButtonLargeClass")
|
||||||
|
)}
|
||||||
|
name="submitAction"
|
||||||
|
value="Cancel"
|
||||||
|
>
|
||||||
|
{msg("doCancel")}
|
||||||
|
</button>
|
||||||
|
I
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</Template>
|
||||||
|
);
|
||||||
|
}
|
11
src/account/pages/PageProps.ts
Normal file
11
src/account/pages/PageProps.ts
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
import type { LazyExoticComponent } from "react";
|
||||||
|
import type { I18n } from "keycloakify/account/i18n";
|
||||||
|
import type { TemplateProps, ClassKey } from "keycloakify/account/TemplateProps";
|
||||||
|
|
||||||
|
export type PageProps<KcContext, I18nExtended extends I18n> = {
|
||||||
|
Template: LazyExoticComponent<(props: TemplateProps<any, any>) => JSX.Element | null>;
|
||||||
|
kcContext: KcContext;
|
||||||
|
i18n: I18nExtended;
|
||||||
|
doUseDefaultCss: boolean;
|
||||||
|
classes?: Partial<Record<ClassKey, string>>;
|
||||||
|
};
|
105
src/account/pages/Password.tsx
Normal file
105
src/account/pages/Password.tsx
Normal file
@ -0,0 +1,105 @@
|
|||||||
|
import { clsx } from "keycloakify/tools/clsx";
|
||||||
|
import type { PageProps } from "keycloakify/account/pages/PageProps";
|
||||||
|
import { useGetClassName } from "keycloakify/account/lib/useGetClassName";
|
||||||
|
import type { KcContext } from "../kcContext";
|
||||||
|
import type { I18n } from "../i18n";
|
||||||
|
|
||||||
|
export default function LogoutConfirm(props: PageProps<Extract<KcContext, { pageId: "password.ftl" }>, I18n>) {
|
||||||
|
const { kcContext, i18n, doUseDefaultCss, Template, classes } = props;
|
||||||
|
|
||||||
|
const { getClassName } = useGetClassName({
|
||||||
|
doUseDefaultCss,
|
||||||
|
"classes": {
|
||||||
|
...classes,
|
||||||
|
"kcBodyClass": clsx(classes?.kcBodyClass, "password")
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const { url, password, account, stateChecker } = kcContext;
|
||||||
|
|
||||||
|
const { msg } = i18n;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Template {...{ kcContext, i18n, doUseDefaultCss, classes }} active="password">
|
||||||
|
<div className="row">
|
||||||
|
<div className="col-md-10">
|
||||||
|
<h2>{msg("changePasswordHtmlTitle")}</h2>
|
||||||
|
</div>
|
||||||
|
<div className="col-md-2 subtitle">
|
||||||
|
<span className="subtitle">${msg("allFieldsRequired")}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<form action={url.passwordUrl} className="form-horizontal" method="post">
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
id="username"
|
||||||
|
name="username"
|
||||||
|
value={account.username ?? ""}
|
||||||
|
autoComplete="username"
|
||||||
|
readOnly
|
||||||
|
style={{ "display": "none;" }}
|
||||||
|
/>
|
||||||
|
|
||||||
|
{password.passwordSet && (
|
||||||
|
<div className="form-group">
|
||||||
|
<div className="col-sm-2 col-md-2">
|
||||||
|
<label htmlFor="password" className="control-label">
|
||||||
|
{msg("password")}
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="col-sm-10 col-md-10">
|
||||||
|
<input type="password" className="form-control" id="password" name="password" autoFocus autoComplete="current-password" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<input type="hidden" id="stateChecker" name="stateChecker" value={stateChecker} />
|
||||||
|
|
||||||
|
<div className="form-group">
|
||||||
|
<div className="col-sm-2 col-md-2">
|
||||||
|
<label htmlFor="password-new" className="control-label">
|
||||||
|
{msg("passwordNew")}
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="col-sm-10 col-md-10">
|
||||||
|
<input type="password" className="form-control" id="password-new" name="password-new" autoComplete="new-password" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="form-group">
|
||||||
|
<div className="col-sm-2 col-md-2">
|
||||||
|
<label htmlFor="password-confirm" className="control-label two-lines">
|
||||||
|
{msg("passwordConfirm")}
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="col-sm-10 col-md-10">
|
||||||
|
<input type="password" className="form-control" id="password-confirm" name="password-confirm" autoComplete="new-password" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="form-group">
|
||||||
|
<div id="kc-form-buttons" className="col-md-offset-2 col-md-10 submit">
|
||||||
|
<div>
|
||||||
|
<button
|
||||||
|
type="submit"
|
||||||
|
className={clsx(
|
||||||
|
getClassName("kcButtonClass"),
|
||||||
|
getClassName("kcButtonPrimaryClass"),
|
||||||
|
getClassName("kcButtonLargeClass")
|
||||||
|
)}
|
||||||
|
name="submitAction"
|
||||||
|
value="Save"
|
||||||
|
>
|
||||||
|
{msg("doSave")}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</Template>
|
||||||
|
);
|
||||||
|
}
|
@ -1,155 +0,0 @@
|
|||||||
import { generateKeycloakThemeResources } from "./generateKeycloakThemeResources";
|
|
||||||
import { generateJavaStackFiles } from "./generateJavaStackFiles";
|
|
||||||
import { join as pathJoin, relative as pathRelative, basename as pathBasename } from "path";
|
|
||||||
import * as child_process from "child_process";
|
|
||||||
import { generateStartKeycloakTestingContainer } from "./generateStartKeycloakTestingContainer";
|
|
||||||
import { URL } from "url";
|
|
||||||
import * as fs from "fs";
|
|
||||||
|
|
||||||
type ParsedPackageJson = {
|
|
||||||
name: string;
|
|
||||||
version: string;
|
|
||||||
homepage?: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
const reactProjectDirPath = process.cwd();
|
|
||||||
|
|
||||||
const doUseExternalAssets = process.argv[2]?.toLowerCase() === "--external-assets";
|
|
||||||
|
|
||||||
const parsedPackageJson: ParsedPackageJson = require(pathJoin(reactProjectDirPath, "package.json"));
|
|
||||||
|
|
||||||
export const keycloakThemeBuildingDirPath = pathJoin(reactProjectDirPath, "build_keycloak");
|
|
||||||
export const keycloakThemeEmailDirPath = pathJoin(keycloakThemeBuildingDirPath, "..", "keycloak_email");
|
|
||||||
|
|
||||||
function sanitizeThemeName(name: string) {
|
|
||||||
return name
|
|
||||||
.replace(/^@(.*)/, "$1")
|
|
||||||
.split("/")
|
|
||||||
.join("-");
|
|
||||||
}
|
|
||||||
|
|
||||||
export function main() {
|
|
||||||
console.log("🔏 Building the keycloak theme...⌚");
|
|
||||||
|
|
||||||
const extraPagesId: string[] = (parsedPackageJson as any)["keycloakify"]?.["extraPages"] ?? [];
|
|
||||||
const extraThemeProperties: string[] = (parsedPackageJson as any)["keycloakify"]?.["extraThemeProperties"] ?? [];
|
|
||||||
const themeName = sanitizeThemeName(parsedPackageJson.name);
|
|
||||||
|
|
||||||
const { doBundleEmailTemplate } = generateKeycloakThemeResources({
|
|
||||||
keycloakThemeBuildingDirPath,
|
|
||||||
keycloakThemeEmailDirPath,
|
|
||||||
"reactAppBuildDirPath": pathJoin(reactProjectDirPath, "build"),
|
|
||||||
themeName,
|
|
||||||
...(() => {
|
|
||||||
const url = (() => {
|
|
||||||
const { homepage } = parsedPackageJson;
|
|
||||||
|
|
||||||
if (homepage !== undefined) {
|
|
||||||
return new URL(homepage);
|
|
||||||
}
|
|
||||||
|
|
||||||
const cnameFilePath = pathJoin(reactProjectDirPath, "public", "CNAME");
|
|
||||||
|
|
||||||
if (fs.existsSync(cnameFilePath)) {
|
|
||||||
return new URL(`https://${fs.readFileSync(cnameFilePath).toString("utf8").replace(/\s+$/, "")}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
return undefined;
|
|
||||||
})();
|
|
||||||
|
|
||||||
return {
|
|
||||||
"urlPathname": url === undefined ? "/" : url.pathname.replace(/([^/])$/, "$1/"),
|
|
||||||
"urlOrigin": !doUseExternalAssets
|
|
||||||
? undefined
|
|
||||||
: (() => {
|
|
||||||
if (url === undefined) {
|
|
||||||
console.error("ERROR: You must specify 'homepage' in your package.json");
|
|
||||||
process.exit(-1);
|
|
||||||
}
|
|
||||||
|
|
||||||
return url.origin;
|
|
||||||
})(),
|
|
||||||
};
|
|
||||||
})(),
|
|
||||||
extraPagesId,
|
|
||||||
extraThemeProperties,
|
|
||||||
//We have to leave it at that otherwise we break our default theme.
|
|
||||||
//Problem is that we can't guarantee that the the old resources
|
|
||||||
//will still be available on the newer keycloak version.
|
|
||||||
"keycloakVersion": "11.0.3",
|
|
||||||
});
|
|
||||||
|
|
||||||
const { jarFilePath } = generateJavaStackFiles({
|
|
||||||
"version": parsedPackageJson.version,
|
|
||||||
themeName,
|
|
||||||
"homepage": parsedPackageJson.homepage,
|
|
||||||
keycloakThemeBuildingDirPath,
|
|
||||||
doBundleEmailTemplate,
|
|
||||||
});
|
|
||||||
|
|
||||||
child_process.execSync("mvn package", {
|
|
||||||
"cwd": keycloakThemeBuildingDirPath,
|
|
||||||
});
|
|
||||||
|
|
||||||
//We want, however, to test in a container running the latest Keycloak version
|
|
||||||
const containerKeycloakVersion = "18.0.2";
|
|
||||||
|
|
||||||
generateStartKeycloakTestingContainer({
|
|
||||||
keycloakThemeBuildingDirPath,
|
|
||||||
themeName,
|
|
||||||
"keycloakVersion": containerKeycloakVersion,
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log(
|
|
||||||
[
|
|
||||||
"",
|
|
||||||
`✅ Your keycloak theme has been generated and bundled into ./${pathRelative(reactProjectDirPath, jarFilePath)} 🚀`,
|
|
||||||
`It is to be placed in "/opt/keycloak/providers" in the container running a quay.io/keycloak/keycloak Docker image.`,
|
|
||||||
"",
|
|
||||||
//TODO: Restore when we find a good Helm chart for Keycloak.
|
|
||||||
//"Using Helm (https://github.com/codecentric/helm-charts), edit to reflect:",
|
|
||||||
"",
|
|
||||||
"value.yaml: ",
|
|
||||||
" extraInitContainers: |",
|
|
||||||
" - name: realm-ext-provider",
|
|
||||||
" image: curlimages/curl",
|
|
||||||
" imagePullPolicy: IfNotPresent",
|
|
||||||
" command:",
|
|
||||||
" - sh",
|
|
||||||
" args:",
|
|
||||||
" - -c",
|
|
||||||
` - curl -L -f -S -o /extensions/${pathBasename(jarFilePath)} https://AN.URL.FOR/${pathBasename(jarFilePath)}`,
|
|
||||||
" volumeMounts:",
|
|
||||||
" - name: extensions",
|
|
||||||
" mountPath: /extensions",
|
|
||||||
" ",
|
|
||||||
" extraVolumeMounts: |",
|
|
||||||
" - name: extensions",
|
|
||||||
" mountPath: /opt/keycloak/providers",
|
|
||||||
" extraEnv: |",
|
|
||||||
" - name: KEYCLOAK_USER",
|
|
||||||
" value: admin",
|
|
||||||
" - name: KEYCLOAK_PASSWORD",
|
|
||||||
" value: xxxxxxxxx",
|
|
||||||
" - name: JAVA_OPTS",
|
|
||||||
" value: -Dkeycloak.profile=preview",
|
|
||||||
"",
|
|
||||||
"",
|
|
||||||
`To test your theme locally you can spin up a Keycloak ${containerKeycloakVersion} container image with the theme pre loaded by running:`,
|
|
||||||
"",
|
|
||||||
`👉 $ ./${pathRelative(reactProjectDirPath, pathJoin(keycloakThemeBuildingDirPath, generateStartKeycloakTestingContainer.basename))} 👈`,
|
|
||||||
"",
|
|
||||||
"Test with different Keycloak versions by editing the .sh file. see available versions here: https://quay.io/repository/keycloak/keycloak?tab=tags",
|
|
||||||
"",
|
|
||||||
"Once your container is up and running: ",
|
|
||||||
"- Log into the admin console 👉 http://localhost:8080/admin username: admin, password: admin 👈",
|
|
||||||
'- Create a realm named "myrealm"',
|
|
||||||
'- Create a client with ID: "myclient", "Root URL": "https://www.keycloak.org/app/" and "Valid redirect URIs": "https://www.keycloak.org/app/*"',
|
|
||||||
`- Select Login Theme: ${themeName} (don't forget to save at the bottom of the page)`,
|
|
||||||
`- Go to 👉 https://www.keycloak.org/app/ 👈 Click "Save" then "Sign in". You should see your login page`,
|
|
||||||
"",
|
|
||||||
"Video demoing this process: https://youtu.be/N3wlBoH4hKg",
|
|
||||||
"",
|
|
||||||
].join("\n"),
|
|
||||||
);
|
|
||||||
}
|
|
@ -1,139 +0,0 @@
|
|||||||
import cheerio from "cheerio";
|
|
||||||
import { replaceImportsFromStaticInJsCode, replaceImportsInInlineCssCode, generateCssCodeToDefineGlobals } from "../replaceImportFromStatic";
|
|
||||||
import * as fs from "fs";
|
|
||||||
import { join as pathJoin } from "path";
|
|
||||||
import { objectKeys } from "tsafe/objectKeys";
|
|
||||||
import { ftlValuesGlobalName } from "../ftlValuesGlobalName";
|
|
||||||
|
|
||||||
// https://github.com/keycloak/keycloak/blob/main/services/src/main/java/org/keycloak/forms/login/freemarker/Templates.java
|
|
||||||
export const pageIds = [
|
|
||||||
"login.ftl",
|
|
||||||
"register.ftl",
|
|
||||||
"register-user-profile.ftl",
|
|
||||||
"info.ftl",
|
|
||||||
"error.ftl",
|
|
||||||
"login-reset-password.ftl",
|
|
||||||
"login-verify-email.ftl",
|
|
||||||
"terms.ftl",
|
|
||||||
"login-otp.ftl",
|
|
||||||
"login-update-profile.ftl",
|
|
||||||
"login-update-password.ftl",
|
|
||||||
"login-idp-link-confirm.ftl",
|
|
||||||
"login-idp-link-email.ftl",
|
|
||||||
"login-page-expired.ftl",
|
|
||||||
"login-config-totp.ftl",
|
|
||||||
"logout-confirm.ftl",
|
|
||||||
] as const;
|
|
||||||
|
|
||||||
export type PageId = typeof pageIds[number];
|
|
||||||
|
|
||||||
export function generateFtlFilesCodeFactory(params: {
|
|
||||||
cssGlobalsToDefine: Record<string, string>;
|
|
||||||
indexHtmlCode: string;
|
|
||||||
urlPathname: string;
|
|
||||||
urlOrigin: undefined | string;
|
|
||||||
}) {
|
|
||||||
const { cssGlobalsToDefine, indexHtmlCode, urlPathname, urlOrigin } = params;
|
|
||||||
|
|
||||||
const $ = cheerio.load(indexHtmlCode);
|
|
||||||
|
|
||||||
$("script:not([src])").each((...[, element]) => {
|
|
||||||
const { fixedJsCode } = replaceImportsFromStaticInJsCode({
|
|
||||||
"jsCode": $(element).html()!,
|
|
||||||
urlOrigin,
|
|
||||||
});
|
|
||||||
|
|
||||||
$(element).text(fixedJsCode);
|
|
||||||
});
|
|
||||||
|
|
||||||
$("style").each((...[, element]) => {
|
|
||||||
const { fixedCssCode } = replaceImportsInInlineCssCode({
|
|
||||||
"cssCode": $(element).html()!,
|
|
||||||
"urlPathname": params.urlPathname,
|
|
||||||
urlOrigin,
|
|
||||||
});
|
|
||||||
|
|
||||||
$(element).text(fixedCssCode);
|
|
||||||
});
|
|
||||||
|
|
||||||
(
|
|
||||||
[
|
|
||||||
["link", "href"],
|
|
||||||
["script", "src"],
|
|
||||||
] as const
|
|
||||||
).forEach(([selector, attrName]) =>
|
|
||||||
$(selector).each((...[, element]) => {
|
|
||||||
const href = $(element).attr(attrName);
|
|
||||||
|
|
||||||
if (href === undefined) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
$(element).attr(
|
|
||||||
attrName,
|
|
||||||
urlOrigin !== undefined
|
|
||||||
? href.replace(/^\//, `${urlOrigin}/`)
|
|
||||||
: href.replace(new RegExp(`^${urlPathname.replace(/\//g, "\\/")}`), "${url.resourcesPath}/build/"),
|
|
||||||
);
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
|
|
||||||
//FTL is no valid html, we can't insert with cheerio, we put placeholder for injecting later.
|
|
||||||
const replaceValueBySearchValue = {
|
|
||||||
'{ "x": "vIdLqMeOed9sdLdIdOxdK0d" }': fs
|
|
||||||
.readFileSync(pathJoin(__dirname, "ftl_object_to_js_code_declaring_an_object.ftl"))
|
|
||||||
.toString("utf8")
|
|
||||||
.match(/^<script>const _=((?:.|\n)+)<\/script>[\n]?$/)![1],
|
|
||||||
"<!-- xIdLqMeOedErIdLsPdNdI9dSlxI -->": [
|
|
||||||
"<#if scripts??>",
|
|
||||||
" <#list scripts as script>",
|
|
||||||
' <script src="${script}" type="text/javascript"></script>',
|
|
||||||
" </#list>",
|
|
||||||
"</#if>",
|
|
||||||
].join("\n"),
|
|
||||||
};
|
|
||||||
|
|
||||||
$("head").prepend(
|
|
||||||
[
|
|
||||||
...(Object.keys(cssGlobalsToDefine).length === 0
|
|
||||||
? []
|
|
||||||
: [
|
|
||||||
"",
|
|
||||||
"<style>",
|
|
||||||
generateCssCodeToDefineGlobals({
|
|
||||||
cssGlobalsToDefine,
|
|
||||||
urlPathname,
|
|
||||||
}).cssCodeToPrependInHead,
|
|
||||||
"</style>",
|
|
||||||
"",
|
|
||||||
]),
|
|
||||||
"<script>",
|
|
||||||
` window.${ftlValuesGlobalName}= ${objectKeys(replaceValueBySearchValue)[0]};`,
|
|
||||||
"</script>",
|
|
||||||
"",
|
|
||||||
objectKeys(replaceValueBySearchValue)[1],
|
|
||||||
].join("\n"),
|
|
||||||
);
|
|
||||||
|
|
||||||
const partiallyFixedIndexHtmlCode = $.html();
|
|
||||||
|
|
||||||
function generateFtlFilesCode(params: { pageId: string }): {
|
|
||||||
ftlCode: string;
|
|
||||||
} {
|
|
||||||
const { pageId } = params;
|
|
||||||
|
|
||||||
const $ = cheerio.load(partiallyFixedIndexHtmlCode);
|
|
||||||
|
|
||||||
let ftlCode = $.html();
|
|
||||||
|
|
||||||
Object.entries({
|
|
||||||
...replaceValueBySearchValue,
|
|
||||||
//If updated, don't forget to change in the ftl script as well.
|
|
||||||
"PAGE_ID_xIgLsPgGId9D8e": pageId,
|
|
||||||
}).map(([searchValue, replaceValue]) => (ftlCode = ftlCode.replace(searchValue, replaceValue)));
|
|
||||||
|
|
||||||
return { ftlCode };
|
|
||||||
}
|
|
||||||
|
|
||||||
return { generateFtlFilesCode };
|
|
||||||
}
|
|
@ -1,168 +0,0 @@
|
|||||||
import { transformCodebase } from "../tools/transformCodebase";
|
|
||||||
import * as fs from "fs";
|
|
||||||
import { join as pathJoin, basename as pathBasename } from "path";
|
|
||||||
import { replaceImportsInCssCode, replaceImportsFromStaticInJsCode } from "./replaceImportFromStatic";
|
|
||||||
import { generateFtlFilesCodeFactory, pageIds } from "./generateFtl";
|
|
||||||
import { downloadBuiltinKeycloakTheme } from "../download-builtin-keycloak-theme";
|
|
||||||
import * as child_process from "child_process";
|
|
||||||
import { mockTestingResourcesCommonPath, mockTestingResourcesPath, mockTestingSubDirOfPublicDirBasename } from "../mockTestingResourcesPath";
|
|
||||||
import { isInside } from "../tools/isInside";
|
|
||||||
|
|
||||||
export function generateKeycloakThemeResources(params: {
|
|
||||||
themeName: string;
|
|
||||||
reactAppBuildDirPath: string;
|
|
||||||
keycloakThemeBuildingDirPath: string;
|
|
||||||
keycloakThemeEmailDirPath: string;
|
|
||||||
urlPathname: string;
|
|
||||||
//If urlOrigin is not undefined then it means --externals-assets
|
|
||||||
urlOrigin: undefined | string;
|
|
||||||
extraPagesId: string[];
|
|
||||||
extraThemeProperties: string[];
|
|
||||||
keycloakVersion: string;
|
|
||||||
}): { doBundleEmailTemplate: boolean } {
|
|
||||||
const {
|
|
||||||
themeName,
|
|
||||||
reactAppBuildDirPath,
|
|
||||||
keycloakThemeBuildingDirPath,
|
|
||||||
keycloakThemeEmailDirPath,
|
|
||||||
urlPathname,
|
|
||||||
urlOrigin,
|
|
||||||
extraPagesId,
|
|
||||||
extraThemeProperties,
|
|
||||||
keycloakVersion,
|
|
||||||
} = params;
|
|
||||||
|
|
||||||
const themeDirPath = pathJoin(keycloakThemeBuildingDirPath, "src", "main", "resources", "theme", themeName, "login");
|
|
||||||
|
|
||||||
let allCssGlobalsToDefine: Record<string, string> = {};
|
|
||||||
|
|
||||||
transformCodebase({
|
|
||||||
"destDirPath": urlOrigin === undefined ? pathJoin(themeDirPath, "resources", "build") : reactAppBuildDirPath,
|
|
||||||
"srcDirPath": reactAppBuildDirPath,
|
|
||||||
"transformSourceCode": ({ filePath, sourceCode }) => {
|
|
||||||
//NOTE: Prevent cycles, excludes the folder we generated for debug in public/
|
|
||||||
if (
|
|
||||||
urlOrigin === undefined &&
|
|
||||||
isInside({
|
|
||||||
"dirPath": pathJoin(reactAppBuildDirPath, mockTestingSubDirOfPublicDirBasename),
|
|
||||||
filePath,
|
|
||||||
})
|
|
||||||
) {
|
|
||||||
return undefined;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (urlOrigin === undefined && /\.css?$/i.test(filePath)) {
|
|
||||||
const { cssGlobalsToDefine, fixedCssCode } = replaceImportsInCssCode({
|
|
||||||
"cssCode": sourceCode.toString("utf8"),
|
|
||||||
});
|
|
||||||
|
|
||||||
allCssGlobalsToDefine = {
|
|
||||||
...allCssGlobalsToDefine,
|
|
||||||
...cssGlobalsToDefine,
|
|
||||||
};
|
|
||||||
|
|
||||||
return {
|
|
||||||
"modifiedSourceCode": Buffer.from(fixedCssCode, "utf8"),
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
if (/\.js?$/i.test(filePath)) {
|
|
||||||
const { fixedJsCode } = replaceImportsFromStaticInJsCode({
|
|
||||||
"jsCode": sourceCode.toString("utf8"),
|
|
||||||
urlOrigin,
|
|
||||||
});
|
|
||||||
|
|
||||||
return {
|
|
||||||
"modifiedSourceCode": Buffer.from(fixedJsCode, "utf8"),
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
return urlOrigin === undefined ? { "modifiedSourceCode": sourceCode } : undefined;
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
let doBundleEmailTemplate: boolean;
|
|
||||||
|
|
||||||
email: {
|
|
||||||
if (!fs.existsSync(keycloakThemeEmailDirPath)) {
|
|
||||||
console.log(
|
|
||||||
[
|
|
||||||
`Not bundling email template because ${pathBasename(keycloakThemeEmailDirPath)} does not exist`,
|
|
||||||
`To start customizing the email template, run: 👉 npx create-keycloak-email-directory 👈`,
|
|
||||||
].join("\n"),
|
|
||||||
);
|
|
||||||
doBundleEmailTemplate = false;
|
|
||||||
break email;
|
|
||||||
}
|
|
||||||
|
|
||||||
doBundleEmailTemplate = true;
|
|
||||||
|
|
||||||
transformCodebase({
|
|
||||||
"srcDirPath": keycloakThemeEmailDirPath,
|
|
||||||
"destDirPath": pathJoin(themeDirPath, "..", "email"),
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const { generateFtlFilesCode } = generateFtlFilesCodeFactory({
|
|
||||||
"cssGlobalsToDefine": allCssGlobalsToDefine,
|
|
||||||
"indexHtmlCode": fs.readFileSync(pathJoin(reactAppBuildDirPath, "index.html")).toString("utf8"),
|
|
||||||
urlPathname,
|
|
||||||
urlOrigin,
|
|
||||||
});
|
|
||||||
|
|
||||||
[...pageIds, ...extraPagesId].forEach(pageId => {
|
|
||||||
const { ftlCode } = generateFtlFilesCode({ pageId });
|
|
||||||
|
|
||||||
fs.mkdirSync(themeDirPath, { "recursive": true });
|
|
||||||
|
|
||||||
fs.writeFileSync(pathJoin(themeDirPath, pageId), Buffer.from(ftlCode, "utf8"));
|
|
||||||
});
|
|
||||||
|
|
||||||
{
|
|
||||||
const tmpDirPath = pathJoin(themeDirPath, "..", "tmp_xxKdLpdIdLd");
|
|
||||||
|
|
||||||
downloadBuiltinKeycloakTheme({
|
|
||||||
keycloakVersion,
|
|
||||||
"destDirPath": tmpDirPath,
|
|
||||||
});
|
|
||||||
|
|
||||||
const themeResourcesDirPath = pathJoin(themeDirPath, "resources");
|
|
||||||
|
|
||||||
transformCodebase({
|
|
||||||
"srcDirPath": pathJoin(tmpDirPath, "keycloak", "login", "resources"),
|
|
||||||
"destDirPath": themeResourcesDirPath,
|
|
||||||
});
|
|
||||||
|
|
||||||
const reactAppPublicDirPath = pathJoin(reactAppBuildDirPath, "..", "public");
|
|
||||||
|
|
||||||
transformCodebase({
|
|
||||||
"srcDirPath": pathJoin(tmpDirPath, "keycloak", "common", "resources"),
|
|
||||||
"destDirPath": pathJoin(themeResourcesDirPath, pathBasename(mockTestingResourcesCommonPath)),
|
|
||||||
});
|
|
||||||
|
|
||||||
transformCodebase({
|
|
||||||
"srcDirPath": themeResourcesDirPath,
|
|
||||||
"destDirPath": pathJoin(reactAppPublicDirPath, mockTestingResourcesPath),
|
|
||||||
});
|
|
||||||
|
|
||||||
const keycloakResourcesWithinPublicDirPath = pathJoin(reactAppPublicDirPath, mockTestingSubDirOfPublicDirBasename);
|
|
||||||
|
|
||||||
fs.writeFileSync(
|
|
||||||
pathJoin(keycloakResourcesWithinPublicDirPath, "README.txt"),
|
|
||||||
Buffer.from(
|
|
||||||
["This is just a test folder that helps develop", "the login and register page without having to run a Keycloak container"].join(" "),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
fs.writeFileSync(pathJoin(keycloakResourcesWithinPublicDirPath, ".gitignore"), Buffer.from("*", "utf8"));
|
|
||||||
|
|
||||||
child_process.execSync(`rm -r ${tmpDirPath}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
fs.writeFileSync(
|
|
||||||
pathJoin(themeDirPath, "theme.properties"),
|
|
||||||
Buffer.from("parent=keycloak".concat("\n\n", extraThemeProperties.join("\n\n")), "utf8"),
|
|
||||||
);
|
|
||||||
|
|
||||||
return { doBundleEmailTemplate };
|
|
||||||
}
|
|
@ -1,44 +0,0 @@
|
|||||||
import * as fs from "fs";
|
|
||||||
import { join as pathJoin } from "path";
|
|
||||||
|
|
||||||
generateStartKeycloakTestingContainer.basename = "start_keycloak_testing_container.sh";
|
|
||||||
|
|
||||||
const containerName = "keycloak-testing-container";
|
|
||||||
|
|
||||||
/** Files for being able to run a hot reload keycloak container */
|
|
||||||
export function generateStartKeycloakTestingContainer(params: { keycloakVersion: string; themeName: string; keycloakThemeBuildingDirPath: string }) {
|
|
||||||
const { themeName, keycloakThemeBuildingDirPath, keycloakVersion } = params;
|
|
||||||
|
|
||||||
fs.writeFileSync(
|
|
||||||
pathJoin(keycloakThemeBuildingDirPath, generateStartKeycloakTestingContainer.basename),
|
|
||||||
Buffer.from(
|
|
||||||
[
|
|
||||||
"#!/bin/bash",
|
|
||||||
"",
|
|
||||||
`docker rm ${containerName} || true`,
|
|
||||||
"",
|
|
||||||
`cd ${keycloakThemeBuildingDirPath}`,
|
|
||||||
"",
|
|
||||||
"docker run \\",
|
|
||||||
" -p 8080:8080 \\",
|
|
||||||
` --name ${containerName} \\`,
|
|
||||||
" -e KEYCLOAK_ADMIN=admin \\",
|
|
||||||
" -e KEYCLOAK_ADMIN_PASSWORD=admin \\",
|
|
||||||
" -e JAVA_OPTS=-Dkeycloak.profile=preview \\",
|
|
||||||
` -v ${pathJoin(
|
|
||||||
keycloakThemeBuildingDirPath,
|
|
||||||
"src",
|
|
||||||
"main",
|
|
||||||
"resources",
|
|
||||||
"theme",
|
|
||||||
themeName,
|
|
||||||
)}:/opt/keycloak/themes/${themeName}:rw \\`,
|
|
||||||
` -it quay.io/keycloak/keycloak:${keycloakVersion} \\`,
|
|
||||||
` start-dev`,
|
|
||||||
"",
|
|
||||||
].join("\n"),
|
|
||||||
"utf8",
|
|
||||||
),
|
|
||||||
{ "mode": 0o755 },
|
|
||||||
);
|
|
||||||
}
|
|
@ -1,8 +0,0 @@
|
|||||||
#!/usr/bin/env node
|
|
||||||
|
|
||||||
export * from "./build-keycloak-theme";
|
|
||||||
import { main } from "./build-keycloak-theme";
|
|
||||||
|
|
||||||
if (require.main === module) {
|
|
||||||
main();
|
|
||||||
}
|
|
@ -1,116 +0,0 @@
|
|||||||
import * as crypto from "crypto";
|
|
||||||
import { ftlValuesGlobalName } from "./ftlValuesGlobalName";
|
|
||||||
|
|
||||||
export function replaceImportsFromStaticInJsCode(params: { jsCode: string; urlOrigin: undefined | string }): { fixedJsCode: string } {
|
|
||||||
/*
|
|
||||||
NOTE:
|
|
||||||
|
|
||||||
When we have urlOrigin defined it means that
|
|
||||||
we are building with --external-assets
|
|
||||||
so we have to make sur that the fixed js code will run
|
|
||||||
inside and outside keycloak.
|
|
||||||
|
|
||||||
When urlOrigin isn't defined we can assume the fixedJsCode
|
|
||||||
will always run in keycloak context.
|
|
||||||
*/
|
|
||||||
|
|
||||||
const { jsCode, urlOrigin } = params;
|
|
||||||
|
|
||||||
const fixedJsCode = jsCode
|
|
||||||
.replace(
|
|
||||||
/([a-zA-Z]+)\.([a-zA-Z]+)=function\(([a-zA-Z]+)\){return"static\/js\/"/g,
|
|
||||||
(...[, n, u, e]) => `
|
|
||||||
${n}[(function(){
|
|
||||||
${
|
|
||||||
urlOrigin === undefined
|
|
||||||
? `
|
|
||||||
Object.defineProperty(${n}, "p", {
|
|
||||||
get: function() { return window.${ftlValuesGlobalName}.url.resourcesPath; },
|
|
||||||
set: function (){}
|
|
||||||
});
|
|
||||||
`
|
|
||||||
: `
|
|
||||||
var p= "";
|
|
||||||
Object.defineProperty(${n}, "p", {
|
|
||||||
get: function() { return ("${ftlValuesGlobalName}" in window ? "${urlOrigin}" : "") + p; },
|
|
||||||
set: function (value){ p = value;}
|
|
||||||
});
|
|
||||||
`
|
|
||||||
}
|
|
||||||
return "${u}";
|
|
||||||
})()] = function(${e}) { return "${urlOrigin === undefined ? "/build/" : ""}static/js/"`,
|
|
||||||
)
|
|
||||||
.replace(/([a-zA-Z]+\.[a-zA-Z]+)\+"static\//g, (...[, group]) =>
|
|
||||||
urlOrigin === undefined
|
|
||||||
? `window.${ftlValuesGlobalName}.url.resourcesPath + "/build/static/`
|
|
||||||
: `("${ftlValuesGlobalName}" in window ? "${urlOrigin}" : "") + ${group} + "static/`,
|
|
||||||
)
|
|
||||||
//TODO: Write a test case for this
|
|
||||||
.replace(/".chunk.css",([a-zA-Z])+=([a-zA-Z]+\.[a-zA-Z]+)\+([a-zA-Z]+),/, (...[, group1, group2, group3]) =>
|
|
||||||
urlOrigin === undefined
|
|
||||||
? `".chunk.css",${group1} = window.${ftlValuesGlobalName}.url.resourcesPath + "/build/" + ${group3},`
|
|
||||||
: `".chunk.css",${group1} = ("${ftlValuesGlobalName}" in window ? "${urlOrigin}" : "") + ${group2} + ${group3},`,
|
|
||||||
);
|
|
||||||
|
|
||||||
return { fixedJsCode };
|
|
||||||
}
|
|
||||||
|
|
||||||
export function replaceImportsInInlineCssCode(params: { cssCode: string; urlPathname: string; urlOrigin: undefined | string }): {
|
|
||||||
fixedCssCode: string;
|
|
||||||
} {
|
|
||||||
const { cssCode, urlPathname, urlOrigin } = params;
|
|
||||||
|
|
||||||
const fixedCssCode = cssCode.replace(
|
|
||||||
urlPathname === "/" ? /url\(["']?\/([^/][^)"']+)["']?\)/g : new RegExp(`url\\(["']?${urlPathname}([^)"']+)["']?\\)`, "g"),
|
|
||||||
(...[, group]) => `url(${urlOrigin === undefined ? "${url.resourcesPath}/build/" + group : params.urlOrigin + urlPathname + group})`,
|
|
||||||
);
|
|
||||||
|
|
||||||
return { fixedCssCode };
|
|
||||||
}
|
|
||||||
|
|
||||||
export function replaceImportsInCssCode(params: { cssCode: string }): {
|
|
||||||
fixedCssCode: string;
|
|
||||||
cssGlobalsToDefine: Record<string, string>;
|
|
||||||
} {
|
|
||||||
const { cssCode } = params;
|
|
||||||
|
|
||||||
const cssGlobalsToDefine: Record<string, string> = {};
|
|
||||||
|
|
||||||
new Set(cssCode.match(/url\(["']?\/[^/][^)"']+["']?\)[^;}]*/g) ?? []).forEach(
|
|
||||||
match => (cssGlobalsToDefine["url" + crypto.createHash("sha256").update(match).digest("hex").substring(0, 15)] = match),
|
|
||||||
);
|
|
||||||
|
|
||||||
let fixedCssCode = cssCode;
|
|
||||||
|
|
||||||
Object.keys(cssGlobalsToDefine).forEach(
|
|
||||||
cssVariableName =>
|
|
||||||
//NOTE: split/join pattern ~ replace all
|
|
||||||
(fixedCssCode = fixedCssCode.split(cssGlobalsToDefine[cssVariableName]).join(`var(--${cssVariableName})`)),
|
|
||||||
);
|
|
||||||
|
|
||||||
return { fixedCssCode, cssGlobalsToDefine };
|
|
||||||
}
|
|
||||||
|
|
||||||
export function generateCssCodeToDefineGlobals(params: { cssGlobalsToDefine: Record<string, string>; urlPathname: string }): {
|
|
||||||
cssCodeToPrependInHead: string;
|
|
||||||
} {
|
|
||||||
const { cssGlobalsToDefine, urlPathname } = params;
|
|
||||||
|
|
||||||
return {
|
|
||||||
"cssCodeToPrependInHead": [
|
|
||||||
":root {",
|
|
||||||
...Object.keys(cssGlobalsToDefine)
|
|
||||||
.map(cssVariableName =>
|
|
||||||
[
|
|
||||||
`--${cssVariableName}:`,
|
|
||||||
cssGlobalsToDefine[cssVariableName].replace(
|
|
||||||
new RegExp(`url\\(${urlPathname.replace(/\//g, "\\/")}`, "g"),
|
|
||||||
"url(${url.resourcesPath}/build/",
|
|
||||||
),
|
|
||||||
].join(" "),
|
|
||||||
)
|
|
||||||
.map(line => ` ${line};`),
|
|
||||||
"}",
|
|
||||||
].join("\n"),
|
|
||||||
};
|
|
||||||
}
|
|
@ -1,36 +0,0 @@
|
|||||||
#!/usr/bin/env node
|
|
||||||
|
|
||||||
import { downloadBuiltinKeycloakTheme } from "./download-builtin-keycloak-theme";
|
|
||||||
import { keycloakThemeEmailDirPath } from "./build-keycloak-theme";
|
|
||||||
import { join as pathJoin, basename as pathBasename } from "path";
|
|
||||||
import { transformCodebase } from "./tools/transformCodebase";
|
|
||||||
import { promptKeycloakVersion } from "./promptKeycloakVersion";
|
|
||||||
import * as fs from "fs";
|
|
||||||
|
|
||||||
if (require.main === module) {
|
|
||||||
(async () => {
|
|
||||||
if (fs.existsSync(keycloakThemeEmailDirPath)) {
|
|
||||||
console.log(`There is already a ./${pathBasename(keycloakThemeEmailDirPath)} directory in your project. Aborting.`);
|
|
||||||
|
|
||||||
process.exit(-1);
|
|
||||||
}
|
|
||||||
|
|
||||||
const { keycloakVersion } = await promptKeycloakVersion();
|
|
||||||
|
|
||||||
const builtinKeycloakThemeTmpDirPath = pathJoin(keycloakThemeEmailDirPath, "..", "tmp_xIdP3_builtin_keycloak_theme");
|
|
||||||
|
|
||||||
downloadBuiltinKeycloakTheme({
|
|
||||||
keycloakVersion,
|
|
||||||
"destDirPath": builtinKeycloakThemeTmpDirPath,
|
|
||||||
});
|
|
||||||
|
|
||||||
transformCodebase({
|
|
||||||
"srcDirPath": pathJoin(builtinKeycloakThemeTmpDirPath, "base", "email"),
|
|
||||||
"destDirPath": keycloakThemeEmailDirPath,
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log(`./${pathBasename(keycloakThemeEmailDirPath)} ready to be customized`);
|
|
||||||
|
|
||||||
fs.rmSync(builtinKeycloakThemeTmpDirPath, { "recursive": true, "force": true });
|
|
||||||
})();
|
|
||||||
}
|
|
@ -1,33 +1,41 @@
|
|||||||
#!/usr/bin/env node
|
#!/usr/bin/env node
|
||||||
|
|
||||||
import { keycloakThemeBuildingDirPath } from "./build-keycloak-theme";
|
|
||||||
import { join as pathJoin } from "path";
|
import { join as pathJoin } from "path";
|
||||||
import { downloadAndUnzip } from "./tools/downloadAndUnzip";
|
import { downloadAndUnzip } from "./tools/downloadAndUnzip";
|
||||||
import { promptKeycloakVersion } from "./promptKeycloakVersion";
|
import { promptKeycloakVersion } from "./promptKeycloakVersion";
|
||||||
|
import { getCliOptions } from "./tools/cliOptions";
|
||||||
|
import { getLogger } from "./tools/logger";
|
||||||
|
import { getKeycloakBuildPath } from "./keycloakify/build-paths";
|
||||||
|
|
||||||
export function downloadBuiltinKeycloakTheme(params: { keycloakVersion: string; destDirPath: string }) {
|
export async function downloadBuiltinKeycloakTheme(params: { keycloakVersion: string; destDirPath: string; isSilent: boolean }) {
|
||||||
const { keycloakVersion, destDirPath } = params;
|
const { keycloakVersion, destDirPath } = params;
|
||||||
|
|
||||||
for (const ext of ["", "-community"]) {
|
await Promise.all(
|
||||||
downloadAndUnzip({
|
["", "-community"].map(ext =>
|
||||||
"destDirPath": destDirPath,
|
downloadAndUnzip({
|
||||||
"url": `https://github.com/keycloak/keycloak/archive/refs/tags/${keycloakVersion}.zip`,
|
"destDirPath": destDirPath,
|
||||||
"pathOfDirToExtractInArchive": `keycloak-${keycloakVersion}/themes/src/main/resources${ext}/theme`,
|
"url": `https://github.com/keycloak/keycloak/archive/refs/tags/${keycloakVersion}.zip`,
|
||||||
});
|
"pathOfDirToExtractInArchive": `keycloak-${keycloakVersion}/themes/src/main/resources${ext}/theme`
|
||||||
}
|
})
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function main() {
|
||||||
|
const { isSilent } = getCliOptions(process.argv.slice(2));
|
||||||
|
const logger = getLogger({ isSilent });
|
||||||
|
const { keycloakVersion } = await promptKeycloakVersion();
|
||||||
|
|
||||||
|
const destDirPath = pathJoin(getKeycloakBuildPath(), "src", "main", "resources", "theme");
|
||||||
|
|
||||||
|
logger.log(`Downloading builtins theme of Keycloak ${keycloakVersion} here ${destDirPath}`);
|
||||||
|
|
||||||
|
await downloadBuiltinKeycloakTheme({
|
||||||
|
keycloakVersion,
|
||||||
|
destDirPath,
|
||||||
|
isSilent
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (require.main === module) {
|
if (require.main === module) {
|
||||||
(async () => {
|
main();
|
||||||
const { keycloakVersion } = await promptKeycloakVersion();
|
|
||||||
|
|
||||||
const destDirPath = pathJoin(keycloakThemeBuildingDirPath, "src", "main", "resources", "theme");
|
|
||||||
|
|
||||||
console.log(`Downloading builtins theme of Keycloak ${keycloakVersion} here ${destDirPath}`);
|
|
||||||
|
|
||||||
downloadBuiltinKeycloakTheme({
|
|
||||||
keycloakVersion,
|
|
||||||
destDirPath,
|
|
||||||
});
|
|
||||||
})();
|
|
||||||
}
|
}
|
||||||
|
71
src/bin/eject-keycloak-page.ts
Normal file
71
src/bin/eject-keycloak-page.ts
Normal file
@ -0,0 +1,71 @@
|
|||||||
|
#!/usr/bin/env node
|
||||||
|
|
||||||
|
import { getProjectRoot } from "./tools/getProjectRoot";
|
||||||
|
import cliSelect from "cli-select";
|
||||||
|
import {
|
||||||
|
loginThemePageIds,
|
||||||
|
accountThemePageIds,
|
||||||
|
type LoginThemePageId,
|
||||||
|
type AccountThemePageId,
|
||||||
|
themeTypes,
|
||||||
|
type ThemeType
|
||||||
|
} from "./keycloakify/generateFtl/generateFtl";
|
||||||
|
import { capitalize } from "tsafe/capitalize";
|
||||||
|
import { readFile, writeFile } from "fs/promises";
|
||||||
|
import { existsSync } from "fs";
|
||||||
|
import { join as pathJoin, relative as pathRelative } from "path";
|
||||||
|
import { kebabCaseToCamelCase } from "./tools/kebabCaseToSnakeCase";
|
||||||
|
import { assert, Equals } from "tsafe/assert";
|
||||||
|
import { getThemeSrcDirPath } from "./keycloakify/build-paths";
|
||||||
|
|
||||||
|
(async () => {
|
||||||
|
const projectRootDir = getProjectRoot();
|
||||||
|
|
||||||
|
console.log("Select a theme type");
|
||||||
|
|
||||||
|
const { value: themeType } = await cliSelect<ThemeType>({
|
||||||
|
"values": [...themeTypes]
|
||||||
|
}).catch(() => {
|
||||||
|
console.log("Aborting");
|
||||||
|
|
||||||
|
process.exit(-1);
|
||||||
|
});
|
||||||
|
|
||||||
|
console.log("Select a page you would like to eject");
|
||||||
|
|
||||||
|
const { value: pageId } = await cliSelect<LoginThemePageId | AccountThemePageId>({
|
||||||
|
"values": (() => {
|
||||||
|
switch (themeType) {
|
||||||
|
case "login":
|
||||||
|
return [...loginThemePageIds];
|
||||||
|
case "account":
|
||||||
|
return [...accountThemePageIds];
|
||||||
|
}
|
||||||
|
assert<Equals<typeof themeType, never>>(false);
|
||||||
|
})()
|
||||||
|
}).catch(() => {
|
||||||
|
console.log("Aborting");
|
||||||
|
|
||||||
|
process.exit(-1);
|
||||||
|
});
|
||||||
|
|
||||||
|
const pageBasename = capitalize(kebabCaseToCamelCase(pageId)).replace(/ftl$/, "tsx");
|
||||||
|
|
||||||
|
const { themeSrcDirPath } = getThemeSrcDirPath();
|
||||||
|
|
||||||
|
if (themeSrcDirPath === undefined) {
|
||||||
|
throw new Error("Couldn't locate your theme sources");
|
||||||
|
}
|
||||||
|
|
||||||
|
const targetFilePath = pathJoin(themeSrcDirPath, themeType, "pages", pageBasename);
|
||||||
|
|
||||||
|
if (existsSync(targetFilePath)) {
|
||||||
|
console.log(`${pageId} is already ejected, ${pathRelative(process.cwd(), targetFilePath)} already exists`);
|
||||||
|
|
||||||
|
process.exit(-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
writeFile(targetFilePath, await readFile(pathJoin(projectRootDir, "src", themeType, "pages", pageBasename)));
|
||||||
|
|
||||||
|
console.log(`${pathRelative(process.cwd(), targetFilePath)} created`);
|
||||||
|
})();
|
@ -1,81 +0,0 @@
|
|||||||
import "minimal-polyfills/Object.fromEntries";
|
|
||||||
import * as fs from "fs";
|
|
||||||
import { join as pathJoin, relative as pathRelative, dirname as pathDirname } from "path";
|
|
||||||
import { crawl } from "./tools/crawl";
|
|
||||||
import { downloadBuiltinKeycloakTheme } from "./download-builtin-keycloak-theme";
|
|
||||||
import { getProjectRoot } from "./tools/getProjectRoot";
|
|
||||||
import { rm_rf, rm_r } from "./tools/rm";
|
|
||||||
|
|
||||||
//NOTE: To run without argument when we want to generate src/i18n/generated_kcMessages files,
|
|
||||||
// update the version array for generating for newer version.
|
|
||||||
|
|
||||||
//@ts-ignore
|
|
||||||
const propertiesParser = require("properties-parser");
|
|
||||||
|
|
||||||
for (const keycloakVersion of ["11.0.3", "15.0.2", "18.0.1"]) {
|
|
||||||
console.log({ keycloakVersion });
|
|
||||||
|
|
||||||
const tmpDirPath = pathJoin(getProjectRoot(), "tmp_xImOef9dOd44");
|
|
||||||
|
|
||||||
rm_rf(tmpDirPath);
|
|
||||||
|
|
||||||
downloadBuiltinKeycloakTheme({
|
|
||||||
keycloakVersion,
|
|
||||||
"destDirPath": tmpDirPath,
|
|
||||||
});
|
|
||||||
|
|
||||||
type Dictionary = { [idiomId: string]: string };
|
|
||||||
|
|
||||||
const record: { [typeOfPage: string]: { [language: string]: Dictionary } } = {};
|
|
||||||
|
|
||||||
{
|
|
||||||
const baseThemeDirPath = pathJoin(tmpDirPath, "base");
|
|
||||||
|
|
||||||
crawl(baseThemeDirPath).forEach(filePath => {
|
|
||||||
const match = filePath.match(/^([^/]+)\/messages\/messages_([^.]+)\.properties$/);
|
|
||||||
|
|
||||||
if (match === null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const [, typeOfPage, language] = match;
|
|
||||||
|
|
||||||
(record[typeOfPage] ??= {})[language.replace(/_/g, "-")] = Object.fromEntries(
|
|
||||||
Object.entries(propertiesParser.parse(fs.readFileSync(pathJoin(baseThemeDirPath, filePath)).toString("utf8"))).map(
|
|
||||||
([key, value]: any) => [key, value.replace(/''/g, "'")],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
rm_r(tmpDirPath);
|
|
||||||
|
|
||||||
Object.keys(record).forEach(pageType => {
|
|
||||||
const recordForPageType = record[pageType];
|
|
||||||
|
|
||||||
Object.keys(recordForPageType).forEach(language => {
|
|
||||||
const filePath = pathJoin(getProjectRoot(), "src", "lib", "i18n", "generated_messages", keycloakVersion, pageType, `${language}.ts`);
|
|
||||||
|
|
||||||
fs.mkdirSync(pathDirname(filePath), { "recursive": true });
|
|
||||||
|
|
||||||
fs.writeFileSync(
|
|
||||||
filePath,
|
|
||||||
Buffer.from(
|
|
||||||
[
|
|
||||||
`//This code was automatically generated by running ${pathRelative(getProjectRoot(), __filename)}`,
|
|
||||||
"//PLEASE DO NOT EDIT MANUALLY",
|
|
||||||
"",
|
|
||||||
"/* spell-checker: disable */",
|
|
||||||
`const messages= ${JSON.stringify(recordForPageType[language], null, 2)};`,
|
|
||||||
"",
|
|
||||||
"export default messages;",
|
|
||||||
"/* spell-checker: enable */",
|
|
||||||
].join("\n"),
|
|
||||||
"utf8",
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
console.log(`${filePath} wrote`);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
58
src/bin/initialize-email-theme.ts
Normal file
58
src/bin/initialize-email-theme.ts
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
#!/usr/bin/env node
|
||||||
|
|
||||||
|
import { downloadBuiltinKeycloakTheme } from "./download-builtin-keycloak-theme";
|
||||||
|
import { join as pathJoin, relative as pathRelative } from "path";
|
||||||
|
import { transformCodebase } from "./tools/transformCodebase";
|
||||||
|
import { promptKeycloakVersion } from "./promptKeycloakVersion";
|
||||||
|
import * as fs from "fs";
|
||||||
|
import { getCliOptions } from "./tools/cliOptions";
|
||||||
|
import { getLogger } from "./tools/logger";
|
||||||
|
import { getEmailThemeSrcDirPath } from "./keycloakify/build-paths";
|
||||||
|
|
||||||
|
export async function main() {
|
||||||
|
const { isSilent } = getCliOptions(process.argv.slice(2));
|
||||||
|
const logger = getLogger({ isSilent });
|
||||||
|
|
||||||
|
const { emailThemeSrcDirPath } = getEmailThemeSrcDirPath();
|
||||||
|
|
||||||
|
if (emailThemeSrcDirPath === undefined) {
|
||||||
|
logger.warn("Couldn't locate your theme source directory");
|
||||||
|
|
||||||
|
process.exit(-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fs.existsSync(emailThemeSrcDirPath)) {
|
||||||
|
logger.warn(`There is already a ${pathRelative(process.cwd(), emailThemeSrcDirPath)} directory in your project. Aborting.`);
|
||||||
|
|
||||||
|
process.exit(-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
const { keycloakVersion } = await promptKeycloakVersion();
|
||||||
|
|
||||||
|
const builtinKeycloakThemeTmpDirPath = pathJoin(emailThemeSrcDirPath, "..", "tmp_xIdP3_builtin_keycloak_theme");
|
||||||
|
|
||||||
|
await downloadBuiltinKeycloakTheme({
|
||||||
|
keycloakVersion,
|
||||||
|
"destDirPath": builtinKeycloakThemeTmpDirPath,
|
||||||
|
isSilent
|
||||||
|
});
|
||||||
|
|
||||||
|
transformCodebase({
|
||||||
|
"srcDirPath": pathJoin(builtinKeycloakThemeTmpDirPath, "base", "email"),
|
||||||
|
"destDirPath": emailThemeSrcDirPath
|
||||||
|
});
|
||||||
|
|
||||||
|
{
|
||||||
|
const themePropertyFilePath = pathJoin(emailThemeSrcDirPath, "theme.properties");
|
||||||
|
|
||||||
|
fs.writeFileSync(themePropertyFilePath, Buffer.from(`parent=base\n${fs.readFileSync(themePropertyFilePath).toString("utf8")}`, "utf8"));
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.log(`${pathRelative(process.cwd(), emailThemeSrcDirPath)} ready to be customized, feel free to remove every file you do not customize`);
|
||||||
|
|
||||||
|
fs.rmSync(builtinKeycloakThemeTmpDirPath, { "recursive": true, "force": true });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (require.main === module) {
|
||||||
|
main();
|
||||||
|
}
|
183
src/bin/keycloakify/BuildOptions.ts
Normal file
183
src/bin/keycloakify/BuildOptions.ts
Normal file
@ -0,0 +1,183 @@
|
|||||||
|
import { assert } from "tsafe/assert";
|
||||||
|
import { id } from "tsafe/id";
|
||||||
|
import { parse as urlParse } from "url";
|
||||||
|
import { typeGuard } from "tsafe/typeGuard";
|
||||||
|
import { symToStr } from "tsafe/symToStr";
|
||||||
|
import { bundlers, getParsedPackageJson, type Bundler } from "./parsedPackageJson";
|
||||||
|
import { getAppInputPath, getKeycloakBuildPath } from "./build-paths";
|
||||||
|
|
||||||
|
/** Consolidated build option gathered form CLI arguments and config in package.json */
|
||||||
|
export type BuildOptions = BuildOptions.Standalone | BuildOptions.ExternalAssets;
|
||||||
|
|
||||||
|
export namespace BuildOptions {
|
||||||
|
export type Common = {
|
||||||
|
isSilent: boolean;
|
||||||
|
version: string;
|
||||||
|
themeName: string;
|
||||||
|
extraLoginPages: string[] | undefined;
|
||||||
|
extraAccountPages: string[] | undefined;
|
||||||
|
extraThemeProperties?: string[];
|
||||||
|
groupId: string;
|
||||||
|
artifactId: string;
|
||||||
|
bundler: Bundler;
|
||||||
|
keycloakVersionDefaultAssets: string;
|
||||||
|
// Directory of your built react project. Defaults to {cwd}/build
|
||||||
|
appInputPath: string;
|
||||||
|
// Directory that keycloakify outputs to. Defaults to {cwd}/build_keycloak
|
||||||
|
keycloakBuildPath: string;
|
||||||
|
customUserAttributes: string[];
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Standalone = Common & {
|
||||||
|
isStandalone: true;
|
||||||
|
urlPathname: string | undefined;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type ExternalAssets = ExternalAssets.SameDomain | ExternalAssets.DifferentDomains;
|
||||||
|
|
||||||
|
export namespace ExternalAssets {
|
||||||
|
export type CommonExternalAssets = Common & {
|
||||||
|
isStandalone: false;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type SameDomain = CommonExternalAssets & {
|
||||||
|
areAppAndKeycloakServerSharingSameDomain: true;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type DifferentDomains = CommonExternalAssets & {
|
||||||
|
areAppAndKeycloakServerSharingSameDomain: false;
|
||||||
|
urlOrigin: string;
|
||||||
|
urlPathname: string | undefined;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function readBuildOptions(params: { CNAME: string | undefined; isExternalAssetsCliParamProvided: boolean; isSilent: boolean }): BuildOptions {
|
||||||
|
const { CNAME, isExternalAssetsCliParamProvided, isSilent } = params;
|
||||||
|
|
||||||
|
const parsedPackageJson = getParsedPackageJson();
|
||||||
|
|
||||||
|
const url = (() => {
|
||||||
|
const { homepage } = parsedPackageJson;
|
||||||
|
|
||||||
|
let url: URL | undefined = undefined;
|
||||||
|
|
||||||
|
if (homepage !== undefined) {
|
||||||
|
url = new URL(homepage);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (CNAME !== undefined) {
|
||||||
|
url = new URL(`https://${CNAME.replace(/\s+$/, "")}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (url === undefined) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
"origin": url.origin,
|
||||||
|
"pathname": (() => {
|
||||||
|
const out = url.pathname.replace(/([^/])$/, "$1/");
|
||||||
|
|
||||||
|
return out === "/" ? undefined : out;
|
||||||
|
})()
|
||||||
|
};
|
||||||
|
})();
|
||||||
|
|
||||||
|
const common: BuildOptions.Common = (() => {
|
||||||
|
const { name, keycloakify = {}, version, homepage } = parsedPackageJson;
|
||||||
|
|
||||||
|
const { extraPages, extraLoginPages, extraAccountPages, extraThemeProperties, groupId, artifactId, bundler, keycloakVersionDefaultAssets } =
|
||||||
|
keycloakify ?? {};
|
||||||
|
|
||||||
|
const themeName =
|
||||||
|
keycloakify.themeName ??
|
||||||
|
name
|
||||||
|
.replace(/^@(.*)/, "$1")
|
||||||
|
.split("/")
|
||||||
|
.join("-");
|
||||||
|
|
||||||
|
return {
|
||||||
|
themeName,
|
||||||
|
"bundler": (() => {
|
||||||
|
const { KEYCLOAKIFY_BUNDLER } = process.env;
|
||||||
|
|
||||||
|
assert(
|
||||||
|
typeGuard<Bundler | undefined>(
|
||||||
|
KEYCLOAKIFY_BUNDLER,
|
||||||
|
[undefined, ...id<readonly string[]>(bundlers)].includes(KEYCLOAKIFY_BUNDLER)
|
||||||
|
),
|
||||||
|
`${symToStr({ KEYCLOAKIFY_BUNDLER })} should be one of ${bundlers.join(", ")}`
|
||||||
|
);
|
||||||
|
|
||||||
|
return KEYCLOAKIFY_BUNDLER ?? bundler ?? "keycloakify";
|
||||||
|
})(),
|
||||||
|
"artifactId": process.env.KEYCLOAKIFY_ARTIFACT_ID ?? artifactId ?? `${themeName}-keycloak-theme`,
|
||||||
|
"groupId": (() => {
|
||||||
|
const fallbackGroupId = `${themeName}.keycloak`;
|
||||||
|
|
||||||
|
return (
|
||||||
|
process.env.KEYCLOAKIFY_GROUP_ID ??
|
||||||
|
groupId ??
|
||||||
|
(!homepage
|
||||||
|
? fallbackGroupId
|
||||||
|
: urlParse(homepage)
|
||||||
|
.host?.replace(/:[0-9]+$/, "")
|
||||||
|
?.split(".")
|
||||||
|
.reverse()
|
||||||
|
.join(".") ?? fallbackGroupId) + ".keycloak"
|
||||||
|
);
|
||||||
|
})(),
|
||||||
|
"version": process.env.KEYCLOAKIFY_VERSION ?? version,
|
||||||
|
"extraLoginPages": [...(extraPages ?? []), ...(extraLoginPages ?? [])],
|
||||||
|
extraAccountPages,
|
||||||
|
extraThemeProperties,
|
||||||
|
isSilent,
|
||||||
|
"keycloakVersionDefaultAssets": keycloakVersionDefaultAssets ?? "11.0.3",
|
||||||
|
appInputPath: getAppInputPath(),
|
||||||
|
keycloakBuildPath: getKeycloakBuildPath(),
|
||||||
|
"customUserAttributes": keycloakify.customUserAttributes ?? []
|
||||||
|
};
|
||||||
|
})();
|
||||||
|
|
||||||
|
if (isExternalAssetsCliParamProvided) {
|
||||||
|
const commonExternalAssets = id<BuildOptions.ExternalAssets.CommonExternalAssets>({
|
||||||
|
...common,
|
||||||
|
"isStandalone": false
|
||||||
|
});
|
||||||
|
|
||||||
|
if (parsedPackageJson.keycloakify?.areAppAndKeycloakServerSharingSameDomain) {
|
||||||
|
return id<BuildOptions.ExternalAssets.SameDomain>({
|
||||||
|
...commonExternalAssets,
|
||||||
|
"areAppAndKeycloakServerSharingSameDomain": true
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
assert(
|
||||||
|
url !== undefined,
|
||||||
|
[
|
||||||
|
"Can't compile in external assets mode if we don't know where",
|
||||||
|
"the app will be hosted.",
|
||||||
|
"You should provide a homepage field in the package.json (or create a",
|
||||||
|
"public/CNAME file.",
|
||||||
|
"Alternatively, if your app and the Keycloak server are on the same domain, ",
|
||||||
|
"eg https://example.com is your app and https://example.com/auth is the keycloak",
|
||||||
|
'admin UI, you can set "keycloakify": { "areAppAndKeycloakServerSharingSameDomain": true }',
|
||||||
|
"in your package.json"
|
||||||
|
].join(" ")
|
||||||
|
);
|
||||||
|
|
||||||
|
return id<BuildOptions.ExternalAssets.DifferentDomains>({
|
||||||
|
...commonExternalAssets,
|
||||||
|
"areAppAndKeycloakServerSharingSameDomain": false,
|
||||||
|
"urlOrigin": url.origin,
|
||||||
|
"urlPathname": url.pathname
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return id<BuildOptions.Standalone>({
|
||||||
|
...common,
|
||||||
|
"isStandalone": true,
|
||||||
|
"urlPathname": url?.pathname
|
||||||
|
});
|
||||||
|
}
|
72
src/bin/keycloakify/build-paths.ts
Normal file
72
src/bin/keycloakify/build-paths.ts
Normal file
@ -0,0 +1,72 @@
|
|||||||
|
import * as fs from "fs";
|
||||||
|
import { exclude } from "tsafe";
|
||||||
|
import { crawl } from "../tools/crawl";
|
||||||
|
import { pathJoin } from "../tools/pathJoin";
|
||||||
|
import { getParsedPackageJson } from "./parsedPackageJson";
|
||||||
|
|
||||||
|
const DEFAULT_APP_INPUT_PATH = "build";
|
||||||
|
|
||||||
|
const DEFAULT_KEYCLOAK_BUILD_PATH = "build_keycloak";
|
||||||
|
|
||||||
|
const THEME_SRC_DIR_BASENAME = "keycloak-theme";
|
||||||
|
|
||||||
|
export const getReactProjectDirPath = () => process.cwd();
|
||||||
|
|
||||||
|
export const getCnamePath = () => pathJoin(getReactProjectDirPath(), "public", "CNAME");
|
||||||
|
|
||||||
|
const parseAppInputPath = (path?: string) => {
|
||||||
|
if (!path) {
|
||||||
|
return pathJoin(process.cwd(), DEFAULT_APP_INPUT_PATH);
|
||||||
|
} else if (path.startsWith("./")) {
|
||||||
|
return pathJoin(process.cwd(), path.replace("./", ""));
|
||||||
|
}
|
||||||
|
return path;
|
||||||
|
};
|
||||||
|
|
||||||
|
const parseKeycloakBuildPath = (path?: string) => {
|
||||||
|
if (!path) {
|
||||||
|
return pathJoin(process.cwd(), DEFAULT_KEYCLOAK_BUILD_PATH);
|
||||||
|
} else if (path.startsWith("./")) {
|
||||||
|
return pathJoin(process.cwd(), path.replace("./", ""));
|
||||||
|
}
|
||||||
|
return path;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getAppInputPath = () => {
|
||||||
|
return parseAppInputPath(getParsedPackageJson().keycloakify?.appInputPath);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getKeycloakBuildPath = () => {
|
||||||
|
return parseKeycloakBuildPath(getParsedPackageJson().keycloakify?.keycloakBuildPath);
|
||||||
|
};
|
||||||
|
export const getThemeSrcDirPath = () => {
|
||||||
|
const srcDirPath = pathJoin(getReactProjectDirPath(), "src");
|
||||||
|
|
||||||
|
const themeSrcDirPath: string | undefined = crawl(srcDirPath)
|
||||||
|
.map(fileRelativePath => {
|
||||||
|
const split = fileRelativePath.split(THEME_SRC_DIR_BASENAME);
|
||||||
|
|
||||||
|
if (split.length !== 2) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
return pathJoin(srcDirPath, split[0] + THEME_SRC_DIR_BASENAME);
|
||||||
|
})
|
||||||
|
.filter(exclude(undefined))[0];
|
||||||
|
if (themeSrcDirPath === undefined) {
|
||||||
|
if (fs.existsSync(pathJoin(srcDirPath, "login")) || fs.existsSync(pathJoin(srcDirPath, "account"))) {
|
||||||
|
return { "themeSrcDirPath": srcDirPath };
|
||||||
|
}
|
||||||
|
return { "themeSrcDirPath": undefined };
|
||||||
|
}
|
||||||
|
|
||||||
|
return { themeSrcDirPath };
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getEmailThemeSrcDirPath = () => {
|
||||||
|
const { themeSrcDirPath } = getThemeSrcDirPath();
|
||||||
|
|
||||||
|
const emailThemeSrcDirPath = themeSrcDirPath === undefined ? undefined : pathJoin(themeSrcDirPath, "email");
|
||||||
|
|
||||||
|
return { emailThemeSrcDirPath };
|
||||||
|
};
|
@ -2,8 +2,7 @@
|
|||||||
<#assign pageId="PAGE_ID_xIgLsPgGId9D8e">
|
<#assign pageId="PAGE_ID_xIgLsPgGId9D8e">
|
||||||
(()=>{
|
(()=>{
|
||||||
|
|
||||||
const out =
|
const out = ${ftl_object_to_js_code_declaring_an_object(.data_model, [])?no_esc};
|
||||||
${ftl_object_to_js_code_declaring_an_object(.data_model, [])?no_esc};
|
|
||||||
|
|
||||||
out["msg"]= function(){ throw new Error("use import { useKcMessage } from 'keycloakify'"); };
|
out["msg"]= function(){ throw new Error("use import { useKcMessage } from 'keycloakify'"); };
|
||||||
out["advancedMsg"]= function(){ throw new Error("use import { useKcMessage } from 'keycloakify'"); };
|
out["advancedMsg"]= function(){ throw new Error("use import { useKcMessage } from 'keycloakify'"); };
|
||||||
@ -14,7 +13,7 @@ ${ftl_object_to_js_code_declaring_an_object(.data_model, [])?no_esc};
|
|||||||
"totp", "totpSecret", "SAMLRequest", "SAMLResponse", "relayState", "device_user_code", "code",
|
"totp", "totpSecret", "SAMLRequest", "SAMLResponse", "relayState", "device_user_code", "code",
|
||||||
"password-new", "rememberMe", "login", "authenticationExecution", "cancel-aia", "clientDataJSON",
|
"password-new", "rememberMe", "login", "authenticationExecution", "cancel-aia", "clientDataJSON",
|
||||||
"authenticatorData", "signature", "credentialId", "userHandle", "error", "authn_use_chk", "authenticationExecution",
|
"authenticatorData", "signature", "credentialId", "userHandle", "error", "authn_use_chk", "authenticationExecution",
|
||||||
"isSetRetry", "try-again", "attestationObject", "publicKeyCredentialId", "authenticatorLabel"
|
"isSetRetry", "try-again", "attestationObject", "publicKeyCredentialId", "authenticatorLabel"CUSTOM_USER_ATTRIBUTES_eKsIY4ZsZ4xeM
|
||||||
]>
|
]>
|
||||||
|
|
||||||
<#attempt>
|
<#attempt>
|
||||||
@ -32,63 +31,94 @@ ${ftl_object_to_js_code_declaring_an_object(.data_model, [])?no_esc};
|
|||||||
"printIfExists": function (fieldName, x) {
|
"printIfExists": function (fieldName, x) {
|
||||||
<#if !messagesPerField?? >
|
<#if !messagesPerField?? >
|
||||||
return undefined;
|
return undefined;
|
||||||
|
<#else>
|
||||||
|
<#list fieldNames as fieldName>
|
||||||
|
if(fieldName === "${fieldName}" ){
|
||||||
|
<#attempt>
|
||||||
|
<#if '${fieldName}' == 'username' || '${fieldName}' == 'password'>
|
||||||
|
return <#if messagesPerField.existsError('username', 'password')>x<#else>undefined</#if>;
|
||||||
|
<#else>
|
||||||
|
return <#if messagesPerField.existsError('${fieldName}')>x<#else>undefined</#if>;
|
||||||
|
</#if>
|
||||||
|
<#recover>
|
||||||
|
</#attempt>
|
||||||
|
}
|
||||||
|
</#list>
|
||||||
|
throw new Error("There is no " + fieldName + " field");
|
||||||
</#if>
|
</#if>
|
||||||
<#list fieldNames as fieldName>
|
|
||||||
if(fieldName === "${fieldName}" ){
|
|
||||||
<#attempt>
|
|
||||||
return "${messagesPerField.printIfExists(fieldName,'1')}" ? x : undefined;
|
|
||||||
<#recover>
|
|
||||||
</#attempt>
|
|
||||||
}
|
|
||||||
</#list>
|
|
||||||
throw new Error("There is no " + fieldName + " field");
|
|
||||||
},
|
},
|
||||||
"existsError": function (fieldName) {
|
"existsError": function (fieldName) {
|
||||||
<#if !messagesPerField?? >
|
<#if !messagesPerField?? >
|
||||||
return false;
|
return false;
|
||||||
|
<#else>
|
||||||
|
<#list fieldNames as fieldName>
|
||||||
|
if(fieldName === "${fieldName}" ){
|
||||||
|
<#attempt>
|
||||||
|
<#if '${fieldName}' == 'username' || '${fieldName}' == 'password'>
|
||||||
|
return <#if messagesPerField.existsError('username', 'password')>true<#else>false</#if>;
|
||||||
|
<#else>
|
||||||
|
return <#if messagesPerField.existsError('${fieldName}')>true<#else>false</#if>;
|
||||||
|
</#if>
|
||||||
|
<#recover>
|
||||||
|
</#attempt>
|
||||||
|
}
|
||||||
|
</#list>
|
||||||
|
throw new Error("There is no " + fieldName + " field");
|
||||||
</#if>
|
</#if>
|
||||||
<#list fieldNames as fieldName>
|
|
||||||
if(fieldName === "${fieldName}" ){
|
|
||||||
<#attempt>
|
|
||||||
return <#if messagesPerField.existsError('${fieldName}')>true<#else>false</#if>;
|
|
||||||
<#recover>
|
|
||||||
</#attempt>
|
|
||||||
}
|
|
||||||
</#list>
|
|
||||||
throw new Error("There is no " + fieldName + " field");
|
|
||||||
},
|
},
|
||||||
"get": function (fieldName) {
|
"get": function (fieldName) {
|
||||||
<#if !messagesPerField?? >
|
<#if !messagesPerField?? >
|
||||||
return '';
|
return '';
|
||||||
|
<#else>
|
||||||
|
<#list fieldNames as fieldName>
|
||||||
|
if(fieldName === "${fieldName}" ){
|
||||||
|
<#attempt>
|
||||||
|
<#if '${fieldName}' == 'username' || '${fieldName}' == 'password'>
|
||||||
|
<#if messagesPerField.existsError('username', 'password')>
|
||||||
|
return 'Invalid username or password.';
|
||||||
|
</#if>
|
||||||
|
<#else>
|
||||||
|
<#if messagesPerField.existsError('${fieldName}')>
|
||||||
|
return "${messagesPerField.get('${fieldName}')?no_esc}";
|
||||||
|
</#if>
|
||||||
|
</#if>
|
||||||
|
<#recover>
|
||||||
|
</#attempt>
|
||||||
|
}
|
||||||
|
</#list>
|
||||||
|
throw new Error("There is no " + fieldName + " field");
|
||||||
</#if>
|
</#if>
|
||||||
<#list fieldNames as fieldName>
|
|
||||||
if(fieldName === "${fieldName}" ){
|
|
||||||
<#attempt>
|
|
||||||
<#if messagesPerField.existsError('${fieldName}')>
|
|
||||||
return "${messagesPerField.get('${fieldName}')?no_esc}";
|
|
||||||
</#if>
|
|
||||||
<#recover>
|
|
||||||
</#attempt>
|
|
||||||
}
|
|
||||||
</#list>
|
|
||||||
throw new Error("There is no " + fieldName + " field");
|
|
||||||
},
|
},
|
||||||
"exists": function (fieldName) {
|
"exists": function (fieldName) {
|
||||||
<#if !messagesPerField?? >
|
<#if !messagesPerField?? >
|
||||||
return false;
|
return false;
|
||||||
|
<#else>
|
||||||
|
<#list fieldNames as fieldName>
|
||||||
|
if(fieldName === "${fieldName}" ){
|
||||||
|
<#attempt>
|
||||||
|
<#if '${fieldName}' == 'username' || '${fieldName}' == 'password'>
|
||||||
|
return <#if messagesPerField.exists('username') || messagesPerField.exists('password')>true<#else>false</#if>;
|
||||||
|
<#else>
|
||||||
|
return <#if messagesPerField.exists('${fieldName}')>true<#else>false</#if>;
|
||||||
|
</#if>
|
||||||
|
<#recover>
|
||||||
|
</#attempt>
|
||||||
|
}
|
||||||
|
</#list>
|
||||||
|
throw new Error("There is no " + fieldName + " field");
|
||||||
</#if>
|
</#if>
|
||||||
<#list fieldNames as fieldName>
|
|
||||||
if(fieldName === "${fieldName}" ){
|
|
||||||
<#attempt>
|
|
||||||
return <#if messagesPerField.exists('${fieldName}')>true<#else>false</#if>;
|
|
||||||
<#recover>
|
|
||||||
</#attempt>
|
|
||||||
}
|
|
||||||
</#list>
|
|
||||||
throw new Error("There is no " + fieldName + " field");
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
<#if account??>
|
||||||
|
out["url"]["getLogoutUrl"] = function () {
|
||||||
|
<#attempt>
|
||||||
|
return "${url.getLogoutUrl()}";
|
||||||
|
<#recover>
|
||||||
|
</#attempt>
|
||||||
|
};
|
||||||
|
</#if>
|
||||||
|
|
||||||
out["pageId"] = "${pageId}";
|
out["pageId"] = "${pageId}";
|
||||||
|
|
||||||
return out;
|
return out;
|
||||||
@ -134,9 +164,9 @@ ${ftl_object_to_js_code_declaring_an_object(.data_model, [])?no_esc};
|
|||||||
key == "updateProfileCtx" &&
|
key == "updateProfileCtx" &&
|
||||||
are_same_path(path, [])
|
are_same_path(path, [])
|
||||||
) || (
|
) || (
|
||||||
<#-- https://github.com/InseeFrLab/keycloakify/pull/65#issuecomment-991896344 (reports with saml-post-form.ftl) -->
|
<#-- https://github.com/keycloakify/keycloakify/pull/65#issuecomment-991896344 (reports with saml-post-form.ftl) -->
|
||||||
<#-- https://github.com/InseeFrLab/keycloakify/issues/91#issue-1212319466 (reports with error.ftl and Kc18) -->
|
<#-- https://github.com/keycloakify/keycloakify/issues/91#issue-1212319466 (reports with error.ftl and Kc18) -->
|
||||||
<#-- https://github.com/InseeFrLab/keycloakify/issues/109#issuecomment-1134610163 -->
|
<#-- https://github.com/keycloakify/keycloakify/issues/109#issuecomment-1134610163 -->
|
||||||
key == "loginAction" &&
|
key == "loginAction" &&
|
||||||
are_same_path(path, ["url"]) &&
|
are_same_path(path, ["url"]) &&
|
||||||
["saml-post-form.ftl", "error.ftl", "info.ftl"]?seq_contains(pageId) &&
|
["saml-post-form.ftl", "error.ftl", "info.ftl"]?seq_contains(pageId) &&
|
||||||
@ -152,6 +182,10 @@ ${ftl_object_to_js_code_declaring_an_object(.data_model, [])?no_esc};
|
|||||||
) || (
|
) || (
|
||||||
["masterAdminClient", "delegateForUpdate", "defaultRole"]?seq_contains(key) &&
|
["masterAdminClient", "delegateForUpdate", "defaultRole"]?seq_contains(key) &&
|
||||||
are_same_path(path, ["realm"])
|
are_same_path(path, ["realm"])
|
||||||
|
) || (
|
||||||
|
"error.ftl" == pageId &&
|
||||||
|
are_same_path(path, ["realm"]) &&
|
||||||
|
!["name", "displayName", "displayNameHtml", "internationalizationEnabled", "registrationEmailAsUsername" ]?seq_contains(key)
|
||||||
)
|
)
|
||||||
>
|
>
|
||||||
<#local out_seq += ["/*If you need '" + key + "' on " + pageId + ", please submit an issue to the Keycloakify repo*/"]>
|
<#local out_seq += ["/*If you need '" + key + "' on " + pageId + ", please submit an issue to the Keycloakify repo*/"]>
|
||||||
@ -272,6 +306,11 @@ ${ftl_object_to_js_code_declaring_an_object(.data_model, [])?no_esc};
|
|||||||
|
|
||||||
<#list object as array_item>
|
<#list object as array_item>
|
||||||
|
|
||||||
|
<#if !array_item??>
|
||||||
|
<#local out_seq += ["null,"]>
|
||||||
|
<#continue>
|
||||||
|
</#if>
|
||||||
|
|
||||||
<#local rec_out = ftl_object_to_js_code_declaring_an_object(array_item, path + [ i ])>
|
<#local rec_out = ftl_object_to_js_code_declaring_an_object(array_item, path + [ i ])>
|
||||||
|
|
||||||
<#local i = i + 1>
|
<#local i = i + 1>
|
203
src/bin/keycloakify/generateFtl/generateFtl.ts
Normal file
203
src/bin/keycloakify/generateFtl/generateFtl.ts
Normal file
@ -0,0 +1,203 @@
|
|||||||
|
import cheerio from "cheerio";
|
||||||
|
import { replaceImportsFromStaticInJsCode } from "../replacers/replaceImportsFromStaticInJsCode";
|
||||||
|
import { generateCssCodeToDefineGlobals } from "../replacers/replaceImportsInCssCode";
|
||||||
|
import { replaceImportsInInlineCssCode } from "../replacers/replaceImportsInInlineCssCode";
|
||||||
|
import * as fs from "fs";
|
||||||
|
import { join as pathJoin } from "path";
|
||||||
|
import { objectKeys } from "tsafe/objectKeys";
|
||||||
|
import { ftlValuesGlobalName } from "../ftlValuesGlobalName";
|
||||||
|
import type { BuildOptions } from "../BuildOptions";
|
||||||
|
import { assert } from "tsafe/assert";
|
||||||
|
|
||||||
|
export const themeTypes = ["login", "account"] as const;
|
||||||
|
|
||||||
|
export type ThemeType = (typeof themeTypes)[number];
|
||||||
|
|
||||||
|
export const loginThemePageIds = [
|
||||||
|
"login.ftl",
|
||||||
|
"login-username.ftl",
|
||||||
|
"login-password.ftl",
|
||||||
|
"webauthn-authenticate.ftl",
|
||||||
|
"register.ftl",
|
||||||
|
"register-user-profile.ftl",
|
||||||
|
"info.ftl",
|
||||||
|
"error.ftl",
|
||||||
|
"login-reset-password.ftl",
|
||||||
|
"login-verify-email.ftl",
|
||||||
|
"terms.ftl",
|
||||||
|
"login-otp.ftl",
|
||||||
|
"login-update-profile.ftl",
|
||||||
|
"login-update-password.ftl",
|
||||||
|
"login-idp-link-confirm.ftl",
|
||||||
|
"login-idp-link-email.ftl",
|
||||||
|
"login-page-expired.ftl",
|
||||||
|
"login-config-totp.ftl",
|
||||||
|
"logout-confirm.ftl",
|
||||||
|
"update-user-profile.ftl",
|
||||||
|
"idp-review-user-profile.ftl",
|
||||||
|
"update-email.ftl",
|
||||||
|
"select-authenticator.ftl"
|
||||||
|
] as const;
|
||||||
|
|
||||||
|
export const accountThemePageIds = ["password.ftl", "account.ftl"] as const;
|
||||||
|
|
||||||
|
export type LoginThemePageId = (typeof loginThemePageIds)[number];
|
||||||
|
export type AccountThemePageId = (typeof accountThemePageIds)[number];
|
||||||
|
|
||||||
|
export type BuildOptionsLike = BuildOptionsLike.Standalone | BuildOptionsLike.ExternalAssets;
|
||||||
|
|
||||||
|
export namespace BuildOptionsLike {
|
||||||
|
export type Common = {
|
||||||
|
customUserAttributes: string[];
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Standalone = Common & {
|
||||||
|
isStandalone: true;
|
||||||
|
urlPathname: string | undefined;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type ExternalAssets = ExternalAssets.SameDomain | ExternalAssets.DifferentDomains;
|
||||||
|
|
||||||
|
export namespace ExternalAssets {
|
||||||
|
export type CommonExternalAssets = {
|
||||||
|
isStandalone: false;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type SameDomain = Common &
|
||||||
|
CommonExternalAssets & {
|
||||||
|
areAppAndKeycloakServerSharingSameDomain: true;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type DifferentDomains = Common &
|
||||||
|
CommonExternalAssets & {
|
||||||
|
areAppAndKeycloakServerSharingSameDomain: false;
|
||||||
|
urlOrigin: string;
|
||||||
|
urlPathname: string | undefined;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assert<BuildOptions extends BuildOptionsLike ? true : false>();
|
||||||
|
|
||||||
|
export function generateFtlFilesCodeFactory(params: {
|
||||||
|
indexHtmlCode: string;
|
||||||
|
//NOTE: Expected to be an empty object if external assets mode is enabled.
|
||||||
|
cssGlobalsToDefine: Record<string, string>;
|
||||||
|
buildOptions: BuildOptionsLike;
|
||||||
|
}) {
|
||||||
|
const { cssGlobalsToDefine, indexHtmlCode, buildOptions } = params;
|
||||||
|
|
||||||
|
const $ = cheerio.load(indexHtmlCode);
|
||||||
|
|
||||||
|
fix_imports_statements: {
|
||||||
|
if (!buildOptions.isStandalone && buildOptions.areAppAndKeycloakServerSharingSameDomain) {
|
||||||
|
break fix_imports_statements;
|
||||||
|
}
|
||||||
|
|
||||||
|
$("script:not([src])").each((...[, element]) => {
|
||||||
|
const { fixedJsCode } = replaceImportsFromStaticInJsCode({
|
||||||
|
"jsCode": $(element).html()!,
|
||||||
|
buildOptions
|
||||||
|
});
|
||||||
|
|
||||||
|
$(element).text(fixedJsCode);
|
||||||
|
});
|
||||||
|
|
||||||
|
$("style").each((...[, element]) => {
|
||||||
|
const { fixedCssCode } = replaceImportsInInlineCssCode({
|
||||||
|
"cssCode": $(element).html()!,
|
||||||
|
buildOptions
|
||||||
|
});
|
||||||
|
|
||||||
|
$(element).text(fixedCssCode);
|
||||||
|
});
|
||||||
|
|
||||||
|
(
|
||||||
|
[
|
||||||
|
["link", "href"],
|
||||||
|
["script", "src"]
|
||||||
|
] as const
|
||||||
|
).forEach(([selector, attrName]) =>
|
||||||
|
$(selector).each((...[, element]) => {
|
||||||
|
const href = $(element).attr(attrName);
|
||||||
|
|
||||||
|
if (href === undefined) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$(element).attr(
|
||||||
|
attrName,
|
||||||
|
buildOptions.isStandalone
|
||||||
|
? href.replace(new RegExp(`^${(buildOptions.urlPathname ?? "/").replace(/\//g, "\\/")}`), "${url.resourcesPath}/build/")
|
||||||
|
: href.replace(/^\//, `${buildOptions.urlOrigin}/`)
|
||||||
|
);
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
if (Object.keys(cssGlobalsToDefine).length !== 0) {
|
||||||
|
$("head").prepend(
|
||||||
|
[
|
||||||
|
"",
|
||||||
|
"<style>",
|
||||||
|
generateCssCodeToDefineGlobals({
|
||||||
|
cssGlobalsToDefine,
|
||||||
|
buildOptions
|
||||||
|
}).cssCodeToPrependInHead,
|
||||||
|
"</style>",
|
||||||
|
""
|
||||||
|
].join("\n")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//FTL is no valid html, we can't insert with cheerio, we put placeholder for injecting later.
|
||||||
|
const replaceValueBySearchValue = {
|
||||||
|
'{ "x": "vIdLqMeOed9sdLdIdOxdK0d" }': fs
|
||||||
|
.readFileSync(pathJoin(__dirname, "ftl_object_to_js_code_declaring_an_object.ftl"))
|
||||||
|
.toString("utf8")
|
||||||
|
.match(/^<script>const _=((?:.|\n)+)<\/script>[\n]?$/)![1]
|
||||||
|
.replace(
|
||||||
|
"CUSTOM_USER_ATTRIBUTES_eKsIY4ZsZ4xeM",
|
||||||
|
buildOptions.customUserAttributes.length === 0 ? "" : ", " + buildOptions.customUserAttributes.map(name => `"${name}"`).join(", ")
|
||||||
|
),
|
||||||
|
"<!-- xIdLqMeOedErIdLsPdNdI9dSlxI -->": [
|
||||||
|
"<#if scripts??>",
|
||||||
|
" <#list scripts as script>",
|
||||||
|
' <script src="${script}" type="text/javascript"></script>',
|
||||||
|
" </#list>",
|
||||||
|
"</#if>"
|
||||||
|
].join("\n")
|
||||||
|
};
|
||||||
|
|
||||||
|
$("head").prepend(
|
||||||
|
[
|
||||||
|
"<script>",
|
||||||
|
` window.${ftlValuesGlobalName}= ${objectKeys(replaceValueBySearchValue)[0]};`,
|
||||||
|
"</script>",
|
||||||
|
"",
|
||||||
|
objectKeys(replaceValueBySearchValue)[1]
|
||||||
|
].join("\n")
|
||||||
|
);
|
||||||
|
|
||||||
|
const partiallyFixedIndexHtmlCode = $.html();
|
||||||
|
|
||||||
|
function generateFtlFilesCode(params: { pageId: string }): {
|
||||||
|
ftlCode: string;
|
||||||
|
} {
|
||||||
|
const { pageId } = params;
|
||||||
|
|
||||||
|
const $ = cheerio.load(partiallyFixedIndexHtmlCode);
|
||||||
|
|
||||||
|
let ftlCode = $.html();
|
||||||
|
|
||||||
|
Object.entries({
|
||||||
|
...replaceValueBySearchValue,
|
||||||
|
//If updated, don't forget to change in the ftl script as well.
|
||||||
|
"PAGE_ID_xIgLsPgGId9D8e": pageId
|
||||||
|
}).map(([searchValue, replaceValue]) => (ftlCode = ftlCode.replace(searchValue, replaceValue)));
|
||||||
|
|
||||||
|
return { ftlCode };
|
||||||
|
}
|
||||||
|
|
||||||
|
return { generateFtlFilesCode };
|
||||||
|
}
|
@ -1,39 +1,40 @@
|
|||||||
import * as url from "url";
|
|
||||||
import * as fs from "fs";
|
import * as fs from "fs";
|
||||||
import { join as pathJoin, dirname as pathDirname } from "path";
|
import { join as pathJoin, dirname as pathDirname } from "path";
|
||||||
|
import { themeTypes } from "./generateFtl/generateFtl";
|
||||||
|
import { assert } from "tsafe/assert";
|
||||||
|
import { Reflect } from "tsafe/Reflect";
|
||||||
|
import type { BuildOptions } from "./BuildOptions";
|
||||||
|
|
||||||
|
export type BuildOptionsLike = {
|
||||||
|
themeName: string;
|
||||||
|
groupId: string;
|
||||||
|
artifactId?: string;
|
||||||
|
version: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
{
|
||||||
|
const buildOptions = Reflect<BuildOptions>();
|
||||||
|
|
||||||
|
assert<typeof buildOptions extends BuildOptionsLike ? true : false>();
|
||||||
|
}
|
||||||
|
|
||||||
export function generateJavaStackFiles(params: {
|
export function generateJavaStackFiles(params: {
|
||||||
version: string;
|
|
||||||
themeName: string;
|
|
||||||
homepage?: string;
|
|
||||||
keycloakThemeBuildingDirPath: string;
|
keycloakThemeBuildingDirPath: string;
|
||||||
doBundleEmailTemplate: boolean;
|
doBundlesEmailTemplate: boolean;
|
||||||
|
buildOptions: BuildOptionsLike;
|
||||||
}): {
|
}): {
|
||||||
jarFilePath: string;
|
jarFilePath: string;
|
||||||
} {
|
} {
|
||||||
const { themeName, version, homepage, keycloakThemeBuildingDirPath, doBundleEmailTemplate } = params;
|
const {
|
||||||
|
buildOptions: { groupId, themeName, version, artifactId },
|
||||||
|
keycloakThemeBuildingDirPath,
|
||||||
|
doBundlesEmailTemplate
|
||||||
|
} = params;
|
||||||
|
|
||||||
{
|
{
|
||||||
const { pomFileCode } = (function generatePomFileCode(): {
|
const { pomFileCode } = (function generatePomFileCode(): {
|
||||||
pomFileCode: string;
|
pomFileCode: string;
|
||||||
} {
|
} {
|
||||||
const groupId = (() => {
|
|
||||||
const fallbackGroupId = `there.was.no.homepage.field.in.the.package.json.${themeName}`;
|
|
||||||
|
|
||||||
return (
|
|
||||||
(!homepage
|
|
||||||
? fallbackGroupId
|
|
||||||
: url
|
|
||||||
.parse(homepage)
|
|
||||||
.host?.replace(/:[0-9]+$/, "")
|
|
||||||
?.split(".")
|
|
||||||
.reverse()
|
|
||||||
.join(".") ?? fallbackGroupId) + ".keycloak"
|
|
||||||
);
|
|
||||||
})();
|
|
||||||
|
|
||||||
const artefactId = `${themeName}-keycloak-theme`;
|
|
||||||
|
|
||||||
const pomFileCode = [
|
const pomFileCode = [
|
||||||
`<?xml version="1.0"?>`,
|
`<?xml version="1.0"?>`,
|
||||||
`<project xmlns="http://maven.apache.org/POM/4.0.0"`,
|
`<project xmlns="http://maven.apache.org/POM/4.0.0"`,
|
||||||
@ -41,11 +42,11 @@ export function generateJavaStackFiles(params: {
|
|||||||
` xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">`,
|
` xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">`,
|
||||||
` <modelVersion>4.0.0</modelVersion>`,
|
` <modelVersion>4.0.0</modelVersion>`,
|
||||||
` <groupId>${groupId}</groupId>`,
|
` <groupId>${groupId}</groupId>`,
|
||||||
` <artifactId>${artefactId}</artifactId>`,
|
` <artifactId>${artifactId}</artifactId>`,
|
||||||
` <version>${version}</version>`,
|
` <version>${version}</version>`,
|
||||||
` <name>${artefactId}</name>`,
|
` <name>${artifactId}</name>`,
|
||||||
` <description />`,
|
` <description />`,
|
||||||
`</project>`,
|
`</project>`
|
||||||
].join("\n");
|
].join("\n");
|
||||||
|
|
||||||
return { pomFileCode };
|
return { pomFileCode };
|
||||||
@ -69,19 +70,19 @@ export function generateJavaStackFiles(params: {
|
|||||||
"themes": [
|
"themes": [
|
||||||
{
|
{
|
||||||
"name": themeName,
|
"name": themeName,
|
||||||
"types": ["login", ...(doBundleEmailTemplate ? ["email"] : [])],
|
"types": [...themeTypes, ...(doBundlesEmailTemplate ? ["email"] : [])]
|
||||||
},
|
}
|
||||||
],
|
]
|
||||||
},
|
},
|
||||||
null,
|
null,
|
||||||
2,
|
2
|
||||||
),
|
),
|
||||||
"utf8",
|
"utf8"
|
||||||
),
|
)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
"jarFilePath": pathJoin(keycloakThemeBuildingDirPath, "target", `${themeName}-${version}.jar`),
|
"jarFilePath": pathJoin(keycloakThemeBuildingDirPath, "target", `${artifactId}-${version}.jar`)
|
||||||
};
|
};
|
||||||
}
|
}
|
238
src/bin/keycloakify/generateKeycloakThemeResources.ts
Normal file
238
src/bin/keycloakify/generateKeycloakThemeResources.ts
Normal file
@ -0,0 +1,238 @@
|
|||||||
|
import { transformCodebase } from "../tools/transformCodebase";
|
||||||
|
import * as fs from "fs";
|
||||||
|
import { join as pathJoin, basename as pathBasename } from "path";
|
||||||
|
import { replaceImportsFromStaticInJsCode } from "./replacers/replaceImportsFromStaticInJsCode";
|
||||||
|
import { replaceImportsInCssCode } from "./replacers/replaceImportsInCssCode";
|
||||||
|
import { generateFtlFilesCodeFactory, loginThemePageIds, accountThemePageIds, themeTypes, type ThemeType } from "./generateFtl";
|
||||||
|
import { downloadBuiltinKeycloakTheme } from "../download-builtin-keycloak-theme";
|
||||||
|
import { mockTestingResourcesCommonPath, mockTestingResourcesPath, mockTestingSubDirOfPublicDirBasename } from "../mockTestingResourcesPath";
|
||||||
|
import { isInside } from "../tools/isInside";
|
||||||
|
import type { BuildOptions } from "./BuildOptions";
|
||||||
|
import { assert } from "tsafe/assert";
|
||||||
|
|
||||||
|
export type BuildOptionsLike = BuildOptionsLike.Standalone | BuildOptionsLike.ExternalAssets;
|
||||||
|
|
||||||
|
export namespace BuildOptionsLike {
|
||||||
|
export type Common = {
|
||||||
|
themeName: string;
|
||||||
|
extraLoginPages?: string[];
|
||||||
|
extraAccountPages?: string[];
|
||||||
|
extraThemeProperties?: string[];
|
||||||
|
isSilent: boolean;
|
||||||
|
customUserAttributes: string[];
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Standalone = Common & {
|
||||||
|
isStandalone: true;
|
||||||
|
urlPathname: string | undefined;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type ExternalAssets = ExternalAssets.SameDomain | ExternalAssets.DifferentDomains;
|
||||||
|
|
||||||
|
export namespace ExternalAssets {
|
||||||
|
export type CommonExternalAssets = Common & {
|
||||||
|
isStandalone: false;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type SameDomain = CommonExternalAssets & {
|
||||||
|
areAppAndKeycloakServerSharingSameDomain: true;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type DifferentDomains = CommonExternalAssets & {
|
||||||
|
areAppAndKeycloakServerSharingSameDomain: false;
|
||||||
|
urlOrigin: string;
|
||||||
|
urlPathname: string | undefined;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assert<BuildOptions extends BuildOptionsLike ? true : false>();
|
||||||
|
|
||||||
|
export async function generateKeycloakThemeResources(params: {
|
||||||
|
reactAppBuildDirPath: string;
|
||||||
|
keycloakThemeBuildingDirPath: string;
|
||||||
|
emailThemeSrcDirPath: string | undefined;
|
||||||
|
keycloakVersion: string;
|
||||||
|
buildOptions: BuildOptionsLike;
|
||||||
|
}): Promise<{ doBundlesEmailTemplate: boolean }> {
|
||||||
|
const { reactAppBuildDirPath, keycloakThemeBuildingDirPath, emailThemeSrcDirPath, keycloakVersion, buildOptions } = params;
|
||||||
|
|
||||||
|
const getThemeDirPath = (themeType: ThemeType | "email") =>
|
||||||
|
pathJoin(keycloakThemeBuildingDirPath, "src", "main", "resources", "theme", buildOptions.themeName, themeType);
|
||||||
|
|
||||||
|
let allCssGlobalsToDefine: Record<string, string> = {};
|
||||||
|
|
||||||
|
let generateFtlFilesCode_glob: ReturnType<typeof generateFtlFilesCodeFactory>["generateFtlFilesCode"] | undefined = undefined;
|
||||||
|
|
||||||
|
for (const themeType of themeTypes) {
|
||||||
|
const themeDirPath = getThemeDirPath(themeType);
|
||||||
|
|
||||||
|
copy_app_resources_to_theme_path: {
|
||||||
|
const isFirstPass = themeType.indexOf(themeType) === 0;
|
||||||
|
|
||||||
|
if (!isFirstPass && !buildOptions.isStandalone) {
|
||||||
|
break copy_app_resources_to_theme_path;
|
||||||
|
}
|
||||||
|
|
||||||
|
transformCodebase({
|
||||||
|
"destDirPath": buildOptions.isStandalone ? pathJoin(themeDirPath, "resources", "build") : reactAppBuildDirPath,
|
||||||
|
"srcDirPath": reactAppBuildDirPath,
|
||||||
|
"transformSourceCode": ({ filePath, sourceCode }) => {
|
||||||
|
//NOTE: Prevent cycles, excludes the folder we generated for debug in public/
|
||||||
|
if (
|
||||||
|
buildOptions.isStandalone &&
|
||||||
|
isInside({
|
||||||
|
"dirPath": pathJoin(reactAppBuildDirPath, mockTestingSubDirOfPublicDirBasename),
|
||||||
|
filePath
|
||||||
|
})
|
||||||
|
) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (/\.css?$/i.test(filePath)) {
|
||||||
|
if (!buildOptions.isStandalone) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { cssGlobalsToDefine, fixedCssCode } = replaceImportsInCssCode({
|
||||||
|
"cssCode": sourceCode.toString("utf8")
|
||||||
|
});
|
||||||
|
|
||||||
|
register_css_variables: {
|
||||||
|
if (!isFirstPass) {
|
||||||
|
break register_css_variables;
|
||||||
|
}
|
||||||
|
|
||||||
|
allCssGlobalsToDefine = {
|
||||||
|
...allCssGlobalsToDefine,
|
||||||
|
...cssGlobalsToDefine
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return { "modifiedSourceCode": Buffer.from(fixedCssCode, "utf8") };
|
||||||
|
}
|
||||||
|
|
||||||
|
if (/\.js?$/i.test(filePath)) {
|
||||||
|
if (!buildOptions.isStandalone && buildOptions.areAppAndKeycloakServerSharingSameDomain) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { fixedJsCode } = replaceImportsFromStaticInJsCode({
|
||||||
|
"jsCode": sourceCode.toString("utf8"),
|
||||||
|
buildOptions
|
||||||
|
});
|
||||||
|
|
||||||
|
return { "modifiedSourceCode": Buffer.from(fixedJsCode, "utf8") };
|
||||||
|
}
|
||||||
|
|
||||||
|
return buildOptions.isStandalone ? { "modifiedSourceCode": sourceCode } : undefined;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const generateFtlFilesCode = (() => {
|
||||||
|
if (generateFtlFilesCode_glob !== undefined) {
|
||||||
|
return generateFtlFilesCode_glob;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { generateFtlFilesCode } = generateFtlFilesCodeFactory({
|
||||||
|
"indexHtmlCode": fs.readFileSync(pathJoin(reactAppBuildDirPath, "index.html")).toString("utf8"),
|
||||||
|
"cssGlobalsToDefine": allCssGlobalsToDefine,
|
||||||
|
buildOptions
|
||||||
|
});
|
||||||
|
|
||||||
|
return generateFtlFilesCode;
|
||||||
|
})();
|
||||||
|
|
||||||
|
[
|
||||||
|
...(() => {
|
||||||
|
switch (themeType) {
|
||||||
|
case "login":
|
||||||
|
return loginThemePageIds;
|
||||||
|
case "account":
|
||||||
|
return accountThemePageIds;
|
||||||
|
}
|
||||||
|
})(),
|
||||||
|
...((() => {
|
||||||
|
switch (themeType) {
|
||||||
|
case "login":
|
||||||
|
return buildOptions.extraLoginPages;
|
||||||
|
case "account":
|
||||||
|
return buildOptions.extraAccountPages;
|
||||||
|
}
|
||||||
|
})() ?? [])
|
||||||
|
].forEach(pageId => {
|
||||||
|
const { ftlCode } = generateFtlFilesCode({ pageId });
|
||||||
|
|
||||||
|
fs.mkdirSync(themeDirPath, { "recursive": true });
|
||||||
|
|
||||||
|
fs.writeFileSync(pathJoin(themeDirPath, pageId), Buffer.from(ftlCode, "utf8"));
|
||||||
|
});
|
||||||
|
|
||||||
|
{
|
||||||
|
const tmpDirPath = pathJoin(themeDirPath, "..", "tmp_xxKdLpdIdLd");
|
||||||
|
|
||||||
|
await downloadBuiltinKeycloakTheme({
|
||||||
|
keycloakVersion,
|
||||||
|
"destDirPath": tmpDirPath,
|
||||||
|
isSilent: buildOptions.isSilent
|
||||||
|
});
|
||||||
|
|
||||||
|
const themeResourcesDirPath = pathJoin(themeDirPath, "resources");
|
||||||
|
|
||||||
|
transformCodebase({
|
||||||
|
"srcDirPath": pathJoin(tmpDirPath, "keycloak", "login", "resources"),
|
||||||
|
"destDirPath": themeResourcesDirPath
|
||||||
|
});
|
||||||
|
|
||||||
|
const reactAppPublicDirPath = pathJoin(reactAppBuildDirPath, "..", "public");
|
||||||
|
|
||||||
|
transformCodebase({
|
||||||
|
"srcDirPath": pathJoin(tmpDirPath, "keycloak", "common", "resources"),
|
||||||
|
"destDirPath": pathJoin(themeResourcesDirPath, pathBasename(mockTestingResourcesCommonPath))
|
||||||
|
});
|
||||||
|
|
||||||
|
transformCodebase({
|
||||||
|
"srcDirPath": themeResourcesDirPath,
|
||||||
|
"destDirPath": pathJoin(reactAppPublicDirPath, mockTestingResourcesPath)
|
||||||
|
});
|
||||||
|
|
||||||
|
const keycloakResourcesWithinPublicDirPath = pathJoin(reactAppPublicDirPath, mockTestingSubDirOfPublicDirBasename);
|
||||||
|
|
||||||
|
fs.writeFileSync(
|
||||||
|
pathJoin(keycloakResourcesWithinPublicDirPath, "README.txt"),
|
||||||
|
Buffer.from(
|
||||||
|
["This is just a test folder that helps develop", "the login and register page without having to run a Keycloak container"].join(
|
||||||
|
" "
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
fs.writeFileSync(pathJoin(keycloakResourcesWithinPublicDirPath, ".gitignore"), Buffer.from("*", "utf8"));
|
||||||
|
fs.rmSync(tmpDirPath, { recursive: true, force: true });
|
||||||
|
}
|
||||||
|
|
||||||
|
fs.writeFileSync(
|
||||||
|
pathJoin(themeDirPath, "theme.properties"),
|
||||||
|
Buffer.from(["parent=keycloak", ...(buildOptions.extraThemeProperties ?? [])].join("\n\n"), "utf8")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
let doBundlesEmailTemplate: boolean;
|
||||||
|
|
||||||
|
email: {
|
||||||
|
if (emailThemeSrcDirPath === undefined) {
|
||||||
|
doBundlesEmailTemplate = false;
|
||||||
|
break email;
|
||||||
|
}
|
||||||
|
|
||||||
|
doBundlesEmailTemplate = true;
|
||||||
|
|
||||||
|
transformCodebase({
|
||||||
|
"srcDirPath": emailThemeSrcDirPath,
|
||||||
|
"destDirPath": getThemeDirPath("email")
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return { doBundlesEmailTemplate };
|
||||||
|
}
|
61
src/bin/keycloakify/generateStartKeycloakTestingContainer.ts
Normal file
61
src/bin/keycloakify/generateStartKeycloakTestingContainer.ts
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
import * as fs from "fs";
|
||||||
|
import { join as pathJoin } from "path";
|
||||||
|
import { assert } from "tsafe/assert";
|
||||||
|
import { Reflect } from "tsafe/Reflect";
|
||||||
|
import type { BuildOptions } from "./BuildOptions";
|
||||||
|
|
||||||
|
export type BuildOptionsLike = {
|
||||||
|
themeName: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
{
|
||||||
|
const buildOptions = Reflect<BuildOptions>();
|
||||||
|
|
||||||
|
assert<typeof buildOptions extends BuildOptionsLike ? true : false>();
|
||||||
|
}
|
||||||
|
|
||||||
|
generateStartKeycloakTestingContainer.basename = "start_keycloak_testing_container.sh";
|
||||||
|
|
||||||
|
const containerName = "keycloak-testing-container";
|
||||||
|
|
||||||
|
/** Files for being able to run a hot reload keycloak container */
|
||||||
|
export function generateStartKeycloakTestingContainer(params: {
|
||||||
|
keycloakVersion: string;
|
||||||
|
keycloakThemeBuildingDirPath: string;
|
||||||
|
buildOptions: BuildOptionsLike;
|
||||||
|
}) {
|
||||||
|
const {
|
||||||
|
keycloakThemeBuildingDirPath,
|
||||||
|
keycloakVersion,
|
||||||
|
buildOptions: { themeName }
|
||||||
|
} = params;
|
||||||
|
|
||||||
|
const keycloakThemePath = pathJoin(keycloakThemeBuildingDirPath, "src", "main", "resources", "theme", themeName).replace(/\\/g, "/");
|
||||||
|
|
||||||
|
fs.writeFileSync(
|
||||||
|
pathJoin(keycloakThemeBuildingDirPath, generateStartKeycloakTestingContainer.basename),
|
||||||
|
|
||||||
|
Buffer.from(
|
||||||
|
[
|
||||||
|
"#!/usr/bin/env bash",
|
||||||
|
"",
|
||||||
|
`docker rm ${containerName} || true`,
|
||||||
|
"",
|
||||||
|
`cd "${keycloakThemeBuildingDirPath.replace(/\\/g, "/")}"`,
|
||||||
|
"",
|
||||||
|
"docker run \\",
|
||||||
|
" -p 8080:8080 \\",
|
||||||
|
` --name ${containerName} \\`,
|
||||||
|
" -e KEYCLOAK_ADMIN=admin \\",
|
||||||
|
" -e KEYCLOAK_ADMIN_PASSWORD=admin \\",
|
||||||
|
" -e JAVA_OPTS=-Dkeycloak.profile=preview \\",
|
||||||
|
` -v "${keycloakThemePath}":"/opt/keycloak/themes/${themeName}":rw \\`,
|
||||||
|
` -it quay.io/keycloak/keycloak:${keycloakVersion} \\`,
|
||||||
|
` start-dev`,
|
||||||
|
""
|
||||||
|
].join("\n"),
|
||||||
|
"utf8"
|
||||||
|
),
|
||||||
|
{ "mode": 0o755 }
|
||||||
|
);
|
||||||
|
}
|
8
src/bin/keycloakify/index.ts
Normal file
8
src/bin/keycloakify/index.ts
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
#!/usr/bin/env node
|
||||||
|
|
||||||
|
export * from "./keycloakify";
|
||||||
|
import { main } from "./keycloakify";
|
||||||
|
|
||||||
|
if (require.main === module) {
|
||||||
|
main();
|
||||||
|
}
|
152
src/bin/keycloakify/keycloakify.ts
Normal file
152
src/bin/keycloakify/keycloakify.ts
Normal file
@ -0,0 +1,152 @@
|
|||||||
|
import { generateKeycloakThemeResources } from "./generateKeycloakThemeResources";
|
||||||
|
import { generateJavaStackFiles } from "./generateJavaStackFiles";
|
||||||
|
import { join as pathJoin, relative as pathRelative, basename as pathBasename, sep as pathSep } from "path";
|
||||||
|
import * as child_process from "child_process";
|
||||||
|
import { generateStartKeycloakTestingContainer } from "./generateStartKeycloakTestingContainer";
|
||||||
|
import * as fs from "fs";
|
||||||
|
import { readBuildOptions } from "./BuildOptions";
|
||||||
|
import { getLogger } from "../tools/logger";
|
||||||
|
import { getCliOptions } from "../tools/cliOptions";
|
||||||
|
import jar from "../tools/jar";
|
||||||
|
import { assert } from "tsafe/assert";
|
||||||
|
import { Equals } from "tsafe";
|
||||||
|
import { getEmailThemeSrcDirPath } from "./build-paths";
|
||||||
|
import { getCnamePath, getAppInputPath, getKeycloakBuildPath, getReactProjectDirPath } from "./build-paths";
|
||||||
|
|
||||||
|
export async function main() {
|
||||||
|
const { isSilent, hasExternalAssets } = getCliOptions(process.argv.slice(2));
|
||||||
|
const logger = getLogger({ isSilent });
|
||||||
|
logger.log("🔏 Building the keycloak theme...⌚");
|
||||||
|
|
||||||
|
const buildOptions = readBuildOptions({
|
||||||
|
"CNAME": (() => {
|
||||||
|
const cnameFilePath = getCnamePath();
|
||||||
|
|
||||||
|
if (!fs.existsSync(cnameFilePath)) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
return fs.readFileSync(cnameFilePath).toString("utf8");
|
||||||
|
})(),
|
||||||
|
"isExternalAssetsCliParamProvided": hasExternalAssets,
|
||||||
|
"isSilent": isSilent
|
||||||
|
});
|
||||||
|
|
||||||
|
const { doBundlesEmailTemplate } = await generateKeycloakThemeResources({
|
||||||
|
keycloakThemeBuildingDirPath: buildOptions.keycloakBuildPath,
|
||||||
|
"emailThemeSrcDirPath": (() => {
|
||||||
|
const { emailThemeSrcDirPath } = getEmailThemeSrcDirPath();
|
||||||
|
|
||||||
|
if (emailThemeSrcDirPath === undefined || !fs.existsSync(emailThemeSrcDirPath)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
return emailThemeSrcDirPath;
|
||||||
|
})(),
|
||||||
|
"reactAppBuildDirPath": getAppInputPath(),
|
||||||
|
buildOptions,
|
||||||
|
"keycloakVersion": buildOptions.keycloakVersionDefaultAssets
|
||||||
|
});
|
||||||
|
|
||||||
|
const { jarFilePath } = generateJavaStackFiles({
|
||||||
|
keycloakThemeBuildingDirPath: buildOptions.keycloakBuildPath,
|
||||||
|
doBundlesEmailTemplate,
|
||||||
|
buildOptions
|
||||||
|
});
|
||||||
|
|
||||||
|
switch (buildOptions.bundler) {
|
||||||
|
case "none":
|
||||||
|
logger.log("😱 Skipping bundling step, there will be no jar");
|
||||||
|
break;
|
||||||
|
case "keycloakify":
|
||||||
|
logger.log("🫶 Let keycloakify do its thang");
|
||||||
|
await jar({
|
||||||
|
"rootPath": pathJoin(buildOptions.keycloakBuildPath, "src", "main", "resources"),
|
||||||
|
"version": buildOptions.version,
|
||||||
|
"groupId": buildOptions.groupId,
|
||||||
|
"artifactId": buildOptions.artifactId,
|
||||||
|
"targetPath": jarFilePath
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
case "mvn":
|
||||||
|
logger.log("🫙 Run maven to deliver a jar");
|
||||||
|
child_process.execSync("mvn package", { "cwd": buildOptions.keycloakBuildPath });
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
assert<Equals<typeof buildOptions.bundler, never>>(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
// We want, however, to test in a container running the latest Keycloak version
|
||||||
|
const containerKeycloakVersion = "20.0.1";
|
||||||
|
|
||||||
|
generateStartKeycloakTestingContainer({
|
||||||
|
keycloakThemeBuildingDirPath: buildOptions.keycloakBuildPath,
|
||||||
|
"keycloakVersion": containerKeycloakVersion,
|
||||||
|
buildOptions
|
||||||
|
});
|
||||||
|
|
||||||
|
logger.log(
|
||||||
|
[
|
||||||
|
"",
|
||||||
|
`✅ Your keycloak theme has been generated and bundled into ./${pathRelative(getReactProjectDirPath(), jarFilePath)} 🚀`,
|
||||||
|
`It is to be placed in "/opt/keycloak/providers" in the container running a quay.io/keycloak/keycloak Docker image.`,
|
||||||
|
"",
|
||||||
|
//TODO: Restore when we find a good Helm chart for Keycloak.
|
||||||
|
//"Using Helm (https://github.com/codecentric/helm-charts), edit to reflect:",
|
||||||
|
"",
|
||||||
|
"value.yaml: ",
|
||||||
|
" extraInitContainers: |",
|
||||||
|
" - name: realm-ext-provider",
|
||||||
|
" image: curlimages/curl",
|
||||||
|
" imagePullPolicy: IfNotPresent",
|
||||||
|
" command:",
|
||||||
|
" - sh",
|
||||||
|
" args:",
|
||||||
|
" - -c",
|
||||||
|
` - curl -L -f -S -o /extensions/${pathBasename(jarFilePath)} https://AN.URL.FOR/${pathBasename(jarFilePath)}`,
|
||||||
|
" volumeMounts:",
|
||||||
|
" - name: extensions",
|
||||||
|
" mountPath: /extensions",
|
||||||
|
" ",
|
||||||
|
" extraVolumeMounts: |",
|
||||||
|
" - name: extensions",
|
||||||
|
" mountPath: /opt/keycloak/providers",
|
||||||
|
" extraEnv: |",
|
||||||
|
" - name: KEYCLOAK_USER",
|
||||||
|
" value: admin",
|
||||||
|
" - name: KEYCLOAK_PASSWORD",
|
||||||
|
" value: xxxxxxxxx",
|
||||||
|
" - name: JAVA_OPTS",
|
||||||
|
" value: -Dkeycloak.profile=preview",
|
||||||
|
"",
|
||||||
|
"",
|
||||||
|
`To test your theme locally you can spin up a Keycloak ${containerKeycloakVersion} container image with the theme pre loaded by running:`,
|
||||||
|
"",
|
||||||
|
`👉 $ .${pathSep}${pathRelative(
|
||||||
|
getReactProjectDirPath(),
|
||||||
|
pathJoin(getKeycloakBuildPath(), generateStartKeycloakTestingContainer.basename)
|
||||||
|
)} 👈`,
|
||||||
|
"",
|
||||||
|
`Test with different Keycloak versions by editing the .sh file. see available versions here: https://quay.io/repository/keycloak/keycloak?tab=tags`,
|
||||||
|
``,
|
||||||
|
`Once your container is up and running: `,
|
||||||
|
"- Log into the admin console 👉 http://localhost:8080/admin username: admin, password: admin 👈",
|
||||||
|
`- Create a realm: myrealm`,
|
||||||
|
`- Enable registration: Realm settings -> Login tab -> User registration: on`,
|
||||||
|
`- Enable the Account theme: Realm settings -> Themes tab -> Account theme, select ${buildOptions.themeName} `,
|
||||||
|
`- Create a client id myclient`,
|
||||||
|
` Root URL: https://www.keycloak.org/app/`,
|
||||||
|
` Valid redirect URIs: https://www.keycloak.org/app* http://localhost* (localhost is optional)`,
|
||||||
|
` Valid post logout redirect URIs: https://www.keycloak.org/app* http://localhost*`,
|
||||||
|
` Web origins: *`,
|
||||||
|
` Login Theme: ${buildOptions.themeName}`,
|
||||||
|
` Save (button at the bottom of the page)`,
|
||||||
|
``,
|
||||||
|
`- Go to 👉 https://www.keycloak.org/app/ 👈 Click "Save" then "Sign in". You should see your login page`,
|
||||||
|
`- Got to 👉 http://localhost:8080/realms/myrealm/account 👈 to see your account theme`,
|
||||||
|
``,
|
||||||
|
`Video tutorial: https://youtu.be/WMyGZNHQkjU`,
|
||||||
|
``
|
||||||
|
].join("\n")
|
||||||
|
);
|
||||||
|
}
|
62
src/bin/keycloakify/parsedPackageJson.ts
Normal file
62
src/bin/keycloakify/parsedPackageJson.ts
Normal file
@ -0,0 +1,62 @@
|
|||||||
|
import * as fs from "fs";
|
||||||
|
import { assert } from "tsafe";
|
||||||
|
import type { Equals } from "tsafe";
|
||||||
|
import { z } from "zod";
|
||||||
|
import { pathJoin } from "../tools/pathJoin";
|
||||||
|
|
||||||
|
const reactProjectDirPath = process.cwd();
|
||||||
|
export const bundlers = ["mvn", "keycloakify", "none"] as const;
|
||||||
|
export type Bundler = (typeof bundlers)[number];
|
||||||
|
type ParsedPackageJson = {
|
||||||
|
name: string;
|
||||||
|
version: string;
|
||||||
|
homepage?: string;
|
||||||
|
keycloakify?: {
|
||||||
|
/** @deprecated: use extraLoginPages instead */
|
||||||
|
extraPages?: string[];
|
||||||
|
extraLoginPages?: string[];
|
||||||
|
extraAccountPages?: string[];
|
||||||
|
extraThemeProperties?: string[];
|
||||||
|
areAppAndKeycloakServerSharingSameDomain?: boolean;
|
||||||
|
artifactId?: string;
|
||||||
|
groupId?: string;
|
||||||
|
bundler?: Bundler;
|
||||||
|
keycloakVersionDefaultAssets?: string;
|
||||||
|
appInputPath?: string;
|
||||||
|
keycloakBuildPath?: string;
|
||||||
|
customUserAttributes?: string[];
|
||||||
|
themeName?: string;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
const zParsedPackageJson = z.object({
|
||||||
|
"name": z.string(),
|
||||||
|
"version": z.string(),
|
||||||
|
"homepage": z.string().optional(),
|
||||||
|
"keycloakify": z
|
||||||
|
.object({
|
||||||
|
"extraPages": z.array(z.string()).optional(),
|
||||||
|
"extraLoginPages": z.array(z.string()).optional(),
|
||||||
|
"extraAccountPages": z.array(z.string()).optional(),
|
||||||
|
"extraThemeProperties": z.array(z.string()).optional(),
|
||||||
|
"areAppAndKeycloakServerSharingSameDomain": z.boolean().optional(),
|
||||||
|
"artifactId": z.string().optional(),
|
||||||
|
"groupId": z.string().optional(),
|
||||||
|
"bundler": z.enum(bundlers).optional(),
|
||||||
|
"keycloakVersionDefaultAssets": z.string().optional(),
|
||||||
|
"appInputPath": z.string().optional(),
|
||||||
|
"keycloakBuildPath": z.string().optional(),
|
||||||
|
"customUserAttributes": z.array(z.string()).optional(),
|
||||||
|
"themeName": z.string().optional()
|
||||||
|
})
|
||||||
|
.optional()
|
||||||
|
});
|
||||||
|
|
||||||
|
assert<Equals<ReturnType<(typeof zParsedPackageJson)["parse"]>, ParsedPackageJson>>();
|
||||||
|
|
||||||
|
let parsedPackageJson: undefined | ReturnType<(typeof zParsedPackageJson)["parse"]>;
|
||||||
|
export const getParsedPackageJson = () => {
|
||||||
|
if (parsedPackageJson) return parsedPackageJson;
|
||||||
|
parsedPackageJson = zParsedPackageJson.parse(JSON.parse(fs.readFileSync(pathJoin(reactProjectDirPath, "package.json")).toString("utf8")));
|
||||||
|
return parsedPackageJson;
|
||||||
|
};
|
@ -0,0 +1,86 @@
|
|||||||
|
import { ftlValuesGlobalName } from "../ftlValuesGlobalName";
|
||||||
|
import type { BuildOptions } from "../BuildOptions";
|
||||||
|
import { assert } from "tsafe/assert";
|
||||||
|
import { is } from "tsafe/is";
|
||||||
|
import { Reflect } from "tsafe/Reflect";
|
||||||
|
|
||||||
|
export type BuildOptionsLike = BuildOptionsLike.Standalone | BuildOptionsLike.ExternalAssets;
|
||||||
|
|
||||||
|
export namespace BuildOptionsLike {
|
||||||
|
export type Standalone = {
|
||||||
|
isStandalone: true;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type ExternalAssets = {
|
||||||
|
isStandalone: false;
|
||||||
|
urlOrigin: string;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
const buildOptions = Reflect<BuildOptions>();
|
||||||
|
|
||||||
|
assert(!is<BuildOptions.ExternalAssets.CommonExternalAssets>(buildOptions));
|
||||||
|
|
||||||
|
assert<typeof buildOptions extends BuildOptionsLike ? true : false>();
|
||||||
|
}
|
||||||
|
|
||||||
|
export function replaceImportsFromStaticInJsCode(params: { jsCode: string; buildOptions: BuildOptionsLike }): { fixedJsCode: string } {
|
||||||
|
/*
|
||||||
|
NOTE:
|
||||||
|
|
||||||
|
When we have urlOrigin defined it means that
|
||||||
|
we are building with --external-assets
|
||||||
|
so we have to make sur that the fixed js code will run
|
||||||
|
inside and outside keycloak.
|
||||||
|
|
||||||
|
When urlOrigin isn't defined we can assume the fixedJsCode
|
||||||
|
will always run in keycloak context.
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { jsCode, buildOptions } = params;
|
||||||
|
|
||||||
|
const getReplaceArgs = (language: "js" | "css"): Parameters<typeof String.prototype.replace> => [
|
||||||
|
new RegExp(`([a-zA-Z_]+)\\.([a-zA-Z]+)=function\\(([a-zA-Z]+)\\){return"static\\/${language}\\/"`, "g"),
|
||||||
|
(...[, n, u, e]) => `
|
||||||
|
${n}[(function(){
|
||||||
|
var pd= Object.getOwnPropertyDescriptor(${n}, "p");
|
||||||
|
if( pd === undefined || pd.configurable ){
|
||||||
|
${
|
||||||
|
buildOptions.isStandalone
|
||||||
|
? `
|
||||||
|
Object.defineProperty(${n}, "p", {
|
||||||
|
get: function() { return window.${ftlValuesGlobalName}.url.resourcesPath; },
|
||||||
|
set: function (){}
|
||||||
|
});
|
||||||
|
`
|
||||||
|
: `
|
||||||
|
var p= "";
|
||||||
|
Object.defineProperty(${n}, "p", {
|
||||||
|
get: function() { return "${ftlValuesGlobalName}" in window ? "${buildOptions.urlOrigin}/" : p; },
|
||||||
|
set: function (value){ p = value;}
|
||||||
|
});
|
||||||
|
`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return "${u}";
|
||||||
|
})()] = function(${e}) { return "${buildOptions.isStandalone ? "/build/" : ""}static/${language}/"`
|
||||||
|
];
|
||||||
|
|
||||||
|
const fixedJsCode = jsCode
|
||||||
|
.replace(...getReplaceArgs("js"))
|
||||||
|
.replace(...getReplaceArgs("css"))
|
||||||
|
.replace(/([a-zA-Z]+\.[a-zA-Z]+)\+"static\//g, (...[, group]) =>
|
||||||
|
buildOptions.isStandalone
|
||||||
|
? `window.${ftlValuesGlobalName}.url.resourcesPath + "/build/static/`
|
||||||
|
: `("${ftlValuesGlobalName}" in window ? "${buildOptions.urlOrigin}/" : ${group}) + "static/`
|
||||||
|
)
|
||||||
|
//TODO: Write a test case for this
|
||||||
|
.replace(/".chunk.css",([a-zA-Z])+=([a-zA-Z]+\.[a-zA-Z]+)\+([a-zA-Z]+),/, (...[, group1, group2, group3]) =>
|
||||||
|
buildOptions.isStandalone
|
||||||
|
? `".chunk.css",${group1} = window.${ftlValuesGlobalName}.url.resourcesPath + "/build/" + ${group3},`
|
||||||
|
: `".chunk.css",${group1} = ("${ftlValuesGlobalName}" in window ? "${buildOptions.urlOrigin}/" : ${group2}) + ${group3},`
|
||||||
|
);
|
||||||
|
|
||||||
|
return { fixedJsCode };
|
||||||
|
}
|
64
src/bin/keycloakify/replacers/replaceImportsInCssCode.ts
Normal file
64
src/bin/keycloakify/replacers/replaceImportsInCssCode.ts
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
import * as crypto from "crypto";
|
||||||
|
import type { BuildOptions } from "../BuildOptions";
|
||||||
|
import { assert } from "tsafe/assert";
|
||||||
|
import { is } from "tsafe/is";
|
||||||
|
import { Reflect } from "tsafe/Reflect";
|
||||||
|
|
||||||
|
export type BuildOptionsLike = {
|
||||||
|
urlPathname: string | undefined;
|
||||||
|
};
|
||||||
|
|
||||||
|
{
|
||||||
|
const buildOptions = Reflect<BuildOptions>();
|
||||||
|
|
||||||
|
assert(!is<BuildOptions.ExternalAssets.CommonExternalAssets>(buildOptions));
|
||||||
|
|
||||||
|
assert<typeof buildOptions extends BuildOptionsLike ? true : false>();
|
||||||
|
}
|
||||||
|
|
||||||
|
export function replaceImportsInCssCode(params: { cssCode: string }): {
|
||||||
|
fixedCssCode: string;
|
||||||
|
cssGlobalsToDefine: Record<string, string>;
|
||||||
|
} {
|
||||||
|
const { cssCode } = params;
|
||||||
|
|
||||||
|
const cssGlobalsToDefine: Record<string, string> = {};
|
||||||
|
|
||||||
|
new Set(cssCode.match(/url\(["']?\/[^/][^)"']+["']?\)[^;}]*/g) ?? []).forEach(
|
||||||
|
match => (cssGlobalsToDefine["url" + crypto.createHash("sha256").update(match).digest("hex").substring(0, 15)] = match)
|
||||||
|
);
|
||||||
|
|
||||||
|
let fixedCssCode = cssCode;
|
||||||
|
|
||||||
|
Object.keys(cssGlobalsToDefine).forEach(
|
||||||
|
cssVariableName =>
|
||||||
|
//NOTE: split/join pattern ~ replace all
|
||||||
|
(fixedCssCode = fixedCssCode.split(cssGlobalsToDefine[cssVariableName]).join(`var(--${cssVariableName})`))
|
||||||
|
);
|
||||||
|
|
||||||
|
return { fixedCssCode, cssGlobalsToDefine };
|
||||||
|
}
|
||||||
|
|
||||||
|
export function generateCssCodeToDefineGlobals(params: { cssGlobalsToDefine: Record<string, string>; buildOptions: BuildOptionsLike }): {
|
||||||
|
cssCodeToPrependInHead: string;
|
||||||
|
} {
|
||||||
|
const { cssGlobalsToDefine, buildOptions } = params;
|
||||||
|
|
||||||
|
return {
|
||||||
|
"cssCodeToPrependInHead": [
|
||||||
|
":root {",
|
||||||
|
...Object.keys(cssGlobalsToDefine)
|
||||||
|
.map(cssVariableName =>
|
||||||
|
[
|
||||||
|
`--${cssVariableName}:`,
|
||||||
|
cssGlobalsToDefine[cssVariableName].replace(
|
||||||
|
new RegExp(`url\\(${(buildOptions.urlPathname ?? "/").replace(/\//g, "\\/")}`, "g"),
|
||||||
|
"url(${url.resourcesPath}/build/"
|
||||||
|
)
|
||||||
|
].join(" ")
|
||||||
|
)
|
||||||
|
.map(line => ` ${line};`),
|
||||||
|
"}"
|
||||||
|
].join("\n")
|
||||||
|
};
|
||||||
|
}
|
@ -0,0 +1,47 @@
|
|||||||
|
import type { BuildOptions } from "../BuildOptions";
|
||||||
|
import { assert } from "tsafe/assert";
|
||||||
|
import { is } from "tsafe/is";
|
||||||
|
import { Reflect } from "tsafe/Reflect";
|
||||||
|
|
||||||
|
export type BuildOptionsLike = BuildOptionsLike.Standalone | BuildOptionsLike.ExternalAssets;
|
||||||
|
|
||||||
|
export namespace BuildOptionsLike {
|
||||||
|
export type Common = {
|
||||||
|
urlPathname: string | undefined;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Standalone = Common & {
|
||||||
|
isStandalone: true;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type ExternalAssets = Common & {
|
||||||
|
isStandalone: false;
|
||||||
|
urlOrigin: string;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
const buildOptions = Reflect<BuildOptions>();
|
||||||
|
|
||||||
|
assert(!is<BuildOptions.ExternalAssets.CommonExternalAssets>(buildOptions));
|
||||||
|
|
||||||
|
assert<typeof buildOptions extends BuildOptionsLike ? true : false>();
|
||||||
|
}
|
||||||
|
|
||||||
|
export function replaceImportsInInlineCssCode(params: { cssCode: string; buildOptions: BuildOptionsLike }): {
|
||||||
|
fixedCssCode: string;
|
||||||
|
} {
|
||||||
|
const { cssCode, buildOptions } = params;
|
||||||
|
|
||||||
|
const fixedCssCode = cssCode.replace(
|
||||||
|
buildOptions.urlPathname === undefined
|
||||||
|
? /url\(["']?\/([^/][^)"']+)["']?\)/g
|
||||||
|
: new RegExp(`url\\(["']?${buildOptions.urlPathname}([^)"']+)["']?\\)`, "g"),
|
||||||
|
(...[, group]) =>
|
||||||
|
`url(${
|
||||||
|
buildOptions.isStandalone ? "${url.resourcesPath}/build/" + group : buildOptions.urlOrigin + (buildOptions.urlPathname ?? "/") + group
|
||||||
|
})`
|
||||||
|
);
|
||||||
|
|
||||||
|
return { fixedCssCode };
|
||||||
|
}
|
@ -1,102 +0,0 @@
|
|||||||
import { execSync } from "child_process";
|
|
||||||
import { join as pathJoin, relative as pathRelative } from "path";
|
|
||||||
import * as fs from "fs";
|
|
||||||
|
|
||||||
const keycloakifyDirPath = pathJoin(__dirname, "..", "..");
|
|
||||||
|
|
||||||
fs.writeFileSync(
|
|
||||||
pathJoin(keycloakifyDirPath, "dist", "package.json"),
|
|
||||||
Buffer.from(
|
|
||||||
JSON.stringify(
|
|
||||||
(() => {
|
|
||||||
const packageJsonParsed = JSON.parse(fs.readFileSync(pathJoin(keycloakifyDirPath, "package.json")).toString("utf8"));
|
|
||||||
|
|
||||||
return {
|
|
||||||
...packageJsonParsed,
|
|
||||||
"main": packageJsonParsed["main"].replace(/^dist\//, ""),
|
|
||||||
"types": packageJsonParsed["types"].replace(/^dist\//, ""),
|
|
||||||
};
|
|
||||||
})(),
|
|
||||||
null,
|
|
||||||
2,
|
|
||||||
),
|
|
||||||
"utf8",
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
const commonThirdPartyDeps = (() => {
|
|
||||||
const namespaceModuleNames = ["@emotion"];
|
|
||||||
const standaloneModuleNames = ["react", "@types/react", "powerhooks", "tss-react", "evt"];
|
|
||||||
|
|
||||||
return [
|
|
||||||
...namespaceModuleNames
|
|
||||||
.map(namespaceModuleName =>
|
|
||||||
fs
|
|
||||||
.readdirSync(pathJoin(keycloakifyDirPath, "node_modules", namespaceModuleName))
|
|
||||||
.map(submoduleName => `${namespaceModuleName}/${submoduleName}`),
|
|
||||||
)
|
|
||||||
.reduce((prev, curr) => [...prev, ...curr], []),
|
|
||||||
...standaloneModuleNames,
|
|
||||||
];
|
|
||||||
})();
|
|
||||||
|
|
||||||
const yarnHomeDirPath = pathJoin(keycloakifyDirPath, ".yarn_home");
|
|
||||||
|
|
||||||
execSync(["rm -rf", "mkdir"].map(cmd => `${cmd} ${yarnHomeDirPath}`).join(" && "));
|
|
||||||
|
|
||||||
const execYarnLink = (params: { targetModuleName?: string; cwd: string }) => {
|
|
||||||
const { targetModuleName, cwd } = params;
|
|
||||||
|
|
||||||
const cmd = ["yarn", "link", ...(targetModuleName !== undefined ? [targetModuleName] : [])].join(" ");
|
|
||||||
|
|
||||||
console.log(`$ cd ${pathRelative(keycloakifyDirPath, cwd) || "."} && ${cmd}`);
|
|
||||||
|
|
||||||
execSync(cmd, {
|
|
||||||
cwd,
|
|
||||||
"env": {
|
|
||||||
...process.env,
|
|
||||||
"HOME": yarnHomeDirPath,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const testAppNames = [process.argv[2] ?? "keycloakify-demo-app"] as const;
|
|
||||||
|
|
||||||
const getTestAppPath = (testAppName: typeof testAppNames[number]) => pathJoin(keycloakifyDirPath, "..", testAppName);
|
|
||||||
|
|
||||||
testAppNames.forEach(testAppName => execSync("yarn install", { "cwd": getTestAppPath(testAppName) }));
|
|
||||||
|
|
||||||
console.log("=== Linking common dependencies ===");
|
|
||||||
|
|
||||||
const total = commonThirdPartyDeps.length;
|
|
||||||
let current = 0;
|
|
||||||
|
|
||||||
commonThirdPartyDeps.forEach(commonThirdPartyDep => {
|
|
||||||
current++;
|
|
||||||
|
|
||||||
console.log(`${current}/${total} ${commonThirdPartyDep}`);
|
|
||||||
|
|
||||||
const localInstallPath = pathJoin(
|
|
||||||
...[keycloakifyDirPath, "node_modules", ...(commonThirdPartyDep.startsWith("@") ? commonThirdPartyDep.split("/") : [commonThirdPartyDep])],
|
|
||||||
);
|
|
||||||
|
|
||||||
execYarnLink({ "cwd": localInstallPath });
|
|
||||||
|
|
||||||
testAppNames.forEach(testAppName =>
|
|
||||||
execYarnLink({
|
|
||||||
"cwd": getTestAppPath(testAppName),
|
|
||||||
"targetModuleName": commonThirdPartyDep,
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log("=== Linking in house dependencies ===");
|
|
||||||
|
|
||||||
execYarnLink({ "cwd": pathJoin(keycloakifyDirPath, "dist") });
|
|
||||||
|
|
||||||
testAppNames.forEach(testAppName =>
|
|
||||||
execYarnLink({
|
|
||||||
"cwd": getTestAppPath(testAppName),
|
|
||||||
"targetModuleName": "keycloakify",
|
|
||||||
}),
|
|
||||||
);
|
|
@ -24,9 +24,9 @@ export async function promptKeycloakVersion() {
|
|||||||
"count": 10,
|
"count": 10,
|
||||||
"doIgnoreBeta": true,
|
"doIgnoreBeta": true,
|
||||||
"owner": "keycloak",
|
"owner": "keycloak",
|
||||||
"repo": "keycloak",
|
"repo": "keycloak"
|
||||||
}).then(arr => arr.map(({ tag }) => tag))),
|
}).then(arr => arr.map(({ tag }) => tag))),
|
||||||
"11.0.3",
|
"11.0.3"
|
||||||
];
|
];
|
||||||
|
|
||||||
if (process.env["GITHUB_ACTIONS"] === "true") {
|
if (process.env["GITHUB_ACTIONS"] === "true") {
|
||||||
@ -34,7 +34,7 @@ export async function promptKeycloakVersion() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const { value: keycloakVersion } = await cliSelect<string>({
|
const { value: keycloakVersion } = await cliSelect<string>({
|
||||||
"values": tags,
|
"values": tags
|
||||||
}).catch(() => {
|
}).catch(() => {
|
||||||
console.log("Aborting");
|
console.log("Aborting");
|
||||||
|
|
||||||
|
@ -20,7 +20,7 @@ export namespace NpmModuleVersion {
|
|||||||
...(() => {
|
...(() => {
|
||||||
const str = match[4];
|
const str = match[4];
|
||||||
return str === undefined ? {} : { "betaPreRelease": parseInt(str) };
|
return str === undefined ? {} : { "betaPreRelease": parseInt(str) };
|
||||||
})(),
|
})()
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
15
src/bin/tools/cliOptions.ts
Normal file
15
src/bin/tools/cliOptions.ts
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
import parseArgv from "minimist";
|
||||||
|
|
||||||
|
export type CliOptions = {
|
||||||
|
isSilent: boolean;
|
||||||
|
hasExternalAssets: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getCliOptions = (processArgv: string[]): CliOptions => {
|
||||||
|
const argv = parseArgv(processArgv);
|
||||||
|
|
||||||
|
return {
|
||||||
|
isSilent: typeof argv["silent"] === "boolean" ? argv["silent"] : false,
|
||||||
|
hasExternalAssets: typeof argv["external-assets"] === "boolean" ? argv["external-assets"] : false
|
||||||
|
};
|
||||||
|
};
|
55
src/bin/tools/crc32.ts
Normal file
55
src/bin/tools/crc32.ts
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
import { Readable } from "stream";
|
||||||
|
|
||||||
|
const crc32tab = [
|
||||||
|
0x00000000, 0x77073096, 0xee0e612c, 0x990951ba, 0x076dc419, 0x706af48f, 0xe963a535, 0x9e6495a3, 0x0edb8832, 0x79dcb8a4, 0xe0d5e91e, 0x97d2d988,
|
||||||
|
0x09b64c2b, 0x7eb17cbd, 0xe7b82d07, 0x90bf1d91, 0x1db71064, 0x6ab020f2, 0xf3b97148, 0x84be41de, 0x1adad47d, 0x6ddde4eb, 0xf4d4b551, 0x83d385c7,
|
||||||
|
0x136c9856, 0x646ba8c0, 0xfd62f97a, 0x8a65c9ec, 0x14015c4f, 0x63066cd9, 0xfa0f3d63, 0x8d080df5, 0x3b6e20c8, 0x4c69105e, 0xd56041e4, 0xa2677172,
|
||||||
|
0x3c03e4d1, 0x4b04d447, 0xd20d85fd, 0xa50ab56b, 0x35b5a8fa, 0x42b2986c, 0xdbbbc9d6, 0xacbcf940, 0x32d86ce3, 0x45df5c75, 0xdcd60dcf, 0xabd13d59,
|
||||||
|
0x26d930ac, 0x51de003a, 0xc8d75180, 0xbfd06116, 0x21b4f4b5, 0x56b3c423, 0xcfba9599, 0xb8bda50f, 0x2802b89e, 0x5f058808, 0xc60cd9b2, 0xb10be924,
|
||||||
|
0x2f6f7c87, 0x58684c11, 0xc1611dab, 0xb6662d3d, 0x76dc4190, 0x01db7106, 0x98d220bc, 0xefd5102a, 0x71b18589, 0x06b6b51f, 0x9fbfe4a5, 0xe8b8d433,
|
||||||
|
0x7807c9a2, 0x0f00f934, 0x9609a88e, 0xe10e9818, 0x7f6a0dbb, 0x086d3d2d, 0x91646c97, 0xe6635c01, 0x6b6b51f4, 0x1c6c6162, 0x856530d8, 0xf262004e,
|
||||||
|
0x6c0695ed, 0x1b01a57b, 0x8208f4c1, 0xf50fc457, 0x65b0d9c6, 0x12b7e950, 0x8bbeb8ea, 0xfcb9887c, 0x62dd1ddf, 0x15da2d49, 0x8cd37cf3, 0xfbd44c65,
|
||||||
|
0x4db26158, 0x3ab551ce, 0xa3bc0074, 0xd4bb30e2, 0x4adfa541, 0x3dd895d7, 0xa4d1c46d, 0xd3d6f4fb, 0x4369e96a, 0x346ed9fc, 0xad678846, 0xda60b8d0,
|
||||||
|
0x44042d73, 0x33031de5, 0xaa0a4c5f, 0xdd0d7cc9, 0x5005713c, 0x270241aa, 0xbe0b1010, 0xc90c2086, 0x5768b525, 0x206f85b3, 0xb966d409, 0xce61e49f,
|
||||||
|
0x5edef90e, 0x29d9c998, 0xb0d09822, 0xc7d7a8b4, 0x59b33d17, 0x2eb40d81, 0xb7bd5c3b, 0xc0ba6cad, 0xedb88320, 0x9abfb3b6, 0x03b6e20c, 0x74b1d29a,
|
||||||
|
0xead54739, 0x9dd277af, 0x04db2615, 0x73dc1683, 0xe3630b12, 0x94643b84, 0x0d6d6a3e, 0x7a6a5aa8, 0xe40ecf0b, 0x9309ff9d, 0x0a00ae27, 0x7d079eb1,
|
||||||
|
0xf00f9344, 0x8708a3d2, 0x1e01f268, 0x6906c2fe, 0xf762575d, 0x806567cb, 0x196c3671, 0x6e6b06e7, 0xfed41b76, 0x89d32be0, 0x10da7a5a, 0x67dd4acc,
|
||||||
|
0xf9b9df6f, 0x8ebeeff9, 0x17b7be43, 0x60b08ed5, 0xd6d6a3e8, 0xa1d1937e, 0x38d8c2c4, 0x4fdff252, 0xd1bb67f1, 0xa6bc5767, 0x3fb506dd, 0x48b2364b,
|
||||||
|
0xd80d2bda, 0xaf0a1b4c, 0x36034af6, 0x41047a60, 0xdf60efc3, 0xa867df55, 0x316e8eef, 0x4669be79, 0xcb61b38c, 0xbc66831a, 0x256fd2a0, 0x5268e236,
|
||||||
|
0xcc0c7795, 0xbb0b4703, 0x220216b9, 0x5505262f, 0xc5ba3bbe, 0xb2bd0b28, 0x2bb45a92, 0x5cb36a04, 0xc2d7ffa7, 0xb5d0cf31, 0x2cd99e8b, 0x5bdeae1d,
|
||||||
|
0x9b64c2b0, 0xec63f226, 0x756aa39c, 0x026d930a, 0x9c0906a9, 0xeb0e363f, 0x72076785, 0x05005713, 0x95bf4a82, 0xe2b87a14, 0x7bb12bae, 0x0cb61b38,
|
||||||
|
0x92d28e9b, 0xe5d5be0d, 0x7cdcefb7, 0x0bdbdf21, 0x86d3d2d4, 0xf1d4e242, 0x68ddb3f8, 0x1fda836e, 0x81be16cd, 0xf6b9265b, 0x6fb077e1, 0x18b74777,
|
||||||
|
0x88085ae6, 0xff0f6a70, 0x66063bca, 0x11010b5c, 0x8f659eff, 0xf862ae69, 0x616bffd3, 0x166ccf45, 0xa00ae278, 0xd70dd2ee, 0x4e048354, 0x3903b3c2,
|
||||||
|
0xa7672661, 0xd06016f7, 0x4969474d, 0x3e6e77db, 0xaed16a4a, 0xd9d65adc, 0x40df0b66, 0x37d83bf0, 0xa9bcae53, 0xdebb9ec5, 0x47b2cf7f, 0x30b5ffe9,
|
||||||
|
0xbdbdf21c, 0xcabac28a, 0x53b39330, 0x24b4a3a6, 0xbad03605, 0xcdd70693, 0x54de5729, 0x23d967bf, 0xb3667a2e, 0xc4614ab8, 0x5d681b02, 0x2a6f2b94,
|
||||||
|
0xb40bbe37, 0xc30c8ea1, 0x5a05df1b, 0x2d02ef8d
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param input either a byte stream, a string or a buffer, you want to have the checksum for
|
||||||
|
* @returns a promise for a checksum (uint32)
|
||||||
|
*/
|
||||||
|
export function crc32(input: Readable | String | Buffer): Promise<number> {
|
||||||
|
if (typeof input === "string") {
|
||||||
|
let crc = ~0;
|
||||||
|
for (let i = 0; i < input.length; i++) crc = (crc >>> 8) ^ crc32tab[(crc ^ input.charCodeAt(i)) & 0xff];
|
||||||
|
return Promise.resolve((crc ^ -1) >>> 0);
|
||||||
|
} else if (input instanceof Buffer) {
|
||||||
|
let crc = ~0;
|
||||||
|
for (let i = 0; i < input.length; i++) crc = (crc >>> 8) ^ crc32tab[(crc ^ input[i]) & 0xff];
|
||||||
|
return Promise.resolve((crc ^ -1) >>> 0);
|
||||||
|
} else if (input instanceof Readable) {
|
||||||
|
return new Promise<number>((resolve, reject) => {
|
||||||
|
let crc = ~0;
|
||||||
|
input.setMaxListeners(Infinity);
|
||||||
|
input.on("end", () => resolve((crc ^ -1) >>> 0));
|
||||||
|
input.on("error", e => reject(e));
|
||||||
|
input.on("data", (chunk: Buffer) => {
|
||||||
|
for (let i = 0; i < chunk.length; i++) crc = (crc >>> 8) ^ crc32tab[(crc ^ chunk[i]) & 0xff];
|
||||||
|
});
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
throw new Error("Unsupported input " + typeof input);
|
||||||
|
}
|
||||||
|
}
|
61
src/bin/tools/deflate.ts
Normal file
61
src/bin/tools/deflate.ts
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
import { PassThrough, Readable, TransformCallback, Writable } from "stream";
|
||||||
|
import { pipeline } from "stream/promises";
|
||||||
|
import { deflateRaw as deflateRawCb, createDeflateRaw } from "zlib";
|
||||||
|
import { promisify } from "util";
|
||||||
|
|
||||||
|
import { crc32 } from "./crc32";
|
||||||
|
import tee from "./tee";
|
||||||
|
|
||||||
|
const deflateRaw = promisify(deflateRawCb);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A stream transformer that records the number of bytes
|
||||||
|
* passed in its `size` property.
|
||||||
|
*/
|
||||||
|
class ByteCounter extends PassThrough {
|
||||||
|
size: number = 0;
|
||||||
|
_transform(chunk: any, encoding: BufferEncoding, callback: TransformCallback) {
|
||||||
|
if ("length" in chunk) this.size += chunk.length;
|
||||||
|
super._transform(chunk, encoding, callback);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param data buffer containing the data to be compressed
|
||||||
|
* @returns a buffer containing the compressed/deflated data and the crc32 checksum
|
||||||
|
* of the source data
|
||||||
|
*/
|
||||||
|
export async function deflateBuffer(data: Buffer) {
|
||||||
|
const [deflated, checksum] = await Promise.all([deflateRaw(data), crc32(data)]);
|
||||||
|
return { deflated, crc32: checksum };
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param input a byte stream, containing data to be compressed
|
||||||
|
* @param sink a method that will accept chunks of compressed data; We don't pass
|
||||||
|
* a writable here, since we don't want the writablestream to be closed after
|
||||||
|
* a single file
|
||||||
|
* @returns a promise, which will resolve with the crc32 checksum and the
|
||||||
|
* compressed size
|
||||||
|
*/
|
||||||
|
export async function deflateStream(input: Readable, sink: (chunk: Buffer) => void) {
|
||||||
|
const deflateWriter = new Writable({
|
||||||
|
write(chunk, _, callback) {
|
||||||
|
sink(chunk);
|
||||||
|
callback();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// tee the input stream, so we can compress and calc crc32 in parallel
|
||||||
|
const [rs1, rs2] = tee(input);
|
||||||
|
const byteCounter = new ByteCounter();
|
||||||
|
const [_, crc] = await Promise.all([
|
||||||
|
// pipe input into zip compressor, count the bytes
|
||||||
|
// returned and pass compressed data to the sink
|
||||||
|
pipeline(rs1, createDeflateRaw(), byteCounter, deflateWriter),
|
||||||
|
// calc checksum
|
||||||
|
crc32(rs2)
|
||||||
|
]);
|
||||||
|
|
||||||
|
return { crc32: crc, compressedSize: byteCounter.size };
|
||||||
|
}
|
@ -1,32 +1,87 @@
|
|||||||
import { basename as pathBasename, join as pathJoin } from "path";
|
import { exec as execCallback } from "child_process";
|
||||||
import { execSync } from "child_process";
|
import { createHash } from "crypto";
|
||||||
import * as fs from "fs";
|
import { mkdir, stat, writeFile } from "fs/promises";
|
||||||
|
import fetch, { type FetchOptions } from "make-fetch-happen";
|
||||||
|
import { dirname as pathDirname, join as pathJoin } from "path";
|
||||||
|
import { assert } from "tsafe";
|
||||||
|
import { promisify } from "util";
|
||||||
|
import { getProjectRoot } from "./getProjectRoot";
|
||||||
import { transformCodebase } from "./transformCodebase";
|
import { transformCodebase } from "./transformCodebase";
|
||||||
import { rm_rf, rm, rm_r } from "./rm";
|
import { unzip } from "./unzip";
|
||||||
|
|
||||||
/** assert url ends with .zip */
|
const exec = promisify(execCallback);
|
||||||
export function downloadAndUnzip(params: { url: string; destDirPath: string; pathOfDirToExtractInArchive?: string }) {
|
|
||||||
|
function hash(s: string) {
|
||||||
|
return createHash("sha256").update(s).digest("hex");
|
||||||
|
}
|
||||||
|
|
||||||
|
async function exists(path: string) {
|
||||||
|
try {
|
||||||
|
await stat(path);
|
||||||
|
return true;
|
||||||
|
} catch (error) {
|
||||||
|
if ((error as Error & { code: string }).code === "ENOENT") return false;
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get npm configuration as map
|
||||||
|
*/
|
||||||
|
async function getNmpConfig(): Promise<Record<string, string>> {
|
||||||
|
const { stdout } = await exec("npm config get", { encoding: "utf8" });
|
||||||
|
return stdout
|
||||||
|
.split("\n")
|
||||||
|
.filter(line => !line.startsWith(";"))
|
||||||
|
.map(line => line.trim())
|
||||||
|
.map(line => line.split("=", 2))
|
||||||
|
.reduce((cfg, [key, value]) => ({ ...cfg, [key]: value }), {});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get proxy configuration from npm config files. Note that we don't care about
|
||||||
|
* proxy config in env vars, because make-fetch-happen will do that for us.
|
||||||
|
*
|
||||||
|
* @returns proxy configuration
|
||||||
|
*/
|
||||||
|
async function getNpmProxyConfig(): Promise<Pick<FetchOptions, "proxy" | "noProxy">> {
|
||||||
|
const cfg = await getNmpConfig();
|
||||||
|
|
||||||
|
const proxy = cfg["https-proxy"] ?? cfg["proxy"];
|
||||||
|
const noProxy = cfg["noproxy"] ?? cfg["no-proxy"];
|
||||||
|
|
||||||
|
return { proxy, noProxy };
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function downloadAndUnzip(params: { url: string; destDirPath: string; pathOfDirToExtractInArchive?: string }) {
|
||||||
const { url, destDirPath, pathOfDirToExtractInArchive } = params;
|
const { url, destDirPath, pathOfDirToExtractInArchive } = params;
|
||||||
|
|
||||||
const tmpDirPath = pathJoin(destDirPath, "..", "tmp_xxKdOxnEdx");
|
const downloadHash = hash(JSON.stringify({ url })).substring(0, 15);
|
||||||
const zipFilePath = pathBasename(url);
|
const projectRoot = getProjectRoot();
|
||||||
|
const cacheRoot = process.env.XDG_CACHE_HOME ?? pathJoin(projectRoot, "node_modules", ".cache");
|
||||||
|
const zipFilePath = pathJoin(cacheRoot, "keycloakify", "zip", `_${downloadHash}.zip`);
|
||||||
|
const extractDirPath = pathJoin(cacheRoot, "keycloakify", "unzip", `_${downloadHash}`);
|
||||||
|
|
||||||
rm_rf(tmpDirPath);
|
if (!(await exists(zipFilePath))) {
|
||||||
|
const proxyOpts = await getNpmProxyConfig();
|
||||||
|
const response = await fetch(url, proxyOpts);
|
||||||
|
await mkdir(pathDirname(zipFilePath), { "recursive": true });
|
||||||
|
/**
|
||||||
|
* The correct way to fix this is to upgrade node-fetch beyond 3.2.5
|
||||||
|
* (see https://github.com/node-fetch/node-fetch/issues/1295#issuecomment-1144061991.)
|
||||||
|
* Unfortunately, octokit (a dependency of keycloakify) also uses node-fetch, and
|
||||||
|
* does not support node-fetch 3.x. So we stick around with this band-aid until
|
||||||
|
* octokit upgrades.
|
||||||
|
*/
|
||||||
|
response.body?.setMaxListeners(Number.MAX_VALUE);
|
||||||
|
assert(typeof response.body !== "undefined" && response.body != null);
|
||||||
|
await writeFile(zipFilePath, response.body);
|
||||||
|
}
|
||||||
|
|
||||||
fs.mkdirSync(tmpDirPath, { "recursive": true });
|
await unzip(zipFilePath, extractDirPath, pathOfDirToExtractInArchive);
|
||||||
|
|
||||||
execSync(`curl -L ${url} -o ${zipFilePath}`, { "cwd": tmpDirPath });
|
|
||||||
|
|
||||||
execSync(`unzip -o ${zipFilePath}${pathOfDirToExtractInArchive === undefined ? "" : ` "${pathOfDirToExtractInArchive}/**/*"`}`, {
|
|
||||||
"cwd": tmpDirPath,
|
|
||||||
});
|
|
||||||
|
|
||||||
rm(pathBasename(url), { "cwd": tmpDirPath });
|
|
||||||
|
|
||||||
transformCodebase({
|
transformCodebase({
|
||||||
"srcDirPath": pathOfDirToExtractInArchive === undefined ? tmpDirPath : pathJoin(tmpDirPath, pathOfDirToExtractInArchive),
|
"srcDirPath": extractDirPath,
|
||||||
destDirPath,
|
"destDirPath": destDirPath
|
||||||
});
|
});
|
||||||
|
|
||||||
rm_r(tmpDirPath);
|
|
||||||
}
|
}
|
||||||
|
@ -1,10 +1,17 @@
|
|||||||
import { getProjectRoot } from "./getProjectRoot";
|
import { getProjectRoot } from "./getProjectRoot";
|
||||||
import { join as pathJoin } from "path";
|
import { join as pathJoin } from "path";
|
||||||
import * as child_process from "child_process";
|
import { constants } from "fs";
|
||||||
import * as fs from "fs";
|
import { chmod, stat } from "fs/promises";
|
||||||
|
|
||||||
Object.entries<string>(JSON.parse(fs.readFileSync(pathJoin(getProjectRoot(), "package.json")).toString("utf8"))["bin"]).forEach(([, scriptPath]) =>
|
(async () => {
|
||||||
child_process.execSync(`chmod +x ${scriptPath}`, {
|
const { bin } = await import(pathJoin(getProjectRoot(), "package.json"));
|
||||||
"cwd": getProjectRoot(),
|
|
||||||
}),
|
const promises = Object.values<string>(bin).map(async scriptPath => {
|
||||||
);
|
const fullPath = pathJoin(getProjectRoot(), scriptPath);
|
||||||
|
const oldMode = (await stat(fullPath)).mode;
|
||||||
|
const newMode = oldMode | constants.S_IXUSR | constants.S_IXGRP | constants.S_IXOTH;
|
||||||
|
await chmod(fullPath, newMode);
|
||||||
|
});
|
||||||
|
|
||||||
|
await Promise.all(promises);
|
||||||
|
})();
|
||||||
|
94
src/bin/tools/jar.ts
Normal file
94
src/bin/tools/jar.ts
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
import { Readable, Transform } from "stream";
|
||||||
|
import { dirname, relative, sep } from "path";
|
||||||
|
import { createWriteStream } from "fs";
|
||||||
|
|
||||||
|
import walk from "./walk";
|
||||||
|
import zip, { type ZipSource } from "./zip";
|
||||||
|
import { mkdir } from "fs/promises";
|
||||||
|
import trimIndent from "./trimIndent";
|
||||||
|
|
||||||
|
type JarArgs = {
|
||||||
|
rootPath: string;
|
||||||
|
targetPath: string;
|
||||||
|
groupId: string;
|
||||||
|
artifactId: string;
|
||||||
|
version: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a jar archive, using the resources found at `rootPath` (a directory) and write the
|
||||||
|
* archive to `targetPath` (a file). Use `groupId`, `artifactId` and `version` to define
|
||||||
|
* the contents of the pom.properties file which is going to be added to the archive.
|
||||||
|
*/
|
||||||
|
export default async function jar({ groupId, artifactId, version, rootPath, targetPath }: JarArgs) {
|
||||||
|
const manifest: ZipSource = {
|
||||||
|
path: "META-INF/MANIFEST.MF",
|
||||||
|
data: Buffer.from(trimIndent`
|
||||||
|
Manifest-Version: 1.0
|
||||||
|
Archiver-Version: Plexus Archiver
|
||||||
|
Created-By: Keycloakify
|
||||||
|
Built-By: unknown
|
||||||
|
Build-Jdk: 19.0.0
|
||||||
|
`)
|
||||||
|
};
|
||||||
|
|
||||||
|
const pomProps: ZipSource = {
|
||||||
|
path: `META-INF/maven/${groupId}/${artifactId}/pom.properties`,
|
||||||
|
data: Buffer.from(trimIndent`# Generated by keycloakify
|
||||||
|
# ${new Date().toString()}
|
||||||
|
artifactId=${artifactId}
|
||||||
|
groupId=${groupId}
|
||||||
|
version=${version}
|
||||||
|
`)
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert every path entry to a ZipSource record, and when all records are
|
||||||
|
* processed, append records for MANIFEST.mf and pom.properties
|
||||||
|
*/
|
||||||
|
const pathToRecord = () =>
|
||||||
|
new Transform({
|
||||||
|
objectMode: true,
|
||||||
|
transform: function (fsPath, _, cb) {
|
||||||
|
const path = relative(rootPath, fsPath).split(sep).join("/");
|
||||||
|
this.push({ path, fsPath });
|
||||||
|
cb();
|
||||||
|
},
|
||||||
|
final: function () {
|
||||||
|
this.push(manifest);
|
||||||
|
this.push(pomProps);
|
||||||
|
this.push(null);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
await mkdir(dirname(targetPath), { recursive: true });
|
||||||
|
|
||||||
|
// Create an async pipeline, wait until everything is fully processed
|
||||||
|
await new Promise<void>((resolve, reject) => {
|
||||||
|
// walk all files in `rootPath` recursively
|
||||||
|
Readable.from(walk(rootPath))
|
||||||
|
// transform every path into a ZipSource object
|
||||||
|
.pipe(pathToRecord())
|
||||||
|
// let the zip lib convert all ZipSource objects into a byte stream
|
||||||
|
.pipe(zip())
|
||||||
|
// write that byte stream to targetPath
|
||||||
|
.pipe(createWriteStream(targetPath, { encoding: "binary" }))
|
||||||
|
.on("finish", () => resolve())
|
||||||
|
.on("error", e => reject(e));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Standalone usage, call e.g. `ts-node jar.ts dirWithSources some-jar.jar`
|
||||||
|
*/
|
||||||
|
if (require.main === module) {
|
||||||
|
const main = () =>
|
||||||
|
jar({
|
||||||
|
rootPath: process.argv[2],
|
||||||
|
targetPath: process.argv[3],
|
||||||
|
artifactId: process.env.ARTIFACT_ID ?? "artifact",
|
||||||
|
groupId: process.env.GROUP_ID ?? "group",
|
||||||
|
version: process.env.VERSION ?? "1.0.0"
|
||||||
|
});
|
||||||
|
main();
|
||||||
|
}
|
7
src/bin/tools/kebabCaseToSnakeCase.ts
Normal file
7
src/bin/tools/kebabCaseToSnakeCase.ts
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
import { capitalize } from "tsafe/capitalize";
|
||||||
|
|
||||||
|
export function kebabCaseToCamelCase(kebabCaseString: string): string {
|
||||||
|
const [first, ...rest] = kebabCaseString.split("-");
|
||||||
|
|
||||||
|
return [first, ...rest.map(capitalize)].join("");
|
||||||
|
}
|
27
src/bin/tools/logger.ts
Normal file
27
src/bin/tools/logger.ts
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
type LoggerOpts = {
|
||||||
|
force?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
type Logger = {
|
||||||
|
log: (message: string, opts?: LoggerOpts) => void;
|
||||||
|
warn: (message: string) => void;
|
||||||
|
error: (message: string) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getLogger = ({ isSilent }: { isSilent?: boolean } = {}): Logger => {
|
||||||
|
return {
|
||||||
|
log: (message, { force } = {}) => {
|
||||||
|
if (isSilent && !force) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(message);
|
||||||
|
},
|
||||||
|
warn: message => {
|
||||||
|
console.warn(message);
|
||||||
|
},
|
||||||
|
error: message => {
|
||||||
|
console.error(message);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
};
|
@ -19,7 +19,7 @@ export function listTagsFactory(params: { octokit: Octokit }) {
|
|||||||
owner,
|
owner,
|
||||||
repo,
|
repo,
|
||||||
per_page,
|
per_page,
|
||||||
"page": page++,
|
"page": page++
|
||||||
});
|
});
|
||||||
|
|
||||||
for (const branch of resp.data.map(({ name }) => name)) {
|
for (const branch of resp.data.map(({ name }) => name)) {
|
||||||
|
11
src/bin/tools/partitionPromiseSettledResults.ts
Normal file
11
src/bin/tools/partitionPromiseSettledResults.ts
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
export type PromiseSettledAndPartitioned<T> = [T[], any[]];
|
||||||
|
|
||||||
|
export function partitionPromiseSettledResults<T>() {
|
||||||
|
return [
|
||||||
|
([successes, failures]: PromiseSettledAndPartitioned<T>, item: PromiseSettledResult<T>) =>
|
||||||
|
item.status === "rejected"
|
||||||
|
? ([successes, [item.reason, ...failures]] as PromiseSettledAndPartitioned<T>)
|
||||||
|
: ([[item.value, ...successes], failures] as PromiseSettledAndPartitioned<T>),
|
||||||
|
[[], []] as PromiseSettledAndPartitioned<T>
|
||||||
|
] as const;
|
||||||
|
}
|
@ -1,31 +0,0 @@
|
|||||||
import { execSync } from "child_process";
|
|
||||||
|
|
||||||
function rmInternal(params: { pathToRemove: string; args: string | undefined; cwd: string | undefined }) {
|
|
||||||
const { pathToRemove, args, cwd } = params;
|
|
||||||
|
|
||||||
execSync(`rm ${args ? `-${args} ` : ""}${pathToRemove.replace(/ /g, "\\ ")}`, cwd !== undefined ? { cwd } : undefined);
|
|
||||||
}
|
|
||||||
|
|
||||||
export function rm(pathToRemove: string, options?: { cwd: string }) {
|
|
||||||
rmInternal({
|
|
||||||
pathToRemove,
|
|
||||||
"args": undefined,
|
|
||||||
"cwd": options?.cwd,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
export function rm_r(pathToRemove: string, options?: { cwd: string }) {
|
|
||||||
rmInternal({
|
|
||||||
pathToRemove,
|
|
||||||
"args": "r",
|
|
||||||
"cwd": options?.cwd,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
export function rm_rf(pathToRemove: string, options?: { cwd: string }) {
|
|
||||||
rmInternal({
|
|
||||||
pathToRemove,
|
|
||||||
"args": "rf",
|
|
||||||
"cwd": options?.cwd,
|
|
||||||
});
|
|
||||||
}
|
|
39
src/bin/tools/tee.ts
Normal file
39
src/bin/tools/tee.ts
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
import { PassThrough, Readable } from "stream";
|
||||||
|
|
||||||
|
export default function tee(input: Readable) {
|
||||||
|
const a = new PassThrough();
|
||||||
|
const b = new PassThrough();
|
||||||
|
|
||||||
|
let aFull = false;
|
||||||
|
let bFull = false;
|
||||||
|
|
||||||
|
a.setMaxListeners(Infinity);
|
||||||
|
|
||||||
|
a.on("drain", () => {
|
||||||
|
aFull = false;
|
||||||
|
if (!aFull && !bFull) input.resume();
|
||||||
|
});
|
||||||
|
b.on("drain", () => {
|
||||||
|
bFull = false;
|
||||||
|
if (!aFull && !bFull) input.resume();
|
||||||
|
});
|
||||||
|
|
||||||
|
input.on("error", e => {
|
||||||
|
a.emit("error", e);
|
||||||
|
b.emit("error", e);
|
||||||
|
});
|
||||||
|
|
||||||
|
input.on("data", chunk => {
|
||||||
|
aFull = !a.write(chunk);
|
||||||
|
bFull = !b.write(chunk);
|
||||||
|
|
||||||
|
if (aFull || bFull) input.pause();
|
||||||
|
});
|
||||||
|
|
||||||
|
input.on("end", () => {
|
||||||
|
a.end();
|
||||||
|
b.end();
|
||||||
|
});
|
||||||
|
|
||||||
|
return [a, b] as const;
|
||||||
|
}
|
@ -16,8 +16,8 @@ export function transformCodebase(params: { srcDirPath: string; destDirPath: str
|
|||||||
srcDirPath,
|
srcDirPath,
|
||||||
destDirPath,
|
destDirPath,
|
||||||
transformSourceCode = id<TransformSourceCode>(({ sourceCode }) => ({
|
transformSourceCode = id<TransformSourceCode>(({ sourceCode }) => ({
|
||||||
"modifiedSourceCode": sourceCode,
|
"modifiedSourceCode": sourceCode
|
||||||
})),
|
}))
|
||||||
} = params;
|
} = params;
|
||||||
|
|
||||||
for (const file_relative_path of crawl(srcDirPath)) {
|
for (const file_relative_path of crawl(srcDirPath)) {
|
||||||
@ -25,7 +25,7 @@ export function transformCodebase(params: { srcDirPath: string; destDirPath: str
|
|||||||
|
|
||||||
const transformSourceCodeResult = transformSourceCode({
|
const transformSourceCodeResult = transformSourceCode({
|
||||||
"sourceCode": fs.readFileSync(filePath),
|
"sourceCode": fs.readFileSync(filePath),
|
||||||
"filePath": path.join(srcDirPath, file_relative_path),
|
"filePath": path.join(srcDirPath, file_relative_path)
|
||||||
});
|
});
|
||||||
|
|
||||||
if (transformSourceCodeResult === undefined) {
|
if (transformSourceCodeResult === undefined) {
|
||||||
@ -33,14 +33,14 @@ export function transformCodebase(params: { srcDirPath: string; destDirPath: str
|
|||||||
}
|
}
|
||||||
|
|
||||||
fs.mkdirSync(path.dirname(path.join(destDirPath, file_relative_path)), {
|
fs.mkdirSync(path.dirname(path.join(destDirPath, file_relative_path)), {
|
||||||
"recursive": true,
|
"recursive": true
|
||||||
});
|
});
|
||||||
|
|
||||||
const { newFileName, modifiedSourceCode } = transformSourceCodeResult;
|
const { newFileName, modifiedSourceCode } = transformSourceCodeResult;
|
||||||
|
|
||||||
fs.writeFileSync(
|
fs.writeFileSync(
|
||||||
path.join(path.dirname(path.join(destDirPath, file_relative_path)), newFileName ?? path.basename(file_relative_path)),
|
path.join(path.dirname(path.join(destDirPath, file_relative_path)), newFileName ?? path.basename(file_relative_path)),
|
||||||
modifiedSourceCode,
|
modifiedSourceCode
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
51
src/bin/tools/trimIndent.ts
Normal file
51
src/bin/tools/trimIndent.ts
Normal file
@ -0,0 +1,51 @@
|
|||||||
|
/**
|
||||||
|
* Concatenate the string fragments and interpolated values
|
||||||
|
* to get a single string.
|
||||||
|
*/
|
||||||
|
function populateTemplate(strings: TemplateStringsArray, ...args: any[]) {
|
||||||
|
const chunks = [];
|
||||||
|
for (let i = 0; i < strings.length; i++) {
|
||||||
|
let lastStringLineLength = 0;
|
||||||
|
if (strings[i]) {
|
||||||
|
chunks.push(strings[i]);
|
||||||
|
// remember last indent of the string portion
|
||||||
|
lastStringLineLength = strings[i].split("\n").at(-1)?.length ?? 0;
|
||||||
|
}
|
||||||
|
if (args[i]) {
|
||||||
|
// if the interpolation value has newlines, indent the interpolation values
|
||||||
|
// using the last known string indent
|
||||||
|
chunks.push(args[i].replace(/([\r?\n])/g, "$1" + " ".repeat(lastStringLineLength)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return chunks.join("");
|
||||||
|
}
|
||||||
|
|
||||||
|
function trimIndentPrivate(removeEmptyLeadingAndTrailingLines: boolean, strings: TemplateStringsArray, ...args: any[]) {
|
||||||
|
// Remove initial and final newlines
|
||||||
|
let string = populateTemplate(strings, ...args);
|
||||||
|
if (removeEmptyLeadingAndTrailingLines) string = string.replace(/^[\r\n]/, "").replace(/[^\S\r\n]*[\r\n]$/, "");
|
||||||
|
const dents = string.match(/^([ \t])+/gm)?.map(s => s.length) ?? [];
|
||||||
|
// No dents? no change required
|
||||||
|
if (!dents || dents.length == 0) return string;
|
||||||
|
const minDent = Math.min(...dents);
|
||||||
|
// The min indentation is 0, no change needed
|
||||||
|
if (!minDent) return string;
|
||||||
|
const dedented = string.replace(new RegExp(`^${" ".repeat(minDent)}`, "gm"), "");
|
||||||
|
return dedented;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Shift all lines left by the *smallest* indentation level,
|
||||||
|
* and remove initial newline and all trailing spaces.
|
||||||
|
*/
|
||||||
|
export default function trimIndent(strings: TemplateStringsArray, ...args: any[]) {
|
||||||
|
return trimIndentPrivate(true, strings, ...args);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Shift all lines left by the *smallest* indentation level,
|
||||||
|
* and _keep_ initial newline and all trailing spaces.
|
||||||
|
*/
|
||||||
|
trimIndent.keepLeadingAndTrailingNewlines = function (strings: TemplateStringsArray, ...args: any[]) {
|
||||||
|
return trimIndentPrivate(false, strings, ...args);
|
||||||
|
};
|
92
src/bin/tools/unzip.ts
Normal file
92
src/bin/tools/unzip.ts
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
import fsp from "node:fs/promises";
|
||||||
|
import fs from "fs";
|
||||||
|
import path from "node:path";
|
||||||
|
import yauzl from "yauzl";
|
||||||
|
import stream from "node:stream";
|
||||||
|
import { promisify } from "node:util";
|
||||||
|
|
||||||
|
const pipeline = promisify(stream.pipeline);
|
||||||
|
|
||||||
|
async function pathExists(path: string) {
|
||||||
|
try {
|
||||||
|
await fsp.stat(path);
|
||||||
|
return true;
|
||||||
|
} catch (error) {
|
||||||
|
if ((error as { code: string }).code === "ENOENT") {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function unzip(file: string, targetFolder: string, unzipSubPath?: string) {
|
||||||
|
// add trailing slash to unzipSubPath and targetFolder
|
||||||
|
if (unzipSubPath && (!unzipSubPath.endsWith("/") || !unzipSubPath.endsWith("\\"))) {
|
||||||
|
unzipSubPath += "/";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!targetFolder.endsWith("/") || !targetFolder.endsWith("\\")) {
|
||||||
|
targetFolder += "/";
|
||||||
|
}
|
||||||
|
if (!fs.existsSync(targetFolder)) {
|
||||||
|
fs.mkdirSync(targetFolder, { recursive: true });
|
||||||
|
}
|
||||||
|
|
||||||
|
return new Promise<void>((resolve, reject) => {
|
||||||
|
yauzl.open(file, { lazyEntries: true }, async (err, zipfile) => {
|
||||||
|
if (err) {
|
||||||
|
reject(err);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
zipfile.readEntry();
|
||||||
|
|
||||||
|
zipfile.on("entry", async entry => {
|
||||||
|
if (unzipSubPath) {
|
||||||
|
// Skip files outside of the unzipSubPath
|
||||||
|
if (!entry.fileName.startsWith(unzipSubPath)) {
|
||||||
|
zipfile.readEntry();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove the unzipSubPath from the file name
|
||||||
|
entry.fileName = entry.fileName.substring(unzipSubPath.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
const target = path.join(targetFolder, entry.fileName);
|
||||||
|
|
||||||
|
// Directory file names end with '/'.
|
||||||
|
// Note that entries for directories themselves are optional.
|
||||||
|
// An entry's fileName implicitly requires its parent directories to exist.
|
||||||
|
if (/[\/\\]$/.test(target)) {
|
||||||
|
await fsp.mkdir(target, { recursive: true });
|
||||||
|
|
||||||
|
zipfile.readEntry();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Skip existing files
|
||||||
|
if (await pathExists(target)) {
|
||||||
|
zipfile.readEntry();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
zipfile.openReadStream(entry, async (err, readStream) => {
|
||||||
|
if (err) {
|
||||||
|
reject(err);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await pipeline(readStream, fs.createWriteStream(target));
|
||||||
|
|
||||||
|
zipfile.readEntry();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
zipfile.once("end", function () {
|
||||||
|
zipfile.close();
|
||||||
|
resolve();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
19
src/bin/tools/walk.ts
Normal file
19
src/bin/tools/walk.ts
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
import { readdir } from "fs/promises";
|
||||||
|
import { resolve } from "path";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Asynchronously and recursively walk a directory tree, yielding every file and directory
|
||||||
|
* found
|
||||||
|
*
|
||||||
|
* @param root the starting directory
|
||||||
|
* @returns AsyncGenerator
|
||||||
|
*/
|
||||||
|
export default async function* walk(root: string): AsyncGenerator<string, void, void> {
|
||||||
|
for (const entry of await readdir(root, { withFileTypes: true })) {
|
||||||
|
const absolutePath = resolve(root, entry.name);
|
||||||
|
if (entry.isDirectory()) {
|
||||||
|
yield absolutePath;
|
||||||
|
yield* walk(absolutePath);
|
||||||
|
} else yield absolutePath;
|
||||||
|
}
|
||||||
|
}
|
246
src/bin/tools/zip.ts
Normal file
246
src/bin/tools/zip.ts
Normal file
@ -0,0 +1,246 @@
|
|||||||
|
import { Transform, TransformOptions } from "stream";
|
||||||
|
import { createReadStream } from "fs";
|
||||||
|
import { stat } from "fs/promises";
|
||||||
|
import { Blob } from "buffer";
|
||||||
|
|
||||||
|
import { deflateBuffer, deflateStream } from "./deflate";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Zip source
|
||||||
|
* @property filename the name of the entry in the archie
|
||||||
|
* @property path of the source file, if the source is an actual file
|
||||||
|
* @property data the actual data buffer, if the source is constructed in-memory
|
||||||
|
*/
|
||||||
|
export type ZipSource = { path: string } & ({ fsPath: string } | { data: Buffer });
|
||||||
|
|
||||||
|
export type ZipRecord = {
|
||||||
|
path: string;
|
||||||
|
compression: "deflate" | undefined;
|
||||||
|
uncompressedSize: number;
|
||||||
|
compressedSize?: number;
|
||||||
|
crc32?: number;
|
||||||
|
offset?: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @returns the actual byte size of an string
|
||||||
|
*/
|
||||||
|
function utf8size(s: string) {
|
||||||
|
return new Blob([s]).size;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param record
|
||||||
|
* @returns a buffer representing a Zip local header
|
||||||
|
* @link https://en.wikipedia.org/wiki/ZIP_(file_format)#Local_file_header
|
||||||
|
*/
|
||||||
|
function localHeader(record: ZipRecord) {
|
||||||
|
const { path, compression, uncompressedSize } = record;
|
||||||
|
const filenameSize = utf8size(path);
|
||||||
|
const buf = Buffer.alloc(30 + filenameSize);
|
||||||
|
|
||||||
|
buf.writeUInt32LE(0x04_03_4b_50, 0); // local header signature
|
||||||
|
buf.writeUInt16LE(10, 4); // min version
|
||||||
|
// we write 0x08 because crc and compressed size are unknown at
|
||||||
|
buf.writeUInt16LE(0x08, 6); // general purpose bit flag
|
||||||
|
buf.writeUInt16LE(compression ? ({ "deflate": 8 } as const)[compression] : 0, 8);
|
||||||
|
buf.writeUInt16LE(0, 10); // modified time
|
||||||
|
buf.writeUInt16LE(0, 12); // modified date
|
||||||
|
buf.writeUInt32LE(0, 14); // crc unknown
|
||||||
|
buf.writeUInt32LE(0, 18); // compressed size unknown
|
||||||
|
buf.writeUInt32LE(uncompressedSize, 22);
|
||||||
|
buf.writeUInt16LE(filenameSize, 26);
|
||||||
|
buf.writeUInt16LE(0, 28); // extra field length
|
||||||
|
buf.write(path, 30, "utf-8");
|
||||||
|
|
||||||
|
return buf;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param record
|
||||||
|
* @returns a buffer representing a Zip central header
|
||||||
|
* @link https://en.wikipedia.org/wiki/ZIP_(file_format)#Central_directory_file_header
|
||||||
|
*/
|
||||||
|
function centralHeader(record: ZipRecord) {
|
||||||
|
const { path, compression, crc32, compressedSize, uncompressedSize, offset } = record;
|
||||||
|
const filenameSize = utf8size(path);
|
||||||
|
const buf = Buffer.alloc(46 + filenameSize);
|
||||||
|
const isFile = !path.endsWith("/");
|
||||||
|
|
||||||
|
if (typeof offset === "undefined") throw new Error("Illegal argument");
|
||||||
|
|
||||||
|
// we don't want to deal with possibly messed up file or directory
|
||||||
|
// permissions, so we ignore the original permissions
|
||||||
|
const externalAttr = isFile ? 0x81a40000 : 0x41ed0000;
|
||||||
|
|
||||||
|
buf.writeUInt32LE(0x0201_4b50, 0); // central header signature
|
||||||
|
buf.writeUInt16LE(10, 4); // version
|
||||||
|
buf.writeUInt16LE(10, 6); // min version
|
||||||
|
buf.writeUInt16LE(0, 8); // general purpose bit flag
|
||||||
|
buf.writeUInt16LE(compression ? ({ "deflate": 8 } as const)[compression] : 0, 10);
|
||||||
|
buf.writeUInt16LE(0, 12); // modified time
|
||||||
|
buf.writeUInt16LE(0, 14); // modified date
|
||||||
|
buf.writeUInt32LE(crc32 || 0, 16);
|
||||||
|
buf.writeUInt32LE(compressedSize || 0, 20);
|
||||||
|
buf.writeUInt32LE(uncompressedSize, 24);
|
||||||
|
buf.writeUInt16LE(filenameSize, 28);
|
||||||
|
buf.writeUInt16LE(0, 30); // extra field length
|
||||||
|
buf.writeUInt16LE(0, 32); // comment field length
|
||||||
|
buf.writeUInt16LE(0, 34); // disk number
|
||||||
|
buf.writeUInt16LE(0, 36); // internal
|
||||||
|
buf.writeUInt32LE(externalAttr, 38); // external
|
||||||
|
buf.writeUInt32LE(offset, 42); // offset where file starts
|
||||||
|
buf.write(path, 46, "utf-8");
|
||||||
|
|
||||||
|
return buf;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @returns a buffer representing an Zip End-Of-Central-Directory block
|
||||||
|
* @link https://en.wikipedia.org/wiki/ZIP_(file_format)#End_of_central_directory_record_(EOCD)
|
||||||
|
*/
|
||||||
|
function eocd({ offset, cdSize, nRecords }: { offset: number; cdSize: number; nRecords: number }) {
|
||||||
|
const buf = Buffer.alloc(22);
|
||||||
|
buf.writeUint32LE(0x06054b50, 0); // eocd signature
|
||||||
|
buf.writeUInt16LE(0, 4); // disc number
|
||||||
|
buf.writeUint16LE(0, 6); // disc where central directory starts
|
||||||
|
buf.writeUint16LE(nRecords, 8); // records on this disc
|
||||||
|
buf.writeUInt16LE(nRecords, 10); // records total
|
||||||
|
buf.writeUInt32LE(cdSize, 12); // byte size of cd
|
||||||
|
buf.writeUInt32LE(offset, 16); // cd offset
|
||||||
|
buf.writeUint16LE(0, 20); // comment length
|
||||||
|
|
||||||
|
return buf;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @returns a stream Transform, which reads a stream of ZipRecords and
|
||||||
|
* writes a bytestream
|
||||||
|
*/
|
||||||
|
export default function zip() {
|
||||||
|
/**
|
||||||
|
* This is called when the input stream of ZipSource items is finished.
|
||||||
|
* Will write central directory and end-of-central-direcotry blocks.
|
||||||
|
*/
|
||||||
|
const final = () => {
|
||||||
|
// write central directory
|
||||||
|
let cdSize = 0;
|
||||||
|
for (const record of records) {
|
||||||
|
const head = centralHeader(record);
|
||||||
|
zipTransform.push(head);
|
||||||
|
cdSize += head.length;
|
||||||
|
}
|
||||||
|
|
||||||
|
// write end-of-central-directory
|
||||||
|
zipTransform.push(eocd({ offset, cdSize, nRecords: records.length }));
|
||||||
|
// signal stream end
|
||||||
|
zipTransform.push(null);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Write a directory entry to the archive
|
||||||
|
* @param path
|
||||||
|
*/
|
||||||
|
const writeDir = async (path: string) => {
|
||||||
|
const record: ZipRecord = {
|
||||||
|
path: path + "/",
|
||||||
|
offset,
|
||||||
|
compression: undefined,
|
||||||
|
uncompressedSize: 0
|
||||||
|
};
|
||||||
|
const head = localHeader(record);
|
||||||
|
zipTransform.push(head);
|
||||||
|
records.push(record);
|
||||||
|
offset += head.length;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Write a file entry to the archive
|
||||||
|
* @param archivePath path of the file in archive
|
||||||
|
* @param fsPath path to file on filesystem
|
||||||
|
* @param size of the actual, uncompressed, file
|
||||||
|
*/
|
||||||
|
const writeFile = async (archivePath: string, fsPath: string, size: number) => {
|
||||||
|
const record: ZipRecord = {
|
||||||
|
path: archivePath,
|
||||||
|
offset,
|
||||||
|
compression: "deflate",
|
||||||
|
uncompressedSize: size
|
||||||
|
};
|
||||||
|
const head = localHeader(record);
|
||||||
|
zipTransform.push(head);
|
||||||
|
|
||||||
|
const { crc32, compressedSize } = await deflateStream(createReadStream(fsPath), chunk => zipTransform.push(chunk));
|
||||||
|
|
||||||
|
record.crc32 = crc32;
|
||||||
|
record.compressedSize = compressedSize;
|
||||||
|
records.push(record);
|
||||||
|
offset += head.length + compressedSize;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Write archive record based on filesystem file or directory
|
||||||
|
* @param archivePath path of item in archive
|
||||||
|
* @param fsPath path to item on filesystem
|
||||||
|
*/
|
||||||
|
const writeFromPath = async (archivePath: string, fsPath: string) => {
|
||||||
|
const fileStats = await stat(fsPath);
|
||||||
|
fileStats.isDirectory() ? await writeDir(archivePath) /**/ : await writeFile(archivePath, fsPath, fileStats.size) /**/;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Write archive record based on data in a buffer
|
||||||
|
* @param path
|
||||||
|
* @param data
|
||||||
|
*/
|
||||||
|
const writeFromBuffer = async (path: string, data: Buffer) => {
|
||||||
|
const { deflated, crc32 } = await deflateBuffer(data);
|
||||||
|
const record: ZipRecord = {
|
||||||
|
path,
|
||||||
|
compression: "deflate",
|
||||||
|
crc32,
|
||||||
|
uncompressedSize: data.length,
|
||||||
|
compressedSize: deflated.length,
|
||||||
|
offset
|
||||||
|
};
|
||||||
|
const head = localHeader(record);
|
||||||
|
zipTransform.push(head);
|
||||||
|
zipTransform.push(deflated);
|
||||||
|
records.push(record);
|
||||||
|
offset += head.length + deflated.length;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Write an archive record
|
||||||
|
* @param source
|
||||||
|
*/
|
||||||
|
const writeRecord = async (source: ZipSource) => {
|
||||||
|
if ("fsPath" in source) await writeFromPath(source.path, source.fsPath);
|
||||||
|
else if ("data" in source) await writeFromBuffer(source.path, source.data);
|
||||||
|
else throw new Error("Illegal argument " + typeof source + " " + JSON.stringify(source));
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The actual stream transform function
|
||||||
|
* @param source
|
||||||
|
* @param _ encoding, ignored
|
||||||
|
* @param cb
|
||||||
|
*/
|
||||||
|
const transform: TransformOptions["transform"] = async (source: ZipSource, _, cb) => {
|
||||||
|
await writeRecord(source);
|
||||||
|
cb();
|
||||||
|
};
|
||||||
|
|
||||||
|
/** offset and records keep local state during processing */
|
||||||
|
let offset = 0;
|
||||||
|
const records: ZipRecord[] = [];
|
||||||
|
|
||||||
|
const zipTransform = new Transform({
|
||||||
|
readableObjectMode: false,
|
||||||
|
writableObjectMode: true,
|
||||||
|
transform,
|
||||||
|
final
|
||||||
|
});
|
||||||
|
|
||||||
|
return zipTransform;
|
||||||
|
}
|
1
src/index.ts
Normal file
1
src/index.ts
Normal file
@ -0,0 +1 @@
|
|||||||
|
export { createKeycloakAdapter } from "keycloakify/lib/keycloakJsAdapter";
|
@ -1,34 +0,0 @@
|
|||||||
import React, { memo } from "react";
|
|
||||||
import Template from "./Template";
|
|
||||||
import type { KcProps } from "./KcProps";
|
|
||||||
import type { KcContextBase } from "../getKcContext/KcContextBase";
|
|
||||||
import type { I18n } from "../i18n";
|
|
||||||
|
|
||||||
const Error = memo(({ kcContext, i18n, ...props }: { kcContext: KcContextBase.Error; i18n: I18n } & KcProps) => {
|
|
||||||
const { message, client } = kcContext;
|
|
||||||
|
|
||||||
const { msg } = i18n;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Template
|
|
||||||
{...{ kcContext, i18n, ...props }}
|
|
||||||
doFetchDefaultThemeResources={true}
|
|
||||||
displayMessage={false}
|
|
||||||
headerNode={msg("errorTitle")}
|
|
||||||
formNode={
|
|
||||||
<div id="kc-error-message">
|
|
||||||
<p className="instruction">{message.summary}</p>
|
|
||||||
{client !== undefined && client.baseUrl !== undefined && (
|
|
||||||
<p>
|
|
||||||
<a id="backToApplication" href={client.baseUrl}>
|
|
||||||
{msg("backToApplication")}
|
|
||||||
</a>
|
|
||||||
</p>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
export default Error;
|
|
@ -1,51 +0,0 @@
|
|||||||
import React, { memo } from "react";
|
|
||||||
import Template from "./Template";
|
|
||||||
import type { KcProps } from "./KcProps";
|
|
||||||
import { assert } from "../tools/assert";
|
|
||||||
import type { KcContextBase } from "../getKcContext/KcContextBase";
|
|
||||||
import type { I18n } from "../i18n";
|
|
||||||
|
|
||||||
const Info = memo(({ kcContext, i18n, ...props }: { kcContext: KcContextBase.Info; i18n: I18n } & KcProps) => {
|
|
||||||
const { msgStr, msg } = i18n;
|
|
||||||
|
|
||||||
assert(kcContext.message !== undefined);
|
|
||||||
|
|
||||||
const { messageHeader, message, requiredActions, skipLink, pageRedirectUri, actionUri, client } = kcContext;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Template
|
|
||||||
{...{ kcContext, i18n, ...props }}
|
|
||||||
doFetchDefaultThemeResources={true}
|
|
||||||
displayMessage={false}
|
|
||||||
headerNode={messageHeader !== undefined ? <>{messageHeader}</> : <>{message.summary}</>}
|
|
||||||
formNode={
|
|
||||||
<div id="kc-info-message">
|
|
||||||
<p className="instruction">
|
|
||||||
{message.summary}
|
|
||||||
|
|
||||||
{requiredActions !== undefined && (
|
|
||||||
<b>{requiredActions.map(requiredAction => msgStr(`requiredAction.${requiredAction}` as const)).join(",")}</b>
|
|
||||||
)}
|
|
||||||
</p>
|
|
||||||
{!skipLink && pageRedirectUri !== undefined ? (
|
|
||||||
<p>
|
|
||||||
<a href={pageRedirectUri}>{msg("backToApplication")}</a>
|
|
||||||
</p>
|
|
||||||
) : actionUri !== undefined ? (
|
|
||||||
<p>
|
|
||||||
<a href={actionUri}>{msg("proceedWithAction")}</a>
|
|
||||||
</p>
|
|
||||||
) : (
|
|
||||||
client.baseUrl !== undefined && (
|
|
||||||
<p>
|
|
||||||
<a href={client.baseUrl}>{msg("backToApplication")}</a>
|
|
||||||
</p>
|
|
||||||
)
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
export default Info;
|
|
@ -1,81 +0,0 @@
|
|||||||
import React, { lazy, memo, Suspense } from "react";
|
|
||||||
import type { KcContextBase } from "../getKcContext/KcContextBase";
|
|
||||||
import type { KcProps } from "./KcProps";
|
|
||||||
import { __unsafe_useI18n as useI18n } from "../i18n";
|
|
||||||
import type { I18n } from "../i18n";
|
|
||||||
|
|
||||||
const Login = lazy(() => import("./Login"));
|
|
||||||
const Register = lazy(() => import("./Register"));
|
|
||||||
const RegisterUserProfile = lazy(() => import("./RegisterUserProfile"));
|
|
||||||
const Info = lazy(() => import("./Info"));
|
|
||||||
const Error = lazy(() => import("./Error"));
|
|
||||||
const LoginResetPassword = lazy(() => import("./LoginResetPassword"));
|
|
||||||
const LoginVerifyEmail = lazy(() => import("./LoginVerifyEmail"));
|
|
||||||
const Terms = lazy(() => import("./Terms"));
|
|
||||||
const LoginOtp = lazy(() => import("./LoginOtp"));
|
|
||||||
const LoginUpdatePassword = lazy(() => import("./LoginUpdatePassword"));
|
|
||||||
const LoginUpdateProfile = lazy(() => import("./LoginUpdateProfile"));
|
|
||||||
const LoginIdpLinkConfirm = lazy(() => import("./LoginIdpLinkConfirm"));
|
|
||||||
const LoginPageExpired = lazy(() => import("./LoginPageExpired"));
|
|
||||||
const LoginIdpLinkEmail = lazy(() => import("./LoginIdpLinkEmail"));
|
|
||||||
const LoginConfigTotp = lazy(() => import("./LoginConfigTotp"));
|
|
||||||
const LogoutConfirm = lazy(() => import("./LogoutConfirm"));
|
|
||||||
|
|
||||||
const KcApp = memo(({ kcContext, i18n: userProvidedI18n, ...props }: { kcContext: KcContextBase; i18n?: I18n } & KcProps) => {
|
|
||||||
const i18n = (function useClosure() {
|
|
||||||
const i18n = useI18n({
|
|
||||||
kcContext,
|
|
||||||
"extraMessages": {},
|
|
||||||
"doSkip": userProvidedI18n !== undefined,
|
|
||||||
});
|
|
||||||
|
|
||||||
return userProvidedI18n ?? i18n;
|
|
||||||
})();
|
|
||||||
|
|
||||||
if (i18n === null) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Suspense>
|
|
||||||
{(() => {
|
|
||||||
switch (kcContext.pageId) {
|
|
||||||
case "login.ftl":
|
|
||||||
return <Login {...{ kcContext, i18n, ...props }} />;
|
|
||||||
case "register.ftl":
|
|
||||||
return <Register {...{ kcContext, i18n, ...props }} />;
|
|
||||||
case "register-user-profile.ftl":
|
|
||||||
return <RegisterUserProfile {...{ kcContext, i18n, ...props }} />;
|
|
||||||
case "info.ftl":
|
|
||||||
return <Info {...{ kcContext, i18n, ...props }} />;
|
|
||||||
case "error.ftl":
|
|
||||||
return <Error {...{ kcContext, i18n, ...props }} />;
|
|
||||||
case "login-reset-password.ftl":
|
|
||||||
return <LoginResetPassword {...{ kcContext, i18n, ...props }} />;
|
|
||||||
case "login-verify-email.ftl":
|
|
||||||
return <LoginVerifyEmail {...{ kcContext, i18n, ...props }} />;
|
|
||||||
case "terms.ftl":
|
|
||||||
return <Terms {...{ kcContext, i18n, ...props }} />;
|
|
||||||
case "login-otp.ftl":
|
|
||||||
return <LoginOtp {...{ kcContext, i18n, ...props }} />;
|
|
||||||
case "login-update-password.ftl":
|
|
||||||
return <LoginUpdatePassword {...{ kcContext, i18n, ...props }} />;
|
|
||||||
case "login-update-profile.ftl":
|
|
||||||
return <LoginUpdateProfile {...{ kcContext, i18n, ...props }} />;
|
|
||||||
case "login-idp-link-confirm.ftl":
|
|
||||||
return <LoginIdpLinkConfirm {...{ kcContext, i18n, ...props }} />;
|
|
||||||
case "login-idp-link-email.ftl":
|
|
||||||
return <LoginIdpLinkEmail {...{ kcContext, i18n, ...props }} />;
|
|
||||||
case "login-page-expired.ftl":
|
|
||||||
return <LoginPageExpired {...{ kcContext, i18n, ...props }} />;
|
|
||||||
case "login-config-totp.ftl":
|
|
||||||
return <LoginConfigTotp {...{ kcContext, i18n, ...props }} />;
|
|
||||||
case "logout-confirm.ftl":
|
|
||||||
return <LogoutConfirm {...{ kcContext, i18n, ...props }} />;
|
|
||||||
}
|
|
||||||
})()}
|
|
||||||
</Suspense>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
export default KcApp;
|
|
@ -1,203 +0,0 @@
|
|||||||
import { allPropertiesValuesToUndefined } from "../tools/allPropertiesValuesToUndefined";
|
|
||||||
import { assert } from "tsafe/assert";
|
|
||||||
|
|
||||||
/** Class names can be provided as an array or separated by whitespace */
|
|
||||||
export type KcPropsGeneric<CssClasses extends string> = {
|
|
||||||
[key in CssClasses]: readonly string[] | string | undefined;
|
|
||||||
};
|
|
||||||
|
|
||||||
export type KcTemplateClassKey =
|
|
||||||
| "stylesCommon"
|
|
||||||
| "styles"
|
|
||||||
| "scripts"
|
|
||||||
| "kcHtmlClass"
|
|
||||||
| "kcLoginClass"
|
|
||||||
| "kcHeaderClass"
|
|
||||||
| "kcHeaderWrapperClass"
|
|
||||||
| "kcFormCardClass"
|
|
||||||
| "kcFormCardAccountClass"
|
|
||||||
| "kcFormHeaderClass"
|
|
||||||
| "kcLocaleWrapperClass"
|
|
||||||
| "kcContentWrapperClass"
|
|
||||||
| "kcLabelWrapperClass"
|
|
||||||
| "kcFormGroupClass"
|
|
||||||
| "kcResetFlowIcon"
|
|
||||||
| "kcFeedbackSuccessIcon"
|
|
||||||
| "kcFeedbackWarningIcon"
|
|
||||||
| "kcFeedbackErrorIcon"
|
|
||||||
| "kcFeedbackInfoIcon"
|
|
||||||
| "kcFormSocialAccountContentClass"
|
|
||||||
| "kcFormSocialAccountClass"
|
|
||||||
| "kcSignUpClass"
|
|
||||||
| "kcInfoAreaWrapperClass";
|
|
||||||
|
|
||||||
export type KcTemplateProps = KcPropsGeneric<KcTemplateClassKey>;
|
|
||||||
|
|
||||||
export const defaultKcTemplateProps = {
|
|
||||||
"stylesCommon": [
|
|
||||||
"node_modules/patternfly/dist/css/patternfly.min.css",
|
|
||||||
"node_modules/patternfly/dist/css/patternfly-additions.min.css",
|
|
||||||
"lib/zocial/zocial.css",
|
|
||||||
],
|
|
||||||
"styles": ["css/login.css"],
|
|
||||||
"scripts": [],
|
|
||||||
"kcHtmlClass": ["login-pf"],
|
|
||||||
"kcLoginClass": ["login-pf-page"],
|
|
||||||
"kcContentWrapperClass": ["row"],
|
|
||||||
"kcHeaderClass": ["login-pf-page-header"],
|
|
||||||
"kcHeaderWrapperClass": [],
|
|
||||||
"kcFormCardClass": ["card-pf"],
|
|
||||||
"kcFormCardAccountClass": ["login-pf-accounts"],
|
|
||||||
"kcFormSocialAccountClass": ["login-pf-social-section"],
|
|
||||||
"kcFormSocialAccountContentClass": ["col-xs-12", "col-sm-6"],
|
|
||||||
"kcFormHeaderClass": ["login-pf-header"],
|
|
||||||
"kcLocaleWrapperClass": [],
|
|
||||||
"kcFeedbackErrorIcon": ["pficon", "pficon-error-circle-o"],
|
|
||||||
"kcFeedbackWarningIcon": ["pficon", "pficon-warning-triangle-o"],
|
|
||||||
"kcFeedbackSuccessIcon": ["pficon", "pficon-ok"],
|
|
||||||
"kcFeedbackInfoIcon": ["pficon", "pficon-info"],
|
|
||||||
"kcResetFlowIcon": ["pficon", "pficon-arrow fa-2x"],
|
|
||||||
"kcFormGroupClass": ["form-group"],
|
|
||||||
"kcLabelWrapperClass": ["col-xs-12", "col-sm-12", "col-md-12", "col-lg-12"],
|
|
||||||
"kcSignUpClass": ["login-pf-signup"],
|
|
||||||
"kcInfoAreaWrapperClass": [],
|
|
||||||
} as const;
|
|
||||||
|
|
||||||
assert<typeof defaultKcTemplateProps extends KcTemplateProps ? true : false>();
|
|
||||||
|
|
||||||
/** Tu use if you don't want any default */
|
|
||||||
export const allClearKcTemplateProps = allPropertiesValuesToUndefined(defaultKcTemplateProps);
|
|
||||||
|
|
||||||
assert<typeof allClearKcTemplateProps extends KcTemplateProps ? true : false>();
|
|
||||||
|
|
||||||
export type KcProps = KcPropsGeneric<
|
|
||||||
| KcTemplateClassKey
|
|
||||||
| "kcLogoLink"
|
|
||||||
| "kcLogoClass"
|
|
||||||
| "kcContainerClass"
|
|
||||||
| "kcContentClass"
|
|
||||||
| "kcFeedbackAreaClass"
|
|
||||||
| "kcLocaleClass"
|
|
||||||
| "kcAlertIconClasserror"
|
|
||||||
| "kcFormAreaClass"
|
|
||||||
| "kcFormSocialAccountListClass"
|
|
||||||
| "kcFormSocialAccountDoubleListClass"
|
|
||||||
| "kcFormSocialAccountListLinkClass"
|
|
||||||
| "kcWebAuthnKeyIcon"
|
|
||||||
| "kcFormClass"
|
|
||||||
| "kcFormGroupErrorClass"
|
|
||||||
| "kcLabelClass"
|
|
||||||
| "kcInputClass"
|
|
||||||
| "kcInputErrorMessageClass"
|
|
||||||
| "kcInputWrapperClass"
|
|
||||||
| "kcFormOptionsClass"
|
|
||||||
| "kcFormButtonsClass"
|
|
||||||
| "kcFormSettingClass"
|
|
||||||
| "kcTextareaClass"
|
|
||||||
| "kcInfoAreaClass"
|
|
||||||
| "kcFormGroupHeader"
|
|
||||||
| "kcButtonClass"
|
|
||||||
| "kcButtonPrimaryClass"
|
|
||||||
| "kcButtonDefaultClass"
|
|
||||||
| "kcButtonLargeClass"
|
|
||||||
| "kcButtonBlockClass"
|
|
||||||
| "kcInputLargeClass"
|
|
||||||
| "kcSrOnlyClass"
|
|
||||||
| "kcSelectAuthListClass"
|
|
||||||
| "kcSelectAuthListItemClass"
|
|
||||||
| "kcSelectAuthListItemInfoClass"
|
|
||||||
| "kcSelectAuthListItemLeftClass"
|
|
||||||
| "kcSelectAuthListItemBodyClass"
|
|
||||||
| "kcSelectAuthListItemDescriptionClass"
|
|
||||||
| "kcSelectAuthListItemHeadingClass"
|
|
||||||
| "kcSelectAuthListItemHelpTextClass"
|
|
||||||
| "kcAuthenticatorDefaultClass"
|
|
||||||
| "kcAuthenticatorPasswordClass"
|
|
||||||
| "kcAuthenticatorOTPClass"
|
|
||||||
| "kcAuthenticatorWebAuthnClass"
|
|
||||||
| "kcAuthenticatorWebAuthnPasswordlessClass"
|
|
||||||
| "kcSelectOTPListClass"
|
|
||||||
| "kcSelectOTPListItemClass"
|
|
||||||
| "kcAuthenticatorOtpCircleClass"
|
|
||||||
| "kcSelectOTPItemHeadingClass"
|
|
||||||
| "kcFormOptionsWrapperClass"
|
|
||||||
>;
|
|
||||||
|
|
||||||
export const defaultKcProps = {
|
|
||||||
...defaultKcTemplateProps,
|
|
||||||
"kcLogoLink": "http://www.keycloak.org",
|
|
||||||
"kcLogoClass": "login-pf-brand",
|
|
||||||
"kcContainerClass": "container-fluid",
|
|
||||||
"kcContentClass": ["col-sm-8", "col-sm-offset-2", "col-md-6", "col-md-offset-3", "col-lg-6", "col-lg-offset-3"],
|
|
||||||
"kcFeedbackAreaClass": ["col-md-12"],
|
|
||||||
"kcLocaleClass": ["col-xs-12", "col-sm-1"],
|
|
||||||
"kcAlertIconClasserror": ["pficon", "pficon-error-circle-o"],
|
|
||||||
|
|
||||||
"kcFormAreaClass": ["col-sm-10", "col-sm-offset-1", "col-md-8", "col-md-offset-2", "col-lg-8", "col-lg-offset-2"],
|
|
||||||
"kcFormSocialAccountListClass": ["login-pf-social", "list-unstyled", "login-pf-social-all"],
|
|
||||||
"kcFormSocialAccountDoubleListClass": ["login-pf-social-double-col"],
|
|
||||||
"kcFormSocialAccountListLinkClass": ["login-pf-social-link"],
|
|
||||||
"kcWebAuthnKeyIcon": ["pficon", "pficon-key"],
|
|
||||||
|
|
||||||
"kcFormClass": ["form-horizontal"],
|
|
||||||
"kcFormGroupErrorClass": ["has-error"],
|
|
||||||
"kcLabelClass": ["control-label"],
|
|
||||||
"kcInputClass": ["form-control"],
|
|
||||||
"kcInputErrorMessageClass": ["pf-c-form__helper-text", "pf-m-error", "required", "kc-feedback-text"],
|
|
||||||
"kcInputWrapperClass": ["col-xs-12", "col-sm-12", "col-md-12", "col-lg-12"],
|
|
||||||
"kcFormOptionsClass": ["col-xs-12", "col-sm-12", "col-md-12", "col-lg-12"],
|
|
||||||
"kcFormButtonsClass": ["col-xs-12", "col-sm-12", "col-md-12", "col-lg-12"],
|
|
||||||
"kcFormSettingClass": ["login-pf-settings"],
|
|
||||||
"kcTextareaClass": ["form-control"],
|
|
||||||
|
|
||||||
"kcInfoAreaClass": ["col-xs-12", "col-sm-4", "col-md-4", "col-lg-5", "details"],
|
|
||||||
|
|
||||||
// user-profile grouping
|
|
||||||
"kcFormGroupHeader": ["pf-c-form__group"],
|
|
||||||
|
|
||||||
// css classes for form buttons main class used for all buttons
|
|
||||||
"kcButtonClass": ["btn"],
|
|
||||||
// classes defining priority of the button - primary or default (there is typically only one priority button for the form)
|
|
||||||
"kcButtonPrimaryClass": ["btn-primary"],
|
|
||||||
"kcButtonDefaultClass": ["btn-default"],
|
|
||||||
// classes defining size of the button
|
|
||||||
"kcButtonLargeClass": ["btn-lg"],
|
|
||||||
"kcButtonBlockClass": ["btn-block"],
|
|
||||||
|
|
||||||
// css classes for input
|
|
||||||
"kcInputLargeClass": ["input-lg"],
|
|
||||||
|
|
||||||
// css classes for form accessability
|
|
||||||
"kcSrOnlyClass": ["sr-only"],
|
|
||||||
|
|
||||||
// css classes for select-authenticator form
|
|
||||||
"kcSelectAuthListClass": ["list-group", "list-view-pf"],
|
|
||||||
"kcSelectAuthListItemClass": ["list-group-item", "list-view-pf-stacked"],
|
|
||||||
"kcSelectAuthListItemInfoClass": ["list-view-pf-main-info"],
|
|
||||||
"kcSelectAuthListItemLeftClass": ["list-view-pf-left"],
|
|
||||||
"kcSelectAuthListItemBodyClass": ["list-view-pf-body"],
|
|
||||||
"kcSelectAuthListItemDescriptionClass": ["list-view-pf-description"],
|
|
||||||
"kcSelectAuthListItemHeadingClass": ["list-group-item-heading"],
|
|
||||||
"kcSelectAuthListItemHelpTextClass": ["list-group-item-text"],
|
|
||||||
|
|
||||||
// css classes for the authenticators
|
|
||||||
"kcAuthenticatorDefaultClass": ["fa", "list-view-pf-icon-lg"],
|
|
||||||
"kcAuthenticatorPasswordClass": ["fa", "fa-unlock list-view-pf-icon-lg"],
|
|
||||||
"kcAuthenticatorOTPClass": ["fa", "fa-mobile", "list-view-pf-icon-lg"],
|
|
||||||
"kcAuthenticatorWebAuthnClass": ["fa", "fa-key", "list-view-pf-icon-lg"],
|
|
||||||
"kcAuthenticatorWebAuthnPasswordlessClass": ["fa", "fa-key", "list-view-pf-icon-lg"],
|
|
||||||
|
|
||||||
//css classes for the OTP Login Form
|
|
||||||
"kcSelectOTPListClass": ["card-pf", "card-pf-view", "card-pf-view-select", "card-pf-view-single-select"],
|
|
||||||
"kcSelectOTPListItemClass": ["card-pf-body", "card-pf-top-element"],
|
|
||||||
"kcAuthenticatorOtpCircleClass": ["fa", "fa-mobile", "card-pf-icon-circle"],
|
|
||||||
"kcSelectOTPItemHeadingClass": ["card-pf-title", "text-center"],
|
|
||||||
"kcFormOptionsWrapperClass": [],
|
|
||||||
} as const;
|
|
||||||
|
|
||||||
assert<typeof defaultKcProps extends KcProps ? true : false>();
|
|
||||||
|
|
||||||
/** Tu use if you don't want any default */
|
|
||||||
export const allClearKcProps = allPropertiesValuesToUndefined(defaultKcProps);
|
|
||||||
|
|
||||||
assert<typeof allClearKcProps extends KcProps ? true : false>();
|
|
@ -1,195 +0,0 @@
|
|||||||
import React, { useState, memo } from "react";
|
|
||||||
import Template from "./Template";
|
|
||||||
import type { KcProps } from "./KcProps";
|
|
||||||
import type { KcContextBase } from "../getKcContext/KcContextBase";
|
|
||||||
import { useCssAndCx } from "tss-react";
|
|
||||||
import { useConstCallback } from "powerhooks/useConstCallback";
|
|
||||||
import type { FormEventHandler } from "react";
|
|
||||||
import type { I18n } from "../i18n";
|
|
||||||
|
|
||||||
const Login = memo(({ kcContext, i18n, ...props }: { kcContext: KcContextBase.Login; i18n: I18n } & KcProps) => {
|
|
||||||
const { social, realm, url, usernameEditDisabled, login, auth, registrationDisabled } = kcContext;
|
|
||||||
|
|
||||||
const { msg, msgStr } = i18n;
|
|
||||||
|
|
||||||
const { cx } = useCssAndCx();
|
|
||||||
|
|
||||||
const [isLoginButtonDisabled, setIsLoginButtonDisabled] = useState(false);
|
|
||||||
|
|
||||||
const onSubmit = useConstCallback<FormEventHandler<HTMLFormElement>>(e => {
|
|
||||||
e.preventDefault();
|
|
||||||
|
|
||||||
setIsLoginButtonDisabled(true);
|
|
||||||
|
|
||||||
const formElement = e.target as HTMLFormElement;
|
|
||||||
|
|
||||||
//NOTE: Even if we login with email Keycloak expect username and password in
|
|
||||||
//the POST request.
|
|
||||||
formElement.querySelector("input[name='email']")?.setAttribute("name", "username");
|
|
||||||
|
|
||||||
formElement.submit();
|
|
||||||
});
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Template
|
|
||||||
{...{ kcContext, i18n, ...props }}
|
|
||||||
doFetchDefaultThemeResources={true}
|
|
||||||
displayInfo={social.displayInfo}
|
|
||||||
displayWide={realm.password && social.providers !== undefined}
|
|
||||||
headerNode={msg("doLogIn")}
|
|
||||||
formNode={
|
|
||||||
<div id="kc-form" className={cx(realm.password && social.providers !== undefined && props.kcContentWrapperClass)}>
|
|
||||||
<div
|
|
||||||
id="kc-form-wrapper"
|
|
||||||
className={cx(realm.password && social.providers && [props.kcFormSocialAccountContentClass, props.kcFormSocialAccountClass])}
|
|
||||||
>
|
|
||||||
{realm.password && (
|
|
||||||
<form id="kc-form-login" onSubmit={onSubmit} action={url.loginAction} method="post">
|
|
||||||
<div className={cx(props.kcFormGroupClass)}>
|
|
||||||
{(() => {
|
|
||||||
const label = !realm.loginWithEmailAllowed
|
|
||||||
? "username"
|
|
||||||
: realm.registrationEmailAsUsername
|
|
||||||
? "email"
|
|
||||||
: "usernameOrEmail";
|
|
||||||
|
|
||||||
const autoCompleteHelper: typeof label = label === "usernameOrEmail" ? "username" : label;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<label htmlFor={autoCompleteHelper} className={cx(props.kcLabelClass)}>
|
|
||||||
{msg(label)}
|
|
||||||
</label>
|
|
||||||
<input
|
|
||||||
tabIndex={1}
|
|
||||||
id={autoCompleteHelper}
|
|
||||||
className={cx(props.kcInputClass)}
|
|
||||||
//NOTE: This is used by Google Chrome auto fill so we use it to tell
|
|
||||||
//the browser how to pre fill the form but before submit we put it back
|
|
||||||
//to username because it is what keycloak expects.
|
|
||||||
name={autoCompleteHelper}
|
|
||||||
defaultValue={login.username ?? ""}
|
|
||||||
type="text"
|
|
||||||
{...(usernameEditDisabled
|
|
||||||
? { "disabled": true }
|
|
||||||
: {
|
|
||||||
"autoFocus": true,
|
|
||||||
"autoComplete": "off",
|
|
||||||
})}
|
|
||||||
/>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
})()}
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcFormGroupClass)}>
|
|
||||||
<label htmlFor="password" className={cx(props.kcLabelClass)}>
|
|
||||||
{msg("password")}
|
|
||||||
</label>
|
|
||||||
<input
|
|
||||||
tabIndex={2}
|
|
||||||
id="password"
|
|
||||||
className={cx(props.kcInputClass)}
|
|
||||||
name="password"
|
|
||||||
type="password"
|
|
||||||
autoComplete="off"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcFormGroupClass, props.kcFormSettingClass)}>
|
|
||||||
<div id="kc-form-options">
|
|
||||||
{realm.rememberMe && !usernameEditDisabled && (
|
|
||||||
<div className="checkbox">
|
|
||||||
<label>
|
|
||||||
<input
|
|
||||||
tabIndex={3}
|
|
||||||
id="rememberMe"
|
|
||||||
name="rememberMe"
|
|
||||||
type="checkbox"
|
|
||||||
{...(login.rememberMe
|
|
||||||
? {
|
|
||||||
"checked": true,
|
|
||||||
}
|
|
||||||
: {})}
|
|
||||||
/>
|
|
||||||
{msg("rememberMe")}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcFormOptionsWrapperClass)}>
|
|
||||||
{realm.resetPasswordAllowed && (
|
|
||||||
<span>
|
|
||||||
<a tabIndex={5} href={url.loginResetCredentialsUrl}>
|
|
||||||
{msg("doForgotPassword")}
|
|
||||||
</a>
|
|
||||||
</span>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div id="kc-form-buttons" className={cx(props.kcFormGroupClass)}>
|
|
||||||
<input
|
|
||||||
type="hidden"
|
|
||||||
id="id-hidden-input"
|
|
||||||
name="credentialId"
|
|
||||||
{...(auth?.selectedCredential !== undefined
|
|
||||||
? {
|
|
||||||
"value": auth.selectedCredential,
|
|
||||||
}
|
|
||||||
: {})}
|
|
||||||
/>
|
|
||||||
<input
|
|
||||||
tabIndex={4}
|
|
||||||
className={cx(
|
|
||||||
props.kcButtonClass,
|
|
||||||
props.kcButtonPrimaryClass,
|
|
||||||
props.kcButtonBlockClass,
|
|
||||||
props.kcButtonLargeClass,
|
|
||||||
)}
|
|
||||||
name="login"
|
|
||||||
id="kc-login"
|
|
||||||
type="submit"
|
|
||||||
value={msgStr("doLogIn")}
|
|
||||||
disabled={isLoginButtonDisabled}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
{realm.password && social.providers !== undefined && (
|
|
||||||
<div id="kc-social-providers" className={cx(props.kcFormSocialAccountContentClass, props.kcFormSocialAccountClass)}>
|
|
||||||
<ul
|
|
||||||
className={cx(
|
|
||||||
props.kcFormSocialAccountListClass,
|
|
||||||
social.providers.length > 4 && props.kcFormSocialAccountDoubleListClass,
|
|
||||||
)}
|
|
||||||
>
|
|
||||||
{social.providers.map(p => (
|
|
||||||
<li key={p.providerId} className={cx(props.kcFormSocialAccountListLinkClass)}>
|
|
||||||
<a href={p.loginUrl} id={`zocial-${p.alias}`} className={cx("zocial", p.providerId)}>
|
|
||||||
<span>{p.displayName}</span>
|
|
||||||
</a>
|
|
||||||
</li>
|
|
||||||
))}
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
infoNode={
|
|
||||||
realm.password &&
|
|
||||||
realm.registrationAllowed &&
|
|
||||||
!registrationDisabled && (
|
|
||||||
<div id="kc-registration">
|
|
||||||
<span>
|
|
||||||
{msg("noAccount")}
|
|
||||||
<a tabIndex={6} href={url.registrationUrl}>
|
|
||||||
{msg("doRegister")}
|
|
||||||
</a>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
export default Login;
|
|
@ -1,186 +0,0 @@
|
|||||||
import React, { memo } from "react";
|
|
||||||
import Template from "./Template";
|
|
||||||
import type { KcProps } from "./KcProps";
|
|
||||||
import type { KcContextBase } from "../getKcContext/KcContextBase";
|
|
||||||
import { useCssAndCx } from "tss-react";
|
|
||||||
import type { I18n } from "../i18n";
|
|
||||||
|
|
||||||
const LoginConfigTotp = memo(({ kcContext, i18n, ...props }: { kcContext: KcContextBase.LoginConfigTotp; i18n: I18n } & KcProps) => {
|
|
||||||
const { url, isAppInitiatedAction, totp, mode, messagesPerField } = kcContext;
|
|
||||||
|
|
||||||
const { cx } = useCssAndCx();
|
|
||||||
|
|
||||||
const { msg, msgStr } = i18n;
|
|
||||||
|
|
||||||
const algToKeyUriAlg: Record<KcContextBase.LoginConfigTotp["totp"]["policy"]["algorithm"], string> = {
|
|
||||||
HmacSHA1: "SHA1",
|
|
||||||
HmacSHA256: "SHA256",
|
|
||||||
HmacSHA512: "SHA512",
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Template
|
|
||||||
{...{ kcContext, i18n, ...props }}
|
|
||||||
doFetchDefaultThemeResources={true}
|
|
||||||
headerNode={msg("loginTotpTitle")}
|
|
||||||
formNode={
|
|
||||||
<>
|
|
||||||
<ol id="kc-totp-settings">
|
|
||||||
<li>
|
|
||||||
<p>{msg("loginTotpStep1")}</p>
|
|
||||||
|
|
||||||
<ul id="kc-totp-supported-apps">
|
|
||||||
{totp.policy.supportedApplications.map(app => (
|
|
||||||
<li>{app}</li>
|
|
||||||
))}
|
|
||||||
</ul>
|
|
||||||
</li>
|
|
||||||
|
|
||||||
{mode && mode == "manual" ? (
|
|
||||||
<>
|
|
||||||
<li>
|
|
||||||
<p>{msg("loginTotpManualStep2")}</p>
|
|
||||||
<p>
|
|
||||||
<span id="kc-totp-secret-key">{totp.totpSecretEncoded}</span>
|
|
||||||
</p>
|
|
||||||
<p>
|
|
||||||
<a href={totp.qrUrl} id="mode-barcode">
|
|
||||||
{msg("loginTotpScanBarcode")}
|
|
||||||
</a>
|
|
||||||
</p>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<p>{msg("loginTotpManualStep3")}</p>
|
|
||||||
<p>
|
|
||||||
<ul>
|
|
||||||
<li id="kc-totp-type">
|
|
||||||
{msg("loginTotpType")}: {msg(`loginTotp.${totp.policy.type}`)}
|
|
||||||
</li>
|
|
||||||
<li id="kc-totp-algorithm">
|
|
||||||
{msg("loginTotpAlgorithm")}: {algToKeyUriAlg?.[totp.policy.algorithm] ?? totp.policy.algorithm}
|
|
||||||
</li>
|
|
||||||
<li id="kc-totp-digits">
|
|
||||||
{msg("loginTotpDigits")}: {totp.policy.digits}
|
|
||||||
</li>
|
|
||||||
{totp.policy.type === "totp" ? (
|
|
||||||
<li id="kc-totp-period">
|
|
||||||
{msg("loginTotpInterval")}: {totp.policy.period}
|
|
||||||
</li>
|
|
||||||
) : (
|
|
||||||
<li id="kc-totp-counter">
|
|
||||||
{msg("loginTotpCounter")}: {totp.policy.initialCounter}
|
|
||||||
</li>
|
|
||||||
)}
|
|
||||||
</ul>
|
|
||||||
</p>
|
|
||||||
</li>
|
|
||||||
</>
|
|
||||||
) : (
|
|
||||||
<li>
|
|
||||||
<p>{msg("loginTotpStep2")}</p>
|
|
||||||
<img id="kc-totp-secret-qr-code" src={`data:image/png;base64, ${totp.totpSecretQrCode}`} alt="Figure: Barcode" />
|
|
||||||
<br />
|
|
||||||
<p>
|
|
||||||
<a href={totp.manualUrl} id="mode-manual">
|
|
||||||
{msg("loginTotpUnableToScan")}
|
|
||||||
</a>
|
|
||||||
</p>
|
|
||||||
</li>
|
|
||||||
)}
|
|
||||||
<li>
|
|
||||||
<p>{msg("loginTotpStep3")}</p>
|
|
||||||
<p>{msg("loginTotpStep3DeviceName")}</p>
|
|
||||||
</li>
|
|
||||||
</ol>
|
|
||||||
|
|
||||||
<form action={url.loginAction} className={cx(props.kcFormClass)} id="kc-totp-settings-form" method="post">
|
|
||||||
<div className={cx(props.kcFormGroupClass)}>
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<label htmlFor="totp" className={cx(props.kcLabelClass)}>
|
|
||||||
{msg("authenticatorCode")}
|
|
||||||
</label>{" "}
|
|
||||||
<span className="required">*</span>
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
id="totp"
|
|
||||||
name="totp"
|
|
||||||
autoComplete="off"
|
|
||||||
className={cx(props.kcInputClass)}
|
|
||||||
aria-invalid={messagesPerField.existsError("totp")}
|
|
||||||
/>
|
|
||||||
|
|
||||||
{messagesPerField.existsError("totp") && (
|
|
||||||
<span id="input-error-otp-code" className={cx(props.kcInputErrorMessageClass)} aria-live="polite">
|
|
||||||
{messagesPerField.get("totp")}
|
|
||||||
</span>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
<input type="hidden" id="totpSecret" name="totpSecret" value={totp.totpSecret} />
|
|
||||||
{mode && <input type="hidden" id="mode" value={mode} />}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className={cx(props.kcFormGroupClass)}>
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<label htmlFor="userLabel" className={cx(props.kcLabelClass)}>
|
|
||||||
{msg("loginTotpDeviceName")}
|
|
||||||
</label>{" "}
|
|
||||||
{totp.otpCredentials.length >= 1 && <span className="required">*</span>}
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
id="userLabel"
|
|
||||||
name="userLabel"
|
|
||||||
autoComplete="off"
|
|
||||||
className={cx(props.kcInputClass)}
|
|
||||||
aria-invalid={messagesPerField.existsError("userLabel")}
|
|
||||||
/>
|
|
||||||
{messagesPerField.existsError("userLabel") && (
|
|
||||||
<span id="input-error-otp-label" className={cx(props.kcInputErrorMessageClass)} aria-live="polite">
|
|
||||||
{messagesPerField.get("userLabel")}
|
|
||||||
</span>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{isAppInitiatedAction ? (
|
|
||||||
<>
|
|
||||||
<input
|
|
||||||
type="submit"
|
|
||||||
className={cx(props.kcButtonClass, props.kcButtonPrimaryClass, props.kcButtonLargeClass)}
|
|
||||||
id="saveTOTPBtn"
|
|
||||||
value={msgStr("doSubmit")}
|
|
||||||
/>
|
|
||||||
<button
|
|
||||||
type="submit"
|
|
||||||
className={cx(
|
|
||||||
props.kcButtonClass,
|
|
||||||
props.kcButtonDefaultClass,
|
|
||||||
props.kcButtonLargeClass,
|
|
||||||
props.kcButtonLargeClass,
|
|
||||||
)}
|
|
||||||
id="cancelTOTPBtn"
|
|
||||||
name="cancel-aia"
|
|
||||||
value="true"
|
|
||||||
>
|
|
||||||
${msg("doCancel")}
|
|
||||||
</button>
|
|
||||||
</>
|
|
||||||
) : (
|
|
||||||
<input
|
|
||||||
type="submit"
|
|
||||||
className={cx(props.kcButtonClass, props.kcButtonPrimaryClass, props.kcButtonLargeClass)}
|
|
||||||
id="saveTOTPBtn"
|
|
||||||
value={msgStr("doSubmit")}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</form>
|
|
||||||
</>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
export default LoginConfigTotp;
|
|
@ -1,48 +0,0 @@
|
|||||||
import React, { memo } from "react";
|
|
||||||
import Template from "./Template";
|
|
||||||
import type { KcProps } from "./KcProps";
|
|
||||||
import type { KcContextBase } from "../getKcContext/KcContextBase";
|
|
||||||
import { useCssAndCx } from "tss-react";
|
|
||||||
import type { I18n } from "../i18n";
|
|
||||||
|
|
||||||
const LoginIdpLinkConfirm = memo(({ kcContext, i18n, ...props }: { kcContext: KcContextBase.LoginIdpLinkConfirm; i18n: I18n } & KcProps) => {
|
|
||||||
const { url, idpAlias } = kcContext;
|
|
||||||
|
|
||||||
const { msg } = i18n;
|
|
||||||
|
|
||||||
const { cx } = useCssAndCx();
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Template
|
|
||||||
{...{ kcContext, i18n, ...props }}
|
|
||||||
doFetchDefaultThemeResources={true}
|
|
||||||
headerNode={msg("confirmLinkIdpTitle")}
|
|
||||||
formNode={
|
|
||||||
<form id="kc-register-form" action={url.loginAction} method="post">
|
|
||||||
<div className={cx(props.kcFormGroupClass)}>
|
|
||||||
<button
|
|
||||||
type="submit"
|
|
||||||
className={cx(props.kcButtonClass, props.kcButtonDefaultClass, props.kcButtonBlockClass, props.kcButtonLargeClass)}
|
|
||||||
name="submitAction"
|
|
||||||
id="updateProfile"
|
|
||||||
value="updateProfile"
|
|
||||||
>
|
|
||||||
{msg("confirmLinkIdpReviewProfile")}
|
|
||||||
</button>
|
|
||||||
<button
|
|
||||||
type="submit"
|
|
||||||
className={cx(props.kcButtonClass, props.kcButtonDefaultClass, props.kcButtonBlockClass, props.kcButtonLargeClass)}
|
|
||||||
name="submitAction"
|
|
||||||
id="linkAccount"
|
|
||||||
value="linkAccount"
|
|
||||||
>
|
|
||||||
{msg("confirmLinkIdpContinue", idpAlias)}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
export default LoginIdpLinkConfirm;
|
|
@ -1,34 +0,0 @@
|
|||||||
import React, { memo } from "react";
|
|
||||||
import Template from "./Template";
|
|
||||||
import type { KcProps } from "./KcProps";
|
|
||||||
import type { KcContextBase } from "../getKcContext/KcContextBase";
|
|
||||||
import type { I18n } from "../i18n";
|
|
||||||
|
|
||||||
const LoginIdpLinkEmail = memo(({ kcContext, i18n, ...props }: { kcContext: KcContextBase.LoginIdpLinkEmail; i18n: I18n } & KcProps) => {
|
|
||||||
const { url, realm, brokerContext, idpAlias } = kcContext;
|
|
||||||
|
|
||||||
const { msg } = i18n;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Template
|
|
||||||
{...{ kcContext, i18n, ...props }}
|
|
||||||
doFetchDefaultThemeResources={true}
|
|
||||||
headerNode={msg("emailLinkIdpTitle", idpAlias)}
|
|
||||||
formNode={
|
|
||||||
<>
|
|
||||||
<p id="instruction1" className="instruction">
|
|
||||||
{msg("emailLinkIdp1", idpAlias, brokerContext.username, realm.displayName)}
|
|
||||||
</p>
|
|
||||||
<p id="instruction2" className="instruction">
|
|
||||||
{msg("emailLinkIdp2")} <a href={url.loginAction}>{msg("doClickHere")}</a> {msg("emailLinkIdp3")}
|
|
||||||
</p>
|
|
||||||
<p id="instruction3" className="instruction">
|
|
||||||
{msg("emailLinkIdp4")} <a href={url.loginAction}>{msg("doClickHere")}</a> {msg("emailLinkIdp5")}
|
|
||||||
</p>
|
|
||||||
</>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
export default LoginIdpLinkEmail;
|
|
@ -1,113 +0,0 @@
|
|||||||
import React, { useEffect, memo } from "react";
|
|
||||||
import Template from "./Template";
|
|
||||||
import type { KcProps } from "./KcProps";
|
|
||||||
import type { KcContextBase } from "../getKcContext/KcContextBase";
|
|
||||||
import { headInsert } from "../tools/headInsert";
|
|
||||||
import { pathJoin } from "../../bin/tools/pathJoin";
|
|
||||||
import { useCssAndCx } from "tss-react";
|
|
||||||
import type { I18n } from "../i18n";
|
|
||||||
|
|
||||||
const LoginOtp = memo(({ kcContext, i18n, ...props }: { kcContext: KcContextBase.LoginOtp; i18n: I18n } & KcProps) => {
|
|
||||||
const { otpLogin, url } = kcContext;
|
|
||||||
|
|
||||||
const { cx } = useCssAndCx();
|
|
||||||
|
|
||||||
const { msg, msgStr } = i18n;
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
let isCleanedUp = false;
|
|
||||||
|
|
||||||
headInsert({
|
|
||||||
"type": "javascript",
|
|
||||||
"src": pathJoin(kcContext.url.resourcesCommonPath, "node_modules/jquery/dist/jquery.min.js"),
|
|
||||||
}).then(() => {
|
|
||||||
if (isCleanedUp) return;
|
|
||||||
|
|
||||||
evaluateInlineScript();
|
|
||||||
});
|
|
||||||
|
|
||||||
return () => {
|
|
||||||
isCleanedUp = true;
|
|
||||||
};
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Template
|
|
||||||
{...{ kcContext, i18n, ...props }}
|
|
||||||
doFetchDefaultThemeResources={true}
|
|
||||||
headerNode={msg("doLogIn")}
|
|
||||||
formNode={
|
|
||||||
<form id="kc-otp-login-form" className={cx(props.kcFormClass)} action={url.loginAction} method="post">
|
|
||||||
{otpLogin.userOtpCredentials.length > 1 && (
|
|
||||||
<div className={cx(props.kcFormGroupClass)}>
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
{otpLogin.userOtpCredentials.map(otpCredential => (
|
|
||||||
<div key={otpCredential.id} className={cx(props.kcSelectOTPListClass)}>
|
|
||||||
<input type="hidden" value="${otpCredential.id}" />
|
|
||||||
<div className={cx(props.kcSelectOTPListItemClass)}>
|
|
||||||
<span className={cx(props.kcAuthenticatorOtpCircleClass)} />
|
|
||||||
<h2 className={cx(props.kcSelectOTPItemHeadingClass)}>{otpCredential.userLabel}</h2>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
<div className={cx(props.kcFormGroupClass)}>
|
|
||||||
<div className={cx(props.kcLabelWrapperClass)}>
|
|
||||||
<label htmlFor="otp" className={cx(props.kcLabelClass)}>
|
|
||||||
{msg("loginOtpOneTime")}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<input id="otp" name="otp" autoComplete="off" type="text" className={cx(props.kcInputClass)} autoFocus />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className={cx(props.kcFormGroupClass)}>
|
|
||||||
<div id="kc-form-options" className={cx(props.kcFormOptionsClass)}>
|
|
||||||
<div className={cx(props.kcFormOptionsWrapperClass)} />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div id="kc-form-buttons" className={cx(props.kcFormButtonsClass)}>
|
|
||||||
<input
|
|
||||||
className={cx(props.kcButtonClass, props.kcButtonPrimaryClass, props.kcButtonBlockClass, props.kcButtonLargeClass)}
|
|
||||||
name="login"
|
|
||||||
id="kc-login"
|
|
||||||
type="submit"
|
|
||||||
value={msgStr("doLogIn")}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
declare const $: any;
|
|
||||||
|
|
||||||
function evaluateInlineScript() {
|
|
||||||
$(document).ready(function () {
|
|
||||||
// Card Single Select
|
|
||||||
$(".card-pf-view-single-select").click(function (this: any) {
|
|
||||||
if ($(this).hasClass("active")) {
|
|
||||||
$(this).removeClass("active");
|
|
||||||
$(this).children().removeAttr("name");
|
|
||||||
} else {
|
|
||||||
$(".card-pf-view-single-select").removeClass("active");
|
|
||||||
$(".card-pf-view-single-select").children().removeAttr("name");
|
|
||||||
$(this).addClass("active");
|
|
||||||
$(this).children().attr("name", "selectedCredentialId");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
var defaultCred = $(".card-pf-view-single-select")[0];
|
|
||||||
if (defaultCred) {
|
|
||||||
defaultCred.click();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
export default LoginOtp;
|
|
@ -1,38 +0,0 @@
|
|||||||
import React, { memo } from "react";
|
|
||||||
import Template from "./Template";
|
|
||||||
import type { KcProps } from "./KcProps";
|
|
||||||
import type { KcContextBase } from "../getKcContext/KcContextBase";
|
|
||||||
import type { I18n } from "../i18n";
|
|
||||||
|
|
||||||
const LoginPageExpired = memo(({ kcContext, i18n, ...props }: { kcContext: KcContextBase.LoginPageExpired; i18n: I18n } & KcProps) => {
|
|
||||||
const { url } = kcContext;
|
|
||||||
|
|
||||||
const { msg } = i18n;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Template
|
|
||||||
{...{ kcContext, i18n, ...props }}
|
|
||||||
doFetchDefaultThemeResources={true}
|
|
||||||
displayMessage={false}
|
|
||||||
headerNode={msg("pageExpiredTitle")}
|
|
||||||
formNode={
|
|
||||||
<>
|
|
||||||
<p id="instruction1" className="instruction">
|
|
||||||
{msg("pageExpiredMsg1")}
|
|
||||||
<a id="loginRestartLink" href={url.loginRestartFlowUrl}>
|
|
||||||
{msg("doClickHere")}
|
|
||||||
</a>{" "}
|
|
||||||
.<br />
|
|
||||||
{msg("pageExpiredMsg2")}{" "}
|
|
||||||
<a id="loginContinueLink" href={url.loginAction}>
|
|
||||||
{msg("doClickHere")}
|
|
||||||
</a>{" "}
|
|
||||||
.
|
|
||||||
</p>
|
|
||||||
</>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
export default LoginPageExpired;
|
|
@ -1,68 +0,0 @@
|
|||||||
import React, { memo } from "react";
|
|
||||||
import Template from "./Template";
|
|
||||||
import type { KcProps } from "./KcProps";
|
|
||||||
import type { KcContextBase } from "../getKcContext/KcContextBase";
|
|
||||||
import { useCssAndCx } from "tss-react";
|
|
||||||
import type { I18n } from "../i18n";
|
|
||||||
|
|
||||||
const LoginResetPassword = memo(({ kcContext, i18n, ...props }: { kcContext: KcContextBase.LoginResetPassword; i18n: I18n } & KcProps) => {
|
|
||||||
const { url, realm, auth } = kcContext;
|
|
||||||
|
|
||||||
const { msg, msgStr } = i18n;
|
|
||||||
|
|
||||||
const { cx } = useCssAndCx();
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Template
|
|
||||||
{...{ kcContext, i18n, ...props }}
|
|
||||||
doFetchDefaultThemeResources={true}
|
|
||||||
displayMessage={false}
|
|
||||||
headerNode={msg("emailForgotTitle")}
|
|
||||||
formNode={
|
|
||||||
<form id="kc-reset-password-form" className={cx(props.kcFormClass)} action={url.loginAction} method="post">
|
|
||||||
<div className={cx(props.kcFormGroupClass)}>
|
|
||||||
<div className={cx(props.kcLabelWrapperClass)}>
|
|
||||||
<label htmlFor="username" className={cx(props.kcLabelClass)}>
|
|
||||||
{!realm.loginWithEmailAllowed
|
|
||||||
? msg("username")
|
|
||||||
: !realm.registrationEmailAsUsername
|
|
||||||
? msg("usernameOrEmail")
|
|
||||||
: msg("email")}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
id="username"
|
|
||||||
name="username"
|
|
||||||
className={cx(props.kcInputClass)}
|
|
||||||
autoFocus
|
|
||||||
defaultValue={auth !== undefined && auth.showUsername ? auth.attemptedUsername : undefined}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcFormGroupClass, props.kcFormSettingClass)}>
|
|
||||||
<div id="kc-form-options" className={cx(props.kcFormOptionsClass)}>
|
|
||||||
<div className={cx(props.kcFormOptionsWrapperClass)}>
|
|
||||||
<span>
|
|
||||||
<a href={url.loginUrl}>{msg("backToLogin")}</a>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div id="kc-form-buttons" className={cx(props.kcFormButtonsClass)}>
|
|
||||||
<input
|
|
||||||
className={cx(props.kcButtonClass, props.kcButtonPrimaryClass, props.kcButtonBlockClass, props.kcButtonLargeClass)}
|
|
||||||
type="submit"
|
|
||||||
value={msgStr("doSubmit")}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
}
|
|
||||||
infoNode={msg("emailInstruction")}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
export default LoginResetPassword;
|
|
@ -1,119 +0,0 @@
|
|||||||
import React, { memo } from "react";
|
|
||||||
import Template from "./Template";
|
|
||||||
import type { KcProps } from "./KcProps";
|
|
||||||
import type { KcContextBase } from "../getKcContext/KcContextBase";
|
|
||||||
import { useCssAndCx } from "tss-react";
|
|
||||||
import type { I18n } from "../i18n";
|
|
||||||
|
|
||||||
const LoginUpdatePassword = memo(({ kcContext, i18n, ...props }: { kcContext: KcContextBase.LoginUpdatePassword; i18n: I18n } & KcProps) => {
|
|
||||||
const { cx } = useCssAndCx();
|
|
||||||
|
|
||||||
const { msg, msgStr } = i18n;
|
|
||||||
|
|
||||||
const { url, messagesPerField, isAppInitiatedAction, username } = kcContext;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Template
|
|
||||||
{...{ kcContext, i18n, ...props }}
|
|
||||||
doFetchDefaultThemeResources={true}
|
|
||||||
headerNode={msg("updatePasswordTitle")}
|
|
||||||
formNode={
|
|
||||||
<form id="kc-passwd-update-form" className={cx(props.kcFormClass)} action={url.loginAction} method="post">
|
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
id="username"
|
|
||||||
name="username"
|
|
||||||
value={username}
|
|
||||||
readOnly={true}
|
|
||||||
autoComplete="username"
|
|
||||||
style={{ display: "none" }}
|
|
||||||
/>
|
|
||||||
<input type="password" id="password" name="password" autoComplete="current-password" style={{ display: "none" }} />
|
|
||||||
|
|
||||||
<div className={cx(props.kcFormGroupClass, messagesPerField.printIfExists("password", props.kcFormGroupErrorClass))}>
|
|
||||||
<div className={cx(props.kcLabelWrapperClass)}>
|
|
||||||
<label htmlFor="password-new" className={cx(props.kcLabelClass)}>
|
|
||||||
{msg("passwordNew")}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<input
|
|
||||||
type="password"
|
|
||||||
id="password-new"
|
|
||||||
name="password-new"
|
|
||||||
autoFocus
|
|
||||||
autoComplete="new-password"
|
|
||||||
className={cx(props.kcInputClass)}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className={cx(props.kcFormGroupClass, messagesPerField.printIfExists("password-confirm", props.kcFormGroupErrorClass))}>
|
|
||||||
<div className={cx(props.kcLabelWrapperClass)}>
|
|
||||||
<label htmlFor="password-confirm" className={cx(props.kcLabelClass)}>
|
|
||||||
{msg("passwordConfirm")}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<input
|
|
||||||
type="password"
|
|
||||||
id="password-confirm"
|
|
||||||
name="password-confirm"
|
|
||||||
autoComplete="new-password"
|
|
||||||
className={cx(props.kcInputClass)}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className={cx(props.kcFormGroupClass)}>
|
|
||||||
<div id="kc-form-options" className={cx(props.kcFormOptionsClass)}>
|
|
||||||
<div className={cx(props.kcFormOptionsWrapperClass)}>
|
|
||||||
{isAppInitiatedAction && (
|
|
||||||
<div className="checkbox">
|
|
||||||
<label>
|
|
||||||
<input type="checkbox" id="logout-sessions" name="logout-sessions" value="on" checked />
|
|
||||||
{msgStr("logoutOtherSessions")}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div id="kc-form-buttons" className={cx(props.kcFormButtonsClass)}>
|
|
||||||
{isAppInitiatedAction ? (
|
|
||||||
<>
|
|
||||||
<input
|
|
||||||
className={cx(props.kcButtonClass, props.kcButtonPrimaryClass, props.kcButtonLargeClass)}
|
|
||||||
type="submit"
|
|
||||||
defaultValue={msgStr("doSubmit")}
|
|
||||||
/>
|
|
||||||
<button
|
|
||||||
className={cx(props.kcButtonClass, props.kcButtonDefaultClass, props.kcButtonLargeClass)}
|
|
||||||
type="submit"
|
|
||||||
name="cancel-aia"
|
|
||||||
value="true"
|
|
||||||
>
|
|
||||||
{msg("doCancel")}
|
|
||||||
</button>
|
|
||||||
</>
|
|
||||||
) : (
|
|
||||||
<input
|
|
||||||
className={cx(
|
|
||||||
props.kcButtonClass,
|
|
||||||
props.kcButtonPrimaryClass,
|
|
||||||
props.kcButtonBlockClass,
|
|
||||||
props.kcButtonLargeClass,
|
|
||||||
)}
|
|
||||||
type="submit"
|
|
||||||
defaultValue={msgStr("doSubmit")}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
export default LoginUpdatePassword;
|
|
@ -1,122 +0,0 @@
|
|||||||
import React, { memo } from "react";
|
|
||||||
import Template from "./Template";
|
|
||||||
import type { KcProps } from "./KcProps";
|
|
||||||
import type { KcContextBase } from "../getKcContext/KcContextBase";
|
|
||||||
import { useCssAndCx } from "tss-react";
|
|
||||||
import type { I18n } from "../i18n";
|
|
||||||
|
|
||||||
const LoginUpdateProfile = memo(({ kcContext, i18n, ...props }: { kcContext: KcContextBase.LoginUpdateProfile; i18n: I18n } & KcProps) => {
|
|
||||||
const { cx } = useCssAndCx();
|
|
||||||
|
|
||||||
const { msg, msgStr } = i18n;
|
|
||||||
|
|
||||||
const { url, user, messagesPerField, isAppInitiatedAction } = kcContext;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Template
|
|
||||||
{...{ kcContext, i18n, ...props }}
|
|
||||||
doFetchDefaultThemeResources={true}
|
|
||||||
headerNode={msg("loginProfileTitle")}
|
|
||||||
formNode={
|
|
||||||
<form id="kc-update-profile-form" className={cx(props.kcFormClass)} action={url.loginAction} method="post">
|
|
||||||
{user.editUsernameAllowed && (
|
|
||||||
<div className={cx(props.kcFormGroupClass, messagesPerField.printIfExists("username", props.kcFormGroupErrorClass))}>
|
|
||||||
<div className={cx(props.kcLabelWrapperClass)}>
|
|
||||||
<label htmlFor="username" className={cx(props.kcLabelClass)}>
|
|
||||||
{msg("username")}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
id="username"
|
|
||||||
name="username"
|
|
||||||
defaultValue={user.username ?? ""}
|
|
||||||
className={cx(props.kcInputClass)}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<div className={cx(props.kcFormGroupClass, messagesPerField.printIfExists("email", props.kcFormGroupErrorClass))}>
|
|
||||||
<div className={cx(props.kcLabelWrapperClass)}>
|
|
||||||
<label htmlFor="email" className={cx(props.kcLabelClass)}>
|
|
||||||
{msg("email")}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<input type="text" id="email" name="email" defaultValue={user.email ?? ""} className={cx(props.kcInputClass)} />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className={cx(props.kcFormGroupClass, messagesPerField.printIfExists("firstName", props.kcFormGroupErrorClass))}>
|
|
||||||
<div className={cx(props.kcLabelWrapperClass)}>
|
|
||||||
<label htmlFor="firstName" className={cx(props.kcLabelClass)}>
|
|
||||||
{msg("firstName")}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
id="firstName"
|
|
||||||
name="firstName"
|
|
||||||
defaultValue={user.firstName ?? ""}
|
|
||||||
className={cx(props.kcInputClass)}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className={cx(props.kcFormGroupClass, messagesPerField.printIfExists("lastName", props.kcFormGroupErrorClass))}>
|
|
||||||
<div className={cx(props.kcLabelWrapperClass)}>
|
|
||||||
<label htmlFor="lastName" className={cx(props.kcLabelClass)}>
|
|
||||||
{msg("lastName")}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<input type="text" id="lastName" name="lastName" defaultValue={user.lastName ?? ""} className={cx(props.kcInputClass)} />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className={cx(props.kcFormGroupClass)}>
|
|
||||||
<div id="kc-form-options" className={cx(props.kcFormOptionsClass)}>
|
|
||||||
<div className={cx(props.kcFormOptionsWrapperClass)} />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div id="kc-form-buttons" className={cx(props.kcFormButtonsClass)}>
|
|
||||||
{isAppInitiatedAction ? (
|
|
||||||
<>
|
|
||||||
<input
|
|
||||||
className={cx(props.kcButtonClass, props.kcButtonPrimaryClass, props.kcButtonLargeClass)}
|
|
||||||
type="submit"
|
|
||||||
defaultValue={msgStr("doSubmit")}
|
|
||||||
/>
|
|
||||||
<button
|
|
||||||
className={cx(props.kcButtonClass, props.kcButtonDefaultClass, props.kcButtonLargeClass)}
|
|
||||||
type="submit"
|
|
||||||
name="cancel-aia"
|
|
||||||
value="true"
|
|
||||||
>
|
|
||||||
{msg("doCancel")}
|
|
||||||
</button>
|
|
||||||
</>
|
|
||||||
) : (
|
|
||||||
<input
|
|
||||||
className={cx(
|
|
||||||
props.kcButtonClass,
|
|
||||||
props.kcButtonPrimaryClass,
|
|
||||||
props.kcButtonBlockClass,
|
|
||||||
props.kcButtonLargeClass,
|
|
||||||
)}
|
|
||||||
type="submit"
|
|
||||||
defaultValue={msgStr("doSubmit")}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
export default LoginUpdateProfile;
|
|
@ -1,34 +0,0 @@
|
|||||||
import React, { memo } from "react";
|
|
||||||
import Template from "./Template";
|
|
||||||
import type { KcProps } from "./KcProps";
|
|
||||||
import type { KcContextBase } from "../getKcContext/KcContextBase";
|
|
||||||
import type { I18n } from "../i18n";
|
|
||||||
|
|
||||||
const LoginVerifyEmail = memo(({ kcContext, i18n, ...props }: { kcContext: KcContextBase.LoginVerifyEmail; i18n: I18n } & KcProps) => {
|
|
||||||
const { msg } = i18n;
|
|
||||||
|
|
||||||
const { url, user } = kcContext;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Template
|
|
||||||
{...{ kcContext, i18n, ...props }}
|
|
||||||
doFetchDefaultThemeResources={true}
|
|
||||||
displayMessage={false}
|
|
||||||
headerNode={msg("emailVerifyTitle")}
|
|
||||||
formNode={
|
|
||||||
<>
|
|
||||||
<p className="instruction">{msg("emailVerifyInstruction1", user?.email)}</p>
|
|
||||||
<p className="instruction">
|
|
||||||
{msg("emailVerifyInstruction2")}
|
|
||||||
<br />
|
|
||||||
<a href={url.loginAction}>{msg("doClickHere")}</a>
|
|
||||||
|
|
||||||
{msg("emailVerifyInstruction3")}
|
|
||||||
</p>
|
|
||||||
</>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
export default LoginVerifyEmail;
|
|
@ -1,62 +0,0 @@
|
|||||||
import React, { memo } from "react";
|
|
||||||
import { useCssAndCx } from "tss-react";
|
|
||||||
import Template from "./Template";
|
|
||||||
import type { KcProps } from "./KcProps";
|
|
||||||
import type { KcContextBase } from "../getKcContext/KcContextBase";
|
|
||||||
import type { I18n } from "../i18n";
|
|
||||||
|
|
||||||
const LogoutConfirm = memo(({ kcContext, i18n, ...props }: { kcContext: KcContextBase.LogoutConfirm; i18n: I18n } & KcProps) => {
|
|
||||||
const { url, client, logoutConfirm } = kcContext;
|
|
||||||
|
|
||||||
const { cx } = useCssAndCx();
|
|
||||||
|
|
||||||
const { msg, msgStr } = i18n;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Template
|
|
||||||
{...{ kcContext, i18n, ...props }}
|
|
||||||
doFetchDefaultThemeResources={true}
|
|
||||||
displayMessage={false}
|
|
||||||
headerNode={msg("logoutConfirmTitle")}
|
|
||||||
formNode={
|
|
||||||
<>
|
|
||||||
<div id="kc-logout-confirm" className="content-area">
|
|
||||||
<p className="instruction">{msg("logoutConfirmHeader")}</p>
|
|
||||||
<form className="form-actions" action={url.logoutConfirmAction} method="POST">
|
|
||||||
<input type="hidden" name="session_code" value={logoutConfirm.code} />
|
|
||||||
<div className={cx(props.kcFormGroupClass)}>
|
|
||||||
<div id="kc-form-options">
|
|
||||||
<div className={cx(props.kcFormOptionsWrapperClass)}></div>
|
|
||||||
</div>
|
|
||||||
<div id="kc-form-buttons" className={cx(props.kcFormGroupClass)}>
|
|
||||||
<input
|
|
||||||
tabIndex={4}
|
|
||||||
className={cx(
|
|
||||||
props.kcButtonClass,
|
|
||||||
props.kcButtonPrimaryClass,
|
|
||||||
props.kcButtonBlockClass,
|
|
||||||
props.kcButtonLargeClass,
|
|
||||||
)}
|
|
||||||
name="confirmLogout"
|
|
||||||
id="kc-logout"
|
|
||||||
type="submit"
|
|
||||||
value={msgStr("doLogout")}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
<div id="kc-info-message">
|
|
||||||
{!logoutConfirm.skipLink && client.baseUrl && (
|
|
||||||
<p>
|
|
||||||
<a href={client.baseUrl} dangerouslySetInnerHTML={{ __html: msgStr("backToApplication") }} />
|
|
||||||
</p>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
export default LogoutConfirm;
|
|
@ -1,158 +0,0 @@
|
|||||||
import React, { memo } from "react";
|
|
||||||
import Template from "./Template";
|
|
||||||
import type { KcProps } from "./KcProps";
|
|
||||||
import type { KcContextBase } from "../getKcContext/KcContextBase";
|
|
||||||
import { useCssAndCx } from "tss-react";
|
|
||||||
import type { I18n } from "../i18n";
|
|
||||||
|
|
||||||
const Register = memo(({ kcContext, i18n, ...props }: { kcContext: KcContextBase.Register; i18n: I18n } & KcProps) => {
|
|
||||||
const { url, messagesPerField, register, realm, passwordRequired, recaptchaRequired, recaptchaSiteKey } = kcContext;
|
|
||||||
|
|
||||||
const { msg, msgStr } = i18n;
|
|
||||||
|
|
||||||
const { cx } = useCssAndCx();
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Template
|
|
||||||
{...{ kcContext, i18n, ...props }}
|
|
||||||
doFetchDefaultThemeResources={true}
|
|
||||||
headerNode={msg("registerTitle")}
|
|
||||||
formNode={
|
|
||||||
<form id="kc-register-form" className={cx(props.kcFormClass)} action={url.registrationAction} method="post">
|
|
||||||
<div className={cx(props.kcFormGroupClass, messagesPerField.printIfExists("firstName", props.kcFormGroupErrorClass))}>
|
|
||||||
<div className={cx(props.kcLabelWrapperClass)}>
|
|
||||||
<label htmlFor="firstName" className={cx(props.kcLabelClass)}>
|
|
||||||
{msg("firstName")}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
id="firstName"
|
|
||||||
className={cx(props.kcInputClass)}
|
|
||||||
name="firstName"
|
|
||||||
defaultValue={register.formData.firstName ?? ""}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className={cx(props.kcFormGroupClass, messagesPerField.printIfExists("lastName", props.kcFormGroupErrorClass))}>
|
|
||||||
<div className={cx(props.kcLabelWrapperClass)}>
|
|
||||||
<label htmlFor="lastName" className={cx(props.kcLabelClass)}>
|
|
||||||
{msg("lastName")}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
id="lastName"
|
|
||||||
className={cx(props.kcInputClass)}
|
|
||||||
name="lastName"
|
|
||||||
defaultValue={register.formData.lastName ?? ""}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className={cx(props.kcFormGroupClass, messagesPerField.printIfExists("email", props.kcFormGroupErrorClass))}>
|
|
||||||
<div className={cx(props.kcLabelWrapperClass)}>
|
|
||||||
<label htmlFor="email" className={cx(props.kcLabelClass)}>
|
|
||||||
{msg("email")}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
id="email"
|
|
||||||
className={cx(props.kcInputClass)}
|
|
||||||
name="email"
|
|
||||||
defaultValue={register.formData.email ?? ""}
|
|
||||||
autoComplete="email"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{!realm.registrationEmailAsUsername && (
|
|
||||||
<div className={cx(props.kcFormGroupClass, messagesPerField.printIfExists("username", props.kcFormGroupErrorClass))}>
|
|
||||||
<div className={cx(props.kcLabelWrapperClass)}>
|
|
||||||
<label htmlFor="username" className={cx(props.kcLabelClass)}>
|
|
||||||
{msg("username")}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
id="username"
|
|
||||||
className={cx(props.kcInputClass)}
|
|
||||||
name="username"
|
|
||||||
defaultValue={register.formData.username ?? ""}
|
|
||||||
autoComplete="username"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
{passwordRequired && (
|
|
||||||
<>
|
|
||||||
<div className={cx(props.kcFormGroupClass, messagesPerField.printIfExists("password", props.kcFormGroupErrorClass))}>
|
|
||||||
<div className={cx(props.kcLabelWrapperClass)}>
|
|
||||||
<label htmlFor="password" className={cx(props.kcLabelClass)}>
|
|
||||||
{msg("password")}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<input
|
|
||||||
type="password"
|
|
||||||
id="password"
|
|
||||||
className={cx(props.kcInputClass)}
|
|
||||||
name="password"
|
|
||||||
autoComplete="new-password"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div
|
|
||||||
className={cx(
|
|
||||||
props.kcFormGroupClass,
|
|
||||||
messagesPerField.printIfExists("password-confirm", props.kcFormGroupErrorClass),
|
|
||||||
)}
|
|
||||||
>
|
|
||||||
<div className={cx(props.kcLabelWrapperClass)}>
|
|
||||||
<label htmlFor="password-confirm" className={cx(props.kcLabelClass)}>
|
|
||||||
{msg("passwordConfirm")}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<input type="password" id="password-confirm" className={cx(props.kcInputClass)} name="password-confirm" />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
{recaptchaRequired && (
|
|
||||||
<div className="form-group">
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<div className="g-recaptcha" data-size="compact" data-sitekey={recaptchaSiteKey}></div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
<div className={cx(props.kcFormGroupClass)}>
|
|
||||||
<div id="kc-form-options" className={cx(props.kcFormOptionsClass)}>
|
|
||||||
<div className={cx(props.kcFormOptionsWrapperClass)}>
|
|
||||||
<span>
|
|
||||||
<a href={url.loginUrl}>{msg("backToLogin")}</a>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div id="kc-form-buttons" className={cx(props.kcFormButtonsClass)}>
|
|
||||||
<input
|
|
||||||
className={cx(props.kcButtonClass, props.kcButtonPrimaryClass, props.kcButtonBlockClass, props.kcButtonLargeClass)}
|
|
||||||
type="submit"
|
|
||||||
value={msgStr("doRegister")}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
export default Register;
|
|
@ -1,220 +0,0 @@
|
|||||||
import React, { useMemo, memo, useEffect, useState, Fragment } from "react";
|
|
||||||
import Template from "./Template";
|
|
||||||
import type { KcProps } from "./KcProps";
|
|
||||||
import type { KcContextBase, Attribute } from "../getKcContext/KcContextBase";
|
|
||||||
import { useCssAndCx } from "tss-react";
|
|
||||||
import type { ReactComponent } from "../tools/ReactComponent";
|
|
||||||
import { useCallbackFactory } from "powerhooks/useCallbackFactory";
|
|
||||||
import { useFormValidationSlice } from "../useFormValidationSlice";
|
|
||||||
import type { I18n } from "../i18n";
|
|
||||||
|
|
||||||
const RegisterUserProfile = memo(({ kcContext, i18n, ...props_ }: { kcContext: KcContextBase.RegisterUserProfile; i18n: I18n } & KcProps) => {
|
|
||||||
const { url, messagesPerField, recaptchaRequired, recaptchaSiteKey } = kcContext;
|
|
||||||
|
|
||||||
const { msg, msgStr } = i18n;
|
|
||||||
|
|
||||||
const { cx, css } = useCssAndCx();
|
|
||||||
|
|
||||||
const props = useMemo(
|
|
||||||
() => ({
|
|
||||||
...props_,
|
|
||||||
"kcFormGroupClass": cx(props_.kcFormGroupClass, css({ "marginBottom": 20 })),
|
|
||||||
}),
|
|
||||||
[cx, css],
|
|
||||||
);
|
|
||||||
|
|
||||||
const [isFomSubmittable, setIsFomSubmittable] = useState(false);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Template
|
|
||||||
{...{ kcContext, i18n, ...props }}
|
|
||||||
displayMessage={messagesPerField.exists("global")}
|
|
||||||
displayRequiredFields={true}
|
|
||||||
doFetchDefaultThemeResources={true}
|
|
||||||
headerNode={msg("registerTitle")}
|
|
||||||
formNode={
|
|
||||||
<form id="kc-register-form" className={cx(props.kcFormClass)} action={url.registrationAction} method="post">
|
|
||||||
<UserProfileFormFields kcContext={kcContext} onIsFormSubmittableValueChange={setIsFomSubmittable} i18n={i18n} {...props} />
|
|
||||||
{recaptchaRequired && (
|
|
||||||
<div className="form-group">
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
<div className="g-recaptcha" data-size="compact" data-sitekey={recaptchaSiteKey} />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
<div className={cx(props.kcFormGroupClass)}>
|
|
||||||
<div id="kc-form-options" className={cx(props.kcFormOptionsClass)}>
|
|
||||||
<div className={cx(props.kcFormOptionsWrapperClass)}>
|
|
||||||
<span>
|
|
||||||
<a href={url.loginUrl}>{msg("backToLogin")}</a>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div id="kc-form-buttons" className={cx(props.kcFormButtonsClass)}>
|
|
||||||
<input
|
|
||||||
className={cx(props.kcButtonClass, props.kcButtonPrimaryClass, props.kcButtonBlockClass, props.kcButtonLargeClass)}
|
|
||||||
type="submit"
|
|
||||||
value={msgStr("doRegister")}
|
|
||||||
disabled={!isFomSubmittable}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
type UserProfileFormFieldsProps = { kcContext: KcContextBase.RegisterUserProfile; i18n: I18n } & KcProps &
|
|
||||||
Partial<Record<"BeforeField" | "AfterField", ReactComponent<{ attribute: Attribute }>>> & {
|
|
||||||
onIsFormSubmittableValueChange: (isFormSubmittable: boolean) => void;
|
|
||||||
};
|
|
||||||
|
|
||||||
const UserProfileFormFields = memo(({ kcContext, onIsFormSubmittableValueChange, i18n, ...props }: UserProfileFormFieldsProps) => {
|
|
||||||
const { cx, css } = useCssAndCx();
|
|
||||||
|
|
||||||
const { advancedMsg } = i18n;
|
|
||||||
|
|
||||||
const {
|
|
||||||
formValidationState: { fieldStateByAttributeName, isFormSubmittable },
|
|
||||||
formValidationReducer,
|
|
||||||
attributesWithPassword,
|
|
||||||
} = useFormValidationSlice({
|
|
||||||
kcContext,
|
|
||||||
i18n,
|
|
||||||
});
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
onIsFormSubmittableValueChange(isFormSubmittable);
|
|
||||||
}, [isFormSubmittable]);
|
|
||||||
|
|
||||||
const onChangeFactory = useCallbackFactory(
|
|
||||||
(
|
|
||||||
[name]: [string],
|
|
||||||
[
|
|
||||||
{
|
|
||||||
target: { value },
|
|
||||||
},
|
|
||||||
]: [React.ChangeEvent<HTMLInputElement | HTMLSelectElement>],
|
|
||||||
) =>
|
|
||||||
formValidationReducer({
|
|
||||||
"action": "update value",
|
|
||||||
name,
|
|
||||||
"newValue": value,
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
|
|
||||||
const onBlurFactory = useCallbackFactory(([name]: [string]) =>
|
|
||||||
formValidationReducer({
|
|
||||||
"action": "focus lost",
|
|
||||||
name,
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
|
|
||||||
let currentGroup = "";
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
{attributesWithPassword.map((attribute, i) => {
|
|
||||||
const { group = "", groupDisplayHeader = "", groupDisplayDescription = "" } = attribute;
|
|
||||||
|
|
||||||
const { value, displayableErrors } = fieldStateByAttributeName[attribute.name];
|
|
||||||
|
|
||||||
const formGroupClassName = cx(props.kcFormGroupClass, displayableErrors.length !== 0 && props.kcFormGroupErrorClass);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Fragment key={i}>
|
|
||||||
{group !== currentGroup && (currentGroup = group) !== "" && (
|
|
||||||
<div className={formGroupClassName}>
|
|
||||||
<div className={cx(props.kcContentWrapperClass)}>
|
|
||||||
<label id={`header-${group}`} className={cx(props.kcFormGroupHeader)}>
|
|
||||||
{advancedMsg(groupDisplayHeader) || currentGroup}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
{groupDisplayDescription !== "" && (
|
|
||||||
<div className={cx(props.kcLabelWrapperClass)}>
|
|
||||||
<label id={`description-${group}`} className={`${cx(props.kcLabelClass)}`}>
|
|
||||||
{advancedMsg(groupDisplayDescription)}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
<div className={formGroupClassName}>
|
|
||||||
<div className={cx(props.kcLabelWrapperClass)}>
|
|
||||||
<label htmlFor={attribute.name} className={cx(props.kcLabelClass)}>
|
|
||||||
{advancedMsg(attribute.displayName ?? "")}
|
|
||||||
</label>
|
|
||||||
{attribute.required && <>*</>}
|
|
||||||
</div>
|
|
||||||
<div className={cx(props.kcInputWrapperClass)}>
|
|
||||||
{(() => {
|
|
||||||
const { options } = attribute.validators;
|
|
||||||
|
|
||||||
if (options !== undefined) {
|
|
||||||
return (
|
|
||||||
<select
|
|
||||||
id={attribute.name}
|
|
||||||
name={attribute.name}
|
|
||||||
onChange={onChangeFactory(attribute.name)}
|
|
||||||
onBlur={onBlurFactory(attribute.name)}
|
|
||||||
value={value}
|
|
||||||
>
|
|
||||||
{options.options.map(option => (
|
|
||||||
<option key={option} value={option}>
|
|
||||||
{option}
|
|
||||||
</option>
|
|
||||||
))}
|
|
||||||
</select>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<input
|
|
||||||
type={(() => {
|
|
||||||
switch (attribute.name) {
|
|
||||||
case "password-confirm":
|
|
||||||
case "password":
|
|
||||||
return "password";
|
|
||||||
default:
|
|
||||||
return "text";
|
|
||||||
}
|
|
||||||
})()}
|
|
||||||
id={attribute.name}
|
|
||||||
name={attribute.name}
|
|
||||||
value={value}
|
|
||||||
onChange={onChangeFactory(attribute.name)}
|
|
||||||
className={cx(props.kcInputClass)}
|
|
||||||
aria-invalid={displayableErrors.length !== 0}
|
|
||||||
disabled={attribute.readOnly}
|
|
||||||
autoComplete={attribute.autocomplete}
|
|
||||||
onBlur={onBlurFactory(attribute.name)}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
})()}
|
|
||||||
{displayableErrors.length !== 0 && (
|
|
||||||
<span
|
|
||||||
id={`input-error-${attribute.name}`}
|
|
||||||
className={cx(
|
|
||||||
props.kcInputErrorMessageClass,
|
|
||||||
css({
|
|
||||||
"position": displayableErrors.length === 1 ? "absolute" : undefined,
|
|
||||||
"& > span": { "display": "block" },
|
|
||||||
}),
|
|
||||||
)}
|
|
||||||
aria-live="polite"
|
|
||||||
>
|
|
||||||
{displayableErrors.map(({ errorMessage }) => errorMessage)}
|
|
||||||
</span>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</Fragment>
|
|
||||||
);
|
|
||||||
})}
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
export default RegisterUserProfile;
|
|
@ -1,107 +0,0 @@
|
|||||||
import React, { useEffect, memo } from "react";
|
|
||||||
import Template from "./Template";
|
|
||||||
import type { KcProps } from "./KcProps";
|
|
||||||
import type { KcContextBase } from "../getKcContext/KcContextBase";
|
|
||||||
import { useCssAndCx } from "tss-react";
|
|
||||||
import { Evt } from "evt";
|
|
||||||
import { useRerenderOnStateChange } from "evt/hooks";
|
|
||||||
import { assert } from "tsafe/assert";
|
|
||||||
import { fallbackLanguageTag } from "../i18n";
|
|
||||||
import type { I18n } from "../i18n";
|
|
||||||
import memoize from "memoizee";
|
|
||||||
import { useConst } from "powerhooks/useConst";
|
|
||||||
import { useConstCallback } from "powerhooks/useConstCallback";
|
|
||||||
|
|
||||||
export const evtTermMarkdown = Evt.create<string | undefined>(undefined);
|
|
||||||
|
|
||||||
export type KcContextLike = {
|
|
||||||
pageId: KcContextBase["pageId"];
|
|
||||||
locale?: {
|
|
||||||
currentLanguageTag: string;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
assert<KcContextBase extends KcContextLike ? true : false>();
|
|
||||||
|
|
||||||
/** Allow to avoid bundling the terms and download it on demand*/
|
|
||||||
export function useDownloadTerms(params: {
|
|
||||||
kcContext: KcContextLike;
|
|
||||||
downloadTermMarkdown: (params: { currentLanguageTag: string }) => Promise<string>;
|
|
||||||
}) {
|
|
||||||
const { kcContext } = params;
|
|
||||||
|
|
||||||
const { downloadTermMarkdownMemoized } = (function useClosure() {
|
|
||||||
const { downloadTermMarkdown } = params;
|
|
||||||
|
|
||||||
const downloadTermMarkdownConst = useConstCallback(downloadTermMarkdown);
|
|
||||||
|
|
||||||
const downloadTermMarkdownMemoized = useConst(() =>
|
|
||||||
memoize((currentLanguageTag: string) => downloadTermMarkdownConst({ currentLanguageTag }), { "promise": true }),
|
|
||||||
);
|
|
||||||
|
|
||||||
return { downloadTermMarkdownMemoized };
|
|
||||||
})();
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (kcContext.pageId !== "terms.ftl") {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
downloadTermMarkdownMemoized(kcContext.locale?.currentLanguageTag ?? fallbackLanguageTag).then(
|
|
||||||
thermMarkdown => (evtTermMarkdown.state = thermMarkdown),
|
|
||||||
);
|
|
||||||
}, []);
|
|
||||||
}
|
|
||||||
|
|
||||||
const Terms = memo(({ kcContext, i18n, ...props }: { kcContext: KcContextBase.Terms; i18n: I18n } & KcProps) => {
|
|
||||||
const { msg, msgStr } = i18n;
|
|
||||||
|
|
||||||
useRerenderOnStateChange(evtTermMarkdown);
|
|
||||||
|
|
||||||
const { cx } = useCssAndCx();
|
|
||||||
|
|
||||||
const { url } = kcContext;
|
|
||||||
|
|
||||||
if (evtTermMarkdown.state === undefined) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Template
|
|
||||||
{...{ kcContext, i18n, ...props }}
|
|
||||||
doFetchDefaultThemeResources={true}
|
|
||||||
displayMessage={false}
|
|
||||||
headerNode={msg("termsTitle")}
|
|
||||||
formNode={
|
|
||||||
<>
|
|
||||||
<div id="kc-terms-text">{evtTermMarkdown.state}</div>
|
|
||||||
<form className="form-actions" action={url.loginAction} method="POST">
|
|
||||||
<input
|
|
||||||
className={cx(
|
|
||||||
props.kcButtonClass,
|
|
||||||
props.kcButtonClass,
|
|
||||||
props.kcButtonClass,
|
|
||||||
props.kcButtonPrimaryClass,
|
|
||||||
props.kcButtonLargeClass,
|
|
||||||
)}
|
|
||||||
name="accept"
|
|
||||||
id="kc-accept"
|
|
||||||
type="submit"
|
|
||||||
value={msgStr("doAccept")}
|
|
||||||
/>
|
|
||||||
<input
|
|
||||||
className={cx(props.kcButtonClass, props.kcButtonDefaultClass, props.kcButtonLargeClass)}
|
|
||||||
name="cancel"
|
|
||||||
id="kc-decline"
|
|
||||||
type="submit"
|
|
||||||
value={msgStr("doDecline")}
|
|
||||||
/>
|
|
||||||
</form>
|
|
||||||
<div className="clearfix" />
|
|
||||||
</>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
export default Terms;
|
|
@ -1,108 +0,0 @@
|
|||||||
import type { KcContextBase, Attribute } from "./KcContextBase";
|
|
||||||
import { kcContextMocks, kcContextCommonMock } from "./kcContextMocks";
|
|
||||||
import type { DeepPartial } from "../tools/DeepPartial";
|
|
||||||
import { deepAssign } from "../tools/deepAssign";
|
|
||||||
import { id } from "tsafe/id";
|
|
||||||
import { exclude } from "tsafe/exclude";
|
|
||||||
import { assert } from "tsafe/assert";
|
|
||||||
import type { ExtendsKcContextBase } from "./getKcContextFromWindow";
|
|
||||||
import { getKcContextFromWindow } from "./getKcContextFromWindow";
|
|
||||||
import { pathJoin } from "../../bin/tools/pathJoin";
|
|
||||||
import { pathBasename } from "../tools/pathBasename";
|
|
||||||
import { mockTestingResourcesCommonPath } from "../../bin/mockTestingResourcesPath";
|
|
||||||
|
|
||||||
export function getKcContext<KcContextExtended extends { pageId: string } = never>(params?: {
|
|
||||||
mockPageId?: ExtendsKcContextBase<KcContextExtended>["pageId"];
|
|
||||||
mockData?: readonly DeepPartial<ExtendsKcContextBase<KcContextExtended>>[];
|
|
||||||
}): { kcContext: ExtendsKcContextBase<KcContextExtended> | undefined } {
|
|
||||||
const { mockPageId, mockData } = params ?? {};
|
|
||||||
|
|
||||||
if (mockPageId !== undefined) {
|
|
||||||
//TODO maybe trow if no mock fo custom page
|
|
||||||
|
|
||||||
const kcContextDefaultMock = kcContextMocks.find(({ pageId }) => pageId === mockPageId);
|
|
||||||
|
|
||||||
const partialKcContextCustomMock = mockData?.find(({ pageId }) => pageId === mockPageId);
|
|
||||||
|
|
||||||
if (kcContextDefaultMock === undefined && partialKcContextCustomMock === undefined) {
|
|
||||||
console.warn(
|
|
||||||
[
|
|
||||||
`WARNING: You declared the non build in page ${mockPageId} but you didn't `,
|
|
||||||
`provide mock data needed to debug the page outside of Keycloak as you are trying to do now.`,
|
|
||||||
`Please check the documentation of the getKcContext function`,
|
|
||||||
].join("\n"),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
const kcContext: any = {};
|
|
||||||
|
|
||||||
deepAssign({
|
|
||||||
"target": kcContext,
|
|
||||||
"source": kcContextDefaultMock !== undefined ? kcContextDefaultMock : { "pageId": mockPageId, ...kcContextCommonMock },
|
|
||||||
});
|
|
||||||
|
|
||||||
if (partialKcContextCustomMock !== undefined) {
|
|
||||||
deepAssign({
|
|
||||||
"target": kcContext,
|
|
||||||
"source": partialKcContextCustomMock,
|
|
||||||
});
|
|
||||||
|
|
||||||
if (partialKcContextCustomMock.pageId === "register-user-profile.ftl") {
|
|
||||||
assert(kcContextDefaultMock?.pageId === "register-user-profile.ftl");
|
|
||||||
|
|
||||||
const { attributes } = kcContextDefaultMock.profile;
|
|
||||||
|
|
||||||
id<KcContextBase.RegisterUserProfile>(kcContext).profile.attributes = [];
|
|
||||||
id<KcContextBase.RegisterUserProfile>(kcContext).profile.attributesByName = {};
|
|
||||||
|
|
||||||
const partialAttributes = [
|
|
||||||
...((partialKcContextCustomMock as DeepPartial<KcContextBase.RegisterUserProfile>).profile?.attributes ?? []),
|
|
||||||
].filter(exclude(undefined));
|
|
||||||
|
|
||||||
attributes.forEach(attribute => {
|
|
||||||
const partialAttribute = partialAttributes.find(({ name }) => name === attribute.name);
|
|
||||||
|
|
||||||
const augmentedAttribute: Attribute = {} as any;
|
|
||||||
|
|
||||||
deepAssign({
|
|
||||||
"target": augmentedAttribute,
|
|
||||||
"source": attribute,
|
|
||||||
});
|
|
||||||
|
|
||||||
if (partialAttribute !== undefined) {
|
|
||||||
partialAttributes.splice(partialAttributes.indexOf(partialAttribute), 1);
|
|
||||||
|
|
||||||
deepAssign({
|
|
||||||
"target": augmentedAttribute,
|
|
||||||
"source": partialAttribute,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
id<KcContextBase.RegisterUserProfile>(kcContext).profile.attributes.push(augmentedAttribute);
|
|
||||||
id<KcContextBase.RegisterUserProfile>(kcContext).profile.attributesByName[augmentedAttribute.name] = augmentedAttribute;
|
|
||||||
});
|
|
||||||
|
|
||||||
partialAttributes.forEach(partialAttribute => {
|
|
||||||
const { name } = partialAttribute;
|
|
||||||
|
|
||||||
assert(name !== undefined, "If you define a mock attribute it must have at least a name");
|
|
||||||
|
|
||||||
id<KcContextBase.RegisterUserProfile>(kcContext).profile.attributes.push(partialAttribute as any);
|
|
||||||
id<KcContextBase.RegisterUserProfile>(kcContext).profile.attributesByName[name] = partialAttribute as any;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return { kcContext };
|
|
||||||
}
|
|
||||||
|
|
||||||
const kcContext = getKcContextFromWindow<KcContextExtended>();
|
|
||||||
|
|
||||||
if (kcContext !== undefined) {
|
|
||||||
const { url } = kcContext;
|
|
||||||
|
|
||||||
url.resourcesCommonPath = pathJoin(url.resourcesPath, pathBasename(mockTestingResourcesCommonPath));
|
|
||||||
}
|
|
||||||
|
|
||||||
return { kcContext };
|
|
||||||
}
|
|
@ -1,11 +0,0 @@
|
|||||||
import type { KcContextBase } from "./KcContextBase";
|
|
||||||
import type { AndByDiscriminatingKey } from "../tools/AndByDiscriminatingKey";
|
|
||||||
import { ftlValuesGlobalName } from "../../bin/build-keycloak-theme/ftlValuesGlobalName";
|
|
||||||
|
|
||||||
export type ExtendsKcContextBase<KcContextExtended extends { pageId: string }> = [KcContextExtended] extends [never]
|
|
||||||
? KcContextBase
|
|
||||||
: AndByDiscriminatingKey<"pageId", KcContextExtended & KcContextBase.Common, KcContextBase>;
|
|
||||||
|
|
||||||
export function getKcContextFromWindow<KcContextExtended extends { pageId: string } = never>(): ExtendsKcContextBase<KcContextExtended> | undefined {
|
|
||||||
return typeof window === "undefined" ? undefined : (window as any)[ftlValuesGlobalName];
|
|
||||||
}
|
|
@ -1,3 +0,0 @@
|
|||||||
export type { KcContextBase, Attribute, Validators } from "./KcContextBase";
|
|
||||||
export type { ExtendsKcContextBase } from "./getKcContextFromWindow";
|
|
||||||
export { getKcContext } from "./getKcContext";
|
|
@ -1 +0,0 @@
|
|||||||
export * from "./kcContextMocks";
|
|
@ -1,137 +0,0 @@
|
|||||||
//This code was automatically generated by running dist/bin/generate-i18n-messages.js
|
|
||||||
//PLEASE DO NOT EDIT MANUALLY
|
|
||||||
|
|
||||||
/* spell-checker: disable */
|
|
||||||
const messages = {
|
|
||||||
"doSave": "Desa",
|
|
||||||
"doCancel": "Cancel·la",
|
|
||||||
"doLogOutAllSessions": "Desconnecta de totes les sessions",
|
|
||||||
"doRemove": "Elimina",
|
|
||||||
"doAdd": "Afegeix",
|
|
||||||
"doSignOut": "Desconnectar",
|
|
||||||
"editAccountHtmlTitle": "Edita compte",
|
|
||||||
"federatedIdentitiesHtmlTitle": "Identitats federades",
|
|
||||||
"accountLogHtmlTitle": "Registre del compte",
|
|
||||||
"changePasswordHtmlTitle": "Canvia contrasenya",
|
|
||||||
"sessionsHtmlTitle": "Sessions",
|
|
||||||
"accountManagementTitle": "Gestió de Compte Keycloak",
|
|
||||||
"authenticatorTitle": "Autenticador",
|
|
||||||
"applicationsHtmlTitle": "Aplicacions",
|
|
||||||
"authenticatorCode": "Codi d'un sol ús",
|
|
||||||
"email": "Email",
|
|
||||||
"firstName": "Nom",
|
|
||||||
"givenName": "Nom de pila",
|
|
||||||
"fullName": "Nom complet",
|
|
||||||
"lastName": "Cognoms",
|
|
||||||
"familyName": "Cognom",
|
|
||||||
"password": "Contrasenya",
|
|
||||||
"passwordConfirm": "Confirma la contrasenya",
|
|
||||||
"passwordNew": "Nova contrasenya",
|
|
||||||
"username": "Usuari",
|
|
||||||
"address": "Adreça",
|
|
||||||
"street": "Carrer",
|
|
||||||
"locality": "Ciutat o Municipi",
|
|
||||||
"region": "Estat, Província, o Regió",
|
|
||||||
"postal_code": "Postal code",
|
|
||||||
"country": "País",
|
|
||||||
"emailVerified": "Email verificat",
|
|
||||||
"gssDelegationCredential": "GSS Delegation Credential",
|
|
||||||
"role_admin": "Administrador",
|
|
||||||
"role_realm-admin": "Administrador del domini",
|
|
||||||
"role_create-realm": "Crear domini",
|
|
||||||
"role_view-realm": "Veure domini",
|
|
||||||
"role_view-users": "Veure usuaris",
|
|
||||||
"role_view-applications": "Veure aplicacions",
|
|
||||||
"role_view-clients": "Veure clients",
|
|
||||||
"role_view-events": "Veure events",
|
|
||||||
"role_view-identity-providers": "Veure proveïdors d'identitat",
|
|
||||||
"role_manage-realm": "Gestionar domini",
|
|
||||||
"role_manage-users": "Gestinar usuaris",
|
|
||||||
"role_manage-applications": "Gestionar aplicacions",
|
|
||||||
"role_manage-identity-providers": "Gestionar proveïdors d'identitat",
|
|
||||||
"role_manage-clients": "Gestionar clients",
|
|
||||||
"role_manage-events": "Gestionar events",
|
|
||||||
"role_view-profile": "Veure perfil",
|
|
||||||
"role_manage-account": "Gestionar compte",
|
|
||||||
"role_read-token": "Llegir token",
|
|
||||||
"role_offline-access": "Accés sense connexió",
|
|
||||||
"client_account": "Compte",
|
|
||||||
"client_security-admin-console": "Consola d'Administració de Seguretat",
|
|
||||||
"client_realm-management": "Gestió de domini",
|
|
||||||
"client_broker": "Broker",
|
|
||||||
"requiredFields": "Camps obligatoris",
|
|
||||||
"allFieldsRequired": "Tots els camps obligatoris",
|
|
||||||
"backToApplication": "« Torna a l'aplicació",
|
|
||||||
"backTo": "Torna a {0}",
|
|
||||||
"date": "Data",
|
|
||||||
"event": "Event",
|
|
||||||
"ip": "IP",
|
|
||||||
"client": "Client",
|
|
||||||
"clients": "Clients",
|
|
||||||
"details": "Detalls",
|
|
||||||
"started": "Iniciat",
|
|
||||||
"lastAccess": "Últim accés",
|
|
||||||
"expires": "Expira",
|
|
||||||
"applications": "Aplicacions",
|
|
||||||
"account": "Compte",
|
|
||||||
"federatedIdentity": "Identitat federada",
|
|
||||||
"authenticator": "Autenticador",
|
|
||||||
"sessions": "Sessions",
|
|
||||||
"log": "Registre",
|
|
||||||
"application": "Aplicació",
|
|
||||||
"availablePermissions": "Permisos disponibles",
|
|
||||||
"grantedPermissions": "Permisos concedits",
|
|
||||||
"grantedPersonalInfo": "Informació personal concedida",
|
|
||||||
"additionalGrants": "Permisos addicionals",
|
|
||||||
"action": "Acció",
|
|
||||||
"inResource": "a",
|
|
||||||
"fullAccess": "Accés total",
|
|
||||||
"offlineToken": "Codi d'autorització offline",
|
|
||||||
"revoke": "Revocar permís",
|
|
||||||
"configureAuthenticators": "Autenticadors configurats",
|
|
||||||
"mobile": "Mòbil",
|
|
||||||
"totpStep1":
|
|
||||||
'Instal·la <a href="https://freeotp.github.io/" target="_blank">FreeOTP</a> o Google Authenticator al teu telèfon mòbil. Les dues aplicacions estan disponibles a <a href="https://play.google.com">Google Play</a> i en l\'App Store d\'Apple.',
|
|
||||||
"totpStep2": "Obre l'aplicació i escaneja el codi o introdueix la clau.",
|
|
||||||
"totpStep3": "Introdueix el codi únic que et mostra l'aplicació d'autenticació i fes clic a Envia per finalitzar la configuració",
|
|
||||||
"missingUsernameMessage": "Si us plau indica el teu usuari.",
|
|
||||||
"missingFirstNameMessage": "Si us plau indica el nom.",
|
|
||||||
"invalidEmailMessage": "Email no vàlid",
|
|
||||||
"missingLastNameMessage": "Si us plau indica els teus cognoms.",
|
|
||||||
"missingEmailMessage": "Si us plau indica l'email.",
|
|
||||||
"missingPasswordMessage": "Si us plau indica la contrasenya.",
|
|
||||||
"notMatchPasswordMessage": "Les contrasenyes no coincideixen.",
|
|
||||||
"missingTotpMessage": "Si us plau indica el teu codi d'autenticació",
|
|
||||||
"invalidPasswordExistingMessage": "La contrasenya actual no és correcta.",
|
|
||||||
"invalidPasswordConfirmMessage": "La confirmació de contrasenya no coincideix.",
|
|
||||||
"invalidTotpMessage": "El código de autenticación no es válido.",
|
|
||||||
"usernameExistsMessage": "L'usuari ja existeix",
|
|
||||||
"emailExistsMessage": "L'email ja existeix",
|
|
||||||
"readOnlyUserMessage": "No pots actualitzar el teu usuari perquè el teu compte és de només lectura.",
|
|
||||||
"readOnlyPasswordMessage": "No pots actualitzar la contrasenya perquè el teu compte és de només lectura.",
|
|
||||||
"successTotpMessage": "Aplicació d'autenticació mòbil configurada.",
|
|
||||||
"successTotpRemovedMessage": "Aplicació d'autenticació mòbil eliminada.",
|
|
||||||
"successGrantRevokedMessage": "Permís revocat correctament",
|
|
||||||
"accountUpdatedMessage": "El teu compte s'ha actualitzat.",
|
|
||||||
"accountPasswordUpdatedMessage": "La contrasenya s'ha actualitzat.",
|
|
||||||
"missingIdentityProviderMessage": "Proveïdor d'identitat no indicat.",
|
|
||||||
"invalidFederatedIdentityActionMessage": "Acció no vàlida o no indicada.",
|
|
||||||
"identityProviderNotFoundMessage": "No s'ha trobat un proveïdor d'identitat.",
|
|
||||||
"federatedIdentityLinkNotActiveMessage": "Aquesta identitat ja no està activa",
|
|
||||||
"federatedIdentityRemovingLastProviderMessage": "No pots eliminar l'última identitat federada perquè no tens fixada una contrasenya.",
|
|
||||||
"identityProviderRedirectErrorMessage": "Error en la redirecció al proveïdor d'identitat",
|
|
||||||
"identityProviderRemovedMessage": "Proveïdor d'identitat esborrat correctament.",
|
|
||||||
"accountDisabledMessage": "El compte està desactivada, contacteu amb l'administrador.",
|
|
||||||
"accountTemporarilyDisabledMessage": "El compte està temporalment desactivat, contacta amb l'administrador o intenta-ho de nou més tard.",
|
|
||||||
"invalidPasswordMinLengthMessage": "Contrasenya incorrecta: longitud mínima {0}.",
|
|
||||||
"invalidPasswordMinLowerCaseCharsMessage": "Contrasenya incorrecta: ha de contenir almenys {0} lletres minúscules.",
|
|
||||||
"invalidPasswordMinDigitsMessage": "Contraseña incorrecta: debe contener al menos {0} caracteres numéricos.",
|
|
||||||
"invalidPasswordMinUpperCaseCharsMessage": "Contrasenya incorrecta: ha de contenir almenys {0} lletres majúscules.",
|
|
||||||
"invalidPasswordMinSpecialCharsMessage": "Contrasenya incorrecta: ha de contenir almenys {0} caràcters especials.",
|
|
||||||
"invalidPasswordNotUsernameMessage": "Contrasenya incorrecta: no pot ser igual al nom d'usuari.",
|
|
||||||
"invalidPasswordRegexPatternMessage": "Contrasenya incorrecta: no compleix l'expressió regular.",
|
|
||||||
"invalidPasswordHistoryMessage": "Contrasenya incorrecta: no pot ser igual a cap de les últimes {0} contrasenyes.",
|
|
||||||
};
|
|
||||||
|
|
||||||
export default messages;
|
|
||||||
/* spell-checker: enable */
|
|
@ -1,156 +0,0 @@
|
|||||||
//This code was automatically generated by running dist/bin/generate-i18n-messages.js
|
|
||||||
//PLEASE DO NOT EDIT MANUALLY
|
|
||||||
|
|
||||||
/* spell-checker: disable */
|
|
||||||
const messages = {
|
|
||||||
"doSave": "Uložit",
|
|
||||||
"doCancel": "Zrušit",
|
|
||||||
"doLogOutAllSessions": "Odhlásit všechny relace",
|
|
||||||
"doRemove": "Odstranit",
|
|
||||||
"doAdd": "Přidat",
|
|
||||||
"doSignOut": "Odhlásit se",
|
|
||||||
"editAccountHtmlTitle": "Upravit účet",
|
|
||||||
"federatedIdentitiesHtmlTitle": "Propojené identity",
|
|
||||||
"accountLogHtmlTitle": "Log účtu",
|
|
||||||
"changePasswordHtmlTitle": "Změnit heslo",
|
|
||||||
"sessionsHtmlTitle": "Relace",
|
|
||||||
"accountManagementTitle": "Správa účtů Keycloak",
|
|
||||||
"authenticatorTitle": "Autentizátor",
|
|
||||||
"applicationsHtmlTitle": "Aplikace",
|
|
||||||
"authenticatorCode": "Jednorázový kód",
|
|
||||||
"email": "E-mail",
|
|
||||||
"firstName": "První křestní jméno",
|
|
||||||
"givenName": "Křestní jména",
|
|
||||||
"fullName": "Celé jméno",
|
|
||||||
"lastName": "Příjmení",
|
|
||||||
"familyName": "Rodinné jméno",
|
|
||||||
"password": "Heslo",
|
|
||||||
"passwordConfirm": "Nové heslo (znovu)",
|
|
||||||
"passwordNew": "Nové heslo",
|
|
||||||
"username": "Uživatelské jméno",
|
|
||||||
"address": "Adresa",
|
|
||||||
"street": "Ulice",
|
|
||||||
"locality": "Město nebo lokalita",
|
|
||||||
"region": "Kraj",
|
|
||||||
"postal_code": "PSČ",
|
|
||||||
"country": "Stát",
|
|
||||||
"emailVerified": "E-mail ověřen",
|
|
||||||
"gssDelegationCredential": "GSS delegované oprávnění",
|
|
||||||
"role_admin": "Správce",
|
|
||||||
"role_realm-admin": "Správce realmu",
|
|
||||||
"role_create-realm": "Vytvořit realm",
|
|
||||||
"role_view-realm": "Zobrazit realm",
|
|
||||||
"role_view-users": "Zobrazit uživatele",
|
|
||||||
"role_view-applications": "Zobrazit aplikace",
|
|
||||||
"role_view-clients": "Zobrazit klienty",
|
|
||||||
"role_view-events": "Zobrazit události",
|
|
||||||
"role_view-identity-providers": "Zobrazit poskytovatele identity",
|
|
||||||
"role_manage-realm": "Spravovat realm",
|
|
||||||
"role_manage-users": "Spravovat uživatele",
|
|
||||||
"role_manage-applications": "Spravovat aplikace",
|
|
||||||
"role_manage-identity-providers": "Spravovat poskytovatele identity",
|
|
||||||
"role_manage-clients": "Spravovat klienty",
|
|
||||||
"role_manage-events": "Spravovat události",
|
|
||||||
"role_view-profile": "Zobrazit profil",
|
|
||||||
"role_manage-account": "Spravovat účet",
|
|
||||||
"role_manage-account-links": "Spravovat odkazy na účet",
|
|
||||||
"role_read-token": "Číst token",
|
|
||||||
"role_offline-access": "Přístup offline",
|
|
||||||
"role_uma_authorization": "Získání oprávnění",
|
|
||||||
"client_account": "Účet",
|
|
||||||
"client_security-admin-console": "Administrátorská bezpečnostní konzole",
|
|
||||||
"client_admin-cli": "Administrátorské CLI",
|
|
||||||
"client_realm-management": "Správa realmů",
|
|
||||||
"client_broker": "Broker",
|
|
||||||
"requiredFields": "Požadovaná pole",
|
|
||||||
"allFieldsRequired": "Všechna pole vyžadovaná",
|
|
||||||
"backToApplication": "« Zpět na aplikaci",
|
|
||||||
"backTo": "Zpět na {0}",
|
|
||||||
"date": "Datum",
|
|
||||||
"event": "Událost",
|
|
||||||
"ip": "IP",
|
|
||||||
"client": "Klient",
|
|
||||||
"clients": "Klienti",
|
|
||||||
"details": "Podrobnosti",
|
|
||||||
"started": "Zahájeno",
|
|
||||||
"lastAccess": "Poslední přístup",
|
|
||||||
"expires": "Vyprší",
|
|
||||||
"applications": "Aplikace",
|
|
||||||
"account": "Účet",
|
|
||||||
"federatedIdentity": "Propojená identita",
|
|
||||||
"authenticator": "Autentizátor",
|
|
||||||
"sessions": "Relace",
|
|
||||||
"log": "Log",
|
|
||||||
"application": "Aplikace",
|
|
||||||
"availablePermissions": "Dostupná oprávnění",
|
|
||||||
"grantedPermissions": "Udělené oprávnění",
|
|
||||||
"grantedPersonalInfo": "Poskytnuté osobní informace",
|
|
||||||
"additionalGrants": "Dodatečné oprávnění",
|
|
||||||
"action": "Akce",
|
|
||||||
"inResource": "v",
|
|
||||||
"fullAccess": "Úplný přístup",
|
|
||||||
"offlineToken": "Offline Token",
|
|
||||||
"revoke": "Zrušit oprávnění",
|
|
||||||
"configureAuthenticators": "Konfigurované autentizátory",
|
|
||||||
"mobile": "Mobilní",
|
|
||||||
"totpStep1": "Nainstalujte jednu z následujících aplikací",
|
|
||||||
"totpStep2": "Otevřete aplikaci a naskenujte čárový kód",
|
|
||||||
"totpStep3": "Zadejte jednorázový kód poskytnutý aplikací a klepnutím na tlačítko Uložit dokončete nastavení.",
|
|
||||||
"totpManualStep2": "Otevřete aplikaci a zadejte klíč",
|
|
||||||
"totpManualStep3": "Použijte následující hodnoty konfigurace, pokud aplikace umožňuje jejich nastavení",
|
|
||||||
"totpUnableToScan": "Nelze skenovat?",
|
|
||||||
"totpScanBarcode": "Skenovat čárový kód?",
|
|
||||||
"totp.totp": "Založeno na čase",
|
|
||||||
"totp.hotp": "Založeno na čítači",
|
|
||||||
"totpType": "Typ",
|
|
||||||
"totpAlgorithm": "Algoritmus",
|
|
||||||
"totpDigits": "Číslice",
|
|
||||||
"totpInterval": "Interval",
|
|
||||||
"totpCounter": "Čítač",
|
|
||||||
"missingUsernameMessage": "Zadejte uživatelské jméno.",
|
|
||||||
"missingFirstNameMessage": "Zadejte prosím křestní jméno.",
|
|
||||||
"invalidEmailMessage": "Neplatná e-mailová adresa.",
|
|
||||||
"missingLastNameMessage": "Zadejte prosím příjmení.",
|
|
||||||
"missingEmailMessage": "Zadejte prosím e-mail.",
|
|
||||||
"missingPasswordMessage": "Zadejte prosím heslo.",
|
|
||||||
"notMatchPasswordMessage": "Hesla se neshodují.",
|
|
||||||
"missingTotpMessage": "Zadejte prosím kód autentizátoru.",
|
|
||||||
"invalidPasswordExistingMessage": "Neplatné stávající heslo.",
|
|
||||||
"invalidPasswordConfirmMessage": "Nová hesla se neshodují.",
|
|
||||||
"invalidTotpMessage": "Neplatný kód autentizátoru.",
|
|
||||||
"usernameExistsMessage": "Uživatelské jméno již existuje.",
|
|
||||||
"emailExistsMessage": "E-mail již existuje.",
|
|
||||||
"readOnlyUserMessage": "Nemůžete svůj účet aktualizovat, protože je pouze pro čtení.",
|
|
||||||
"readOnlyUsernameMessage": "Nemůžete aktualizovat své uživatelské jméno, protože je pouze pro čtení.",
|
|
||||||
"readOnlyPasswordMessage": "Nemůžete aktualizovat své heslo, protože váš účet je jen pro čtení.",
|
|
||||||
"successTotpMessage": "Ověření pomocí OTP úspěšně konfigurováno.",
|
|
||||||
"successTotpRemovedMessage": "Ověření pomocí OTP úspěšně odstraněno.",
|
|
||||||
"successGrantRevokedMessage": "Oprávnění bylo úspěšně zrušeno.",
|
|
||||||
"accountUpdatedMessage": "Váš účet byl aktualizován.",
|
|
||||||
"accountPasswordUpdatedMessage": "Vaše heslo bylo aktualizováno.",
|
|
||||||
"missingIdentityProviderMessage": "Chybějící poskytovatel identity.",
|
|
||||||
"invalidFederatedIdentityActionMessage": "Neplatná nebo chybějící akce.",
|
|
||||||
"identityProviderNotFoundMessage": "Poskytovatel identity nenalezen.",
|
|
||||||
"federatedIdentityLinkNotActiveMessage": "Tato identita již není aktivní.",
|
|
||||||
"federatedIdentityRemovingLastProviderMessage": "Nemůžete odstranit poslední propojenou identitu, protože nemáte heslo.",
|
|
||||||
"identityProviderRedirectErrorMessage": "Nepodařilo se přesměrovat na poskytovatele identity.",
|
|
||||||
"identityProviderRemovedMessage": "Poskytovatel identity byl úspěšně odstraněn.",
|
|
||||||
"identityProviderAlreadyLinkedMessage": "Propojená identita vrácená uživatelem {0} je již propojena s jiným uživatelem.",
|
|
||||||
"staleCodeAccountMessage": "Platnost vypršela. Zkuste to ještě jednou.",
|
|
||||||
"consentDenied": "Souhlas byl zamítnut.",
|
|
||||||
"accountDisabledMessage": "Účet je zakázán, kontaktujte správce.",
|
|
||||||
"accountTemporarilyDisabledMessage": "Účet je dočasně zakázán, kontaktujte správce nebo zkuste to později.",
|
|
||||||
"invalidPasswordMinLengthMessage": "Neplatné heslo: musí obsahovat minimálně {0} malých znaků.",
|
|
||||||
"invalidPasswordMinLowerCaseCharsMessage": "Neplatné heslo: musí obsahovat minimálně {0} malé znaky.",
|
|
||||||
"invalidPasswordMinDigitsMessage": "Neplatné heslo: musí obsahovat nejméně {0} číslic.",
|
|
||||||
"invalidPasswordMinUpperCaseCharsMessage": "Neplatné heslo: musí obsahovat nejméně {0} velkých písmenen.",
|
|
||||||
"invalidPasswordMinSpecialCharsMessage": "Neplatné heslo: musí obsahovat nejméně {0} speciálních znaků.",
|
|
||||||
"invalidPasswordNotUsernameMessage": "Neplatné heslo: nesmí být totožné s uživatelským jménem.",
|
|
||||||
"invalidPasswordRegexPatternMessage": "Neplatné heslo: neshoduje se zadaným regulárním výrazem.",
|
|
||||||
"invalidPasswordHistoryMessage": "Neplatné heslo: Nesmí se opakovat žádné z posledních {0} hesel.",
|
|
||||||
"invalidPasswordBlacklistedMessage": "Neplatné heslo: heslo je na černé listině.",
|
|
||||||
"invalidPasswordGenericMessage": "Neplatné heslo: nové heslo neodpovídá pravidlům hesla.",
|
|
||||||
};
|
|
||||||
|
|
||||||
export default messages;
|
|
||||||
/* spell-checker: enable */
|
|
@ -1,156 +0,0 @@
|
|||||||
//This code was automatically generated by running dist/bin/generate-i18n-messages.js
|
|
||||||
//PLEASE DO NOT EDIT MANUALLY
|
|
||||||
|
|
||||||
/* spell-checker: disable */
|
|
||||||
const messages = {
|
|
||||||
"doSave": "Speichern",
|
|
||||||
"doCancel": "Abbrechen",
|
|
||||||
"doLogOutAllSessions": "Alle Sitzungen abmelden",
|
|
||||||
"doRemove": "Entfernen",
|
|
||||||
"doAdd": "Hinzufügen",
|
|
||||||
"doSignOut": "Abmelden",
|
|
||||||
"editAccountHtmlTitle": "Benutzerkonto bearbeiten",
|
|
||||||
"federatedIdentitiesHtmlTitle": "Föderierte Identitäten",
|
|
||||||
"accountLogHtmlTitle": "Benutzerkonto Log",
|
|
||||||
"changePasswordHtmlTitle": "Passwort Ändern",
|
|
||||||
"sessionsHtmlTitle": "Sitzungen",
|
|
||||||
"accountManagementTitle": "Keycloak Benutzerkontoverwaltung",
|
|
||||||
"authenticatorTitle": "Mehrfachauthentifizierung",
|
|
||||||
"applicationsHtmlTitle": "Applikationen",
|
|
||||||
"authenticatorCode": "One-time Code",
|
|
||||||
"email": "E-Mail",
|
|
||||||
"firstName": "Vorname",
|
|
||||||
"givenName": "Vorname",
|
|
||||||
"fullName": "Voller Name",
|
|
||||||
"lastName": "Nachname",
|
|
||||||
"familyName": "Nachname",
|
|
||||||
"password": "Passwort",
|
|
||||||
"passwordConfirm": "Passwort bestätigen",
|
|
||||||
"passwordNew": "Neues Passwort",
|
|
||||||
"username": "Benutzername",
|
|
||||||
"address": "Adresse",
|
|
||||||
"street": "Straße",
|
|
||||||
"region": "Staat, Provinz, Region",
|
|
||||||
"postal_code": "PLZ",
|
|
||||||
"locality": "Stadt oder Ortschaft",
|
|
||||||
"country": "Land",
|
|
||||||
"emailVerified": "E-Mail verifiziert",
|
|
||||||
"gssDelegationCredential": "GSS delegierte Berechtigung",
|
|
||||||
"role_admin": "Admin",
|
|
||||||
"role_realm-admin": "Realm Admin",
|
|
||||||
"role_create-realm": "Realm erstellen",
|
|
||||||
"role_view-realm": "Realm ansehen",
|
|
||||||
"role_view-users": "Benutzer ansehen",
|
|
||||||
"role_view-applications": "Applikationen ansehen",
|
|
||||||
"role_view-clients": "Clients ansehen",
|
|
||||||
"role_view-events": "Events ansehen",
|
|
||||||
"role_view-identity-providers": "Identity Provider ansehen",
|
|
||||||
"role_manage-realm": "Realm verwalten",
|
|
||||||
"role_manage-users": "Benutzer verwalten",
|
|
||||||
"role_manage-applications": "Applikationen verwalten",
|
|
||||||
"role_manage-identity-providers": "Identity Provider verwalten",
|
|
||||||
"role_manage-clients": "Clients verwalten",
|
|
||||||
"role_manage-events": "Events verwalten",
|
|
||||||
"role_view-profile": "Profile ansehen",
|
|
||||||
"role_manage-account": "Profile verwalten",
|
|
||||||
"role_manage-account-links": "Profil-Links verwalten",
|
|
||||||
"role_read-token": "Token lesen",
|
|
||||||
"role_offline-access": "Offline-Zugriff",
|
|
||||||
"role_uma_authorization": "Berechtigungen einholen",
|
|
||||||
"client_account": "Clientkonto",
|
|
||||||
"client_security-admin-console": "Security Adminkonsole",
|
|
||||||
"client_realm-management": "Realm-Management",
|
|
||||||
"client_broker": "Broker",
|
|
||||||
"requiredFields": "Erforderliche Felder",
|
|
||||||
"allFieldsRequired": "Alle Felder sind erforderlich",
|
|
||||||
"backToApplication": "« Zurück zur Applikation",
|
|
||||||
"backTo": "Zurück zu {0}",
|
|
||||||
"date": "Datum",
|
|
||||||
"event": "Ereignis",
|
|
||||||
"ip": "IP",
|
|
||||||
"client": "Client",
|
|
||||||
"clients": "Clients",
|
|
||||||
"details": "Details",
|
|
||||||
"started": "Startdatum",
|
|
||||||
"lastAccess": "Letzter Zugriff",
|
|
||||||
"expires": "Ablaufdatum",
|
|
||||||
"applications": "Applikationen",
|
|
||||||
"account": "Benutzerkonto",
|
|
||||||
"federatedIdentity": "Föderierte Identität",
|
|
||||||
"authenticator": "Mehrfachauthentifizierung",
|
|
||||||
"sessions": "Sitzungen",
|
|
||||||
"log": "Log",
|
|
||||||
"application": "Applikation",
|
|
||||||
"availablePermissions": "verfügbare Berechtigungen",
|
|
||||||
"grantedPermissions": "gewährte Berechtigungen",
|
|
||||||
"grantedPersonalInfo": "gewährte persönliche Informationen",
|
|
||||||
"additionalGrants": "zusätzliche Berechtigungen",
|
|
||||||
"action": "Aktion",
|
|
||||||
"inResource": "in",
|
|
||||||
"fullAccess": "Vollzugriff",
|
|
||||||
"offlineToken": "Offline-Token",
|
|
||||||
"revoke": "Berechtigung widerrufen",
|
|
||||||
"configureAuthenticators": "Mehrfachauthentifizierung konfigurieren",
|
|
||||||
"mobile": "Mobil",
|
|
||||||
"totpStep1": "Installieren Sie eine der folgenden Applikationen auf Ihrem Smartphone:",
|
|
||||||
"totpStep2": "Öffnen Sie die Applikation und scannen Sie den Barcode.",
|
|
||||||
"totpStep3": "Geben Sie den von der Applikation generierten One-time Code ein und klicken Sie auf Speichern.",
|
|
||||||
"totpManualStep2": "Öffnen Sie die Applikation und geben Sie den folgenden Schlüssel ein.",
|
|
||||||
"totpManualStep3": "Verwenden Sie die folgenden Konfigurationswerte, falls Sie diese für die Applikation anpassen können:",
|
|
||||||
"totpUnableToScan": "Sie können den Barcode nicht scannen?",
|
|
||||||
"totpScanBarcode": "Barcode scannen?",
|
|
||||||
"totp.totp": "zeitbasiert (time-based)",
|
|
||||||
"totp.hotp": "zählerbasiert (counter-based)",
|
|
||||||
"totpType": "Typ",
|
|
||||||
"totpAlgorithm": "Algorithmus",
|
|
||||||
"totpDigits": "Ziffern",
|
|
||||||
"totpInterval": "Intervall",
|
|
||||||
"totpCounter": "Zähler",
|
|
||||||
"missingUsernameMessage": "Bitte geben Sie einen Benutzernamen ein.",
|
|
||||||
"missingFirstNameMessage": "Bitte geben Sie einen Vornamen ein.",
|
|
||||||
"invalidEmailMessage": "Ungültige E-Mail Adresse.",
|
|
||||||
"missingLastNameMessage": "Bitte geben Sie einen Nachnamen ein.",
|
|
||||||
"missingEmailMessage": "Bitte geben Sie eine E-Mail Adresse ein.",
|
|
||||||
"missingPasswordMessage": "Bitte geben Sie ein Passwort ein.",
|
|
||||||
"notMatchPasswordMessage": "Die Passwörter sind nicht identisch.",
|
|
||||||
"missingTotpMessage": "Bitte geben Sie den One-time Code ein.",
|
|
||||||
"invalidPasswordExistingMessage": "Das aktuelle Passwort ist ungültig.",
|
|
||||||
"invalidPasswordConfirmMessage": "Die Passwortbestätigung ist nicht identisch.",
|
|
||||||
"invalidTotpMessage": "Ungültiger One-time Code.",
|
|
||||||
"usernameExistsMessage": "Der Benutzername existiert bereits.",
|
|
||||||
"emailExistsMessage": "Die E-Mail-Adresse existiert bereits.",
|
|
||||||
"readOnlyUserMessage": "Sie können Ihr Benutzerkonto nicht ändern, da es schreibgeschützt ist.",
|
|
||||||
"readOnlyUsernameMessage": "Sie können Ihren Benutzernamen nicht ändern, da er schreibgeschützt ist.",
|
|
||||||
"readOnlyPasswordMessage": "Sie können Ihr Passwort nicht ändern, da es schreibgeschützt ist.",
|
|
||||||
"successTotpMessage": "Mehrfachauthentifizierung erfolgreich konfiguriert.",
|
|
||||||
"successTotpRemovedMessage": "Mehrfachauthentifizierung erfolgreich entfernt.",
|
|
||||||
"successGrantRevokedMessage": "Berechtigung erfolgreich widerrufen.",
|
|
||||||
"accountUpdatedMessage": "Ihr Benutzerkonto wurde aktualisiert.",
|
|
||||||
"accountPasswordUpdatedMessage": "Ihr Passwort wurde aktualisiert.",
|
|
||||||
"missingIdentityProviderMessage": "Identity Provider nicht angegeben.",
|
|
||||||
"invalidFederatedIdentityActionMessage": "Ungültige oder fehlende Aktion.",
|
|
||||||
"identityProviderNotFoundMessage": "Angegebener Identity Provider nicht gefunden.",
|
|
||||||
"federatedIdentityLinkNotActiveMessage": "Diese Identität ist nicht mehr aktiv.",
|
|
||||||
"federatedIdentityRemovingLastProviderMessage": "Sie können den letzten Eintrag nicht entfernen, da Sie kein Passwort haben.",
|
|
||||||
"identityProviderRedirectErrorMessage": "Fehler bei der Weiterleitung zum Identity Provider.",
|
|
||||||
"identityProviderRemovedMessage": "Identity Provider erfolgreich entfernt.",
|
|
||||||
"identityProviderAlreadyLinkedMessage": "Die föderierte Identität von {0} ist bereits einem anderen Benutzer zugewiesen.",
|
|
||||||
"staleCodeAccountMessage": "Diese Seite ist nicht mehr gültig, bitte versuchen Sie es noch einmal.",
|
|
||||||
"consentDenied": "Einverständnis verweigert.",
|
|
||||||
"accountDisabledMessage": "Ihr Benutzerkonto ist gesperrt, bitte kontaktieren Sie den Admin.",
|
|
||||||
"accountTemporarilyDisabledMessage":
|
|
||||||
"Ihr Benutzerkonto ist temporär gesperrt, bitte kontaktieren Sie den Admin oder versuchen Sie es später noch einmal.",
|
|
||||||
"invalidPasswordMinLengthMessage": "Ungültiges Passwort: Es muss mindestens {0} Zeichen lang sein.",
|
|
||||||
"invalidPasswordMinLowerCaseCharsMessage": "Ungültiges Passwort: Es muss mindestens {0} Kleinbuchstaben beinhalten.",
|
|
||||||
"invalidPasswordMinDigitsMessage": "Ungültiges Passwort: Es muss mindestens {0} Zahl(en) beinhalten.",
|
|
||||||
"invalidPasswordMinUpperCaseCharsMessage": "Ungültiges Passwort: Es muss mindestens {0} Großbuchstaben beinhalten.",
|
|
||||||
"invalidPasswordMinSpecialCharsMessage": "Ungültiges Passwort: Es muss mindestens {0} Sonderzeichen beinhalten.",
|
|
||||||
"invalidPasswordNotUsernameMessage": "Ungültiges Passwort: Es darf nicht gleich sein wie der Benutzername.",
|
|
||||||
"invalidPasswordRegexPatternMessage": "Ungültiges Passwort: Es entspricht nicht dem Regex-Muster.",
|
|
||||||
"invalidPasswordHistoryMessage": "Ungültiges Passwort: Es darf nicht einem der letzten {0} Passwörter entsprechen.",
|
|
||||||
"invalidPasswordBlacklistedMessage": "Ungültiges Passwort: Das Passwort steht auf der Blocklist (schwarzen Liste).",
|
|
||||||
"invalidPasswordGenericMessge": "Ungültiges Passwort: Das neue Passwort verletzt die Passwort-Richtlinien.",
|
|
||||||
};
|
|
||||||
|
|
||||||
export default messages;
|
|
||||||
/* spell-checker: enable */
|
|
@ -1,325 +0,0 @@
|
|||||||
//This code was automatically generated by running dist/bin/generate-i18n-messages.js
|
|
||||||
//PLEASE DO NOT EDIT MANUALLY
|
|
||||||
|
|
||||||
/* spell-checker: disable */
|
|
||||||
const messages = {
|
|
||||||
"doSave": "Save",
|
|
||||||
"doCancel": "Cancel",
|
|
||||||
"doLogOutAllSessions": "Log out all sessions",
|
|
||||||
"doRemove": "Remove",
|
|
||||||
"doAdd": "Add",
|
|
||||||
"doSignOut": "Sign Out",
|
|
||||||
"doLogIn": "Log In",
|
|
||||||
"doLink": "Link",
|
|
||||||
"editAccountHtmlTitle": "Edit Account",
|
|
||||||
"personalInfoHtmlTitle": "Personal Info",
|
|
||||||
"federatedIdentitiesHtmlTitle": "Federated Identities",
|
|
||||||
"accountLogHtmlTitle": "Account Log",
|
|
||||||
"changePasswordHtmlTitle": "Change Password",
|
|
||||||
"deviceActivityHtmlTitle": "Device Activity",
|
|
||||||
"sessionsHtmlTitle": "Sessions",
|
|
||||||
"accountManagementTitle": "Keycloak Account Management",
|
|
||||||
"authenticatorTitle": "Authenticator",
|
|
||||||
"applicationsHtmlTitle": "Applications",
|
|
||||||
"linkedAccountsHtmlTitle": "Linked Accounts",
|
|
||||||
"accountManagementWelcomeMessage": "Welcome to Keycloak Account Management",
|
|
||||||
"personalInfoIntroMessage": "Manage your basic information",
|
|
||||||
"accountSecurityTitle": "Account Security",
|
|
||||||
"accountSecurityIntroMessage": "Control your password and account access",
|
|
||||||
"applicationsIntroMessage": "Track and manage your app permission to access your account",
|
|
||||||
"resourceIntroMessage": "Share your resources among team members",
|
|
||||||
"passwordLastUpdateMessage": "Your password was updated at",
|
|
||||||
"updatePasswordTitle": "Update Password",
|
|
||||||
"updatePasswordMessageTitle": "Make sure you choose a strong password",
|
|
||||||
"updatePasswordMessage":
|
|
||||||
"A strong password contains a mix of numbers, letters, and symbols. It is hard to guess, does not resemble a real word, and is only used for this account.",
|
|
||||||
"personalSubTitle": "Your Personal Info",
|
|
||||||
"personalSubMessage": "Manage this basic information: your first name, last name and email",
|
|
||||||
"authenticatorCode": "One-time code",
|
|
||||||
"email": "Email",
|
|
||||||
"firstName": "First name",
|
|
||||||
"givenName": "Given name",
|
|
||||||
"fullName": "Full name",
|
|
||||||
"lastName": "Last name",
|
|
||||||
"familyName": "Family name",
|
|
||||||
"password": "Password",
|
|
||||||
"currentPassword": "Current Password",
|
|
||||||
"passwordConfirm": "Confirmation",
|
|
||||||
"passwordNew": "New Password",
|
|
||||||
"username": "Username",
|
|
||||||
"address": "Address",
|
|
||||||
"street": "Street",
|
|
||||||
"locality": "City or Locality",
|
|
||||||
"region": "State, Province, or Region",
|
|
||||||
"postal_code": "Zip or Postal code",
|
|
||||||
"country": "Country",
|
|
||||||
"emailVerified": "Email verified",
|
|
||||||
"gssDelegationCredential": "GSS Delegation Credential",
|
|
||||||
"profileScopeConsentText": "User profile",
|
|
||||||
"emailScopeConsentText": "Email address",
|
|
||||||
"addressScopeConsentText": "Address",
|
|
||||||
"phoneScopeConsentText": "Phone number",
|
|
||||||
"offlineAccessScopeConsentText": "Offline Access",
|
|
||||||
"samlRoleListScopeConsentText": "My Roles",
|
|
||||||
"rolesScopeConsentText": "User roles",
|
|
||||||
"role_admin": "Admin",
|
|
||||||
"role_realm-admin": "Realm Admin",
|
|
||||||
"role_create-realm": "Create realm",
|
|
||||||
"role_view-realm": "View realm",
|
|
||||||
"role_view-users": "View users",
|
|
||||||
"role_view-applications": "View applications",
|
|
||||||
"role_view-clients": "View clients",
|
|
||||||
"role_view-events": "View events",
|
|
||||||
"role_view-identity-providers": "View identity providers",
|
|
||||||
"role_view-consent": "View consents",
|
|
||||||
"role_manage-realm": "Manage realm",
|
|
||||||
"role_manage-users": "Manage users",
|
|
||||||
"role_manage-applications": "Manage applications",
|
|
||||||
"role_manage-identity-providers": "Manage identity providers",
|
|
||||||
"role_manage-clients": "Manage clients",
|
|
||||||
"role_manage-events": "Manage events",
|
|
||||||
"role_view-profile": "View profile",
|
|
||||||
"role_manage-account": "Manage account",
|
|
||||||
"role_manage-account-links": "Manage account links",
|
|
||||||
"role_manage-consent": "Manage consents",
|
|
||||||
"role_read-token": "Read token",
|
|
||||||
"role_offline-access": "Offline access",
|
|
||||||
"role_uma_authorization": "Obtain permissions",
|
|
||||||
"client_account": "Account",
|
|
||||||
"client_account-console": "Account Console",
|
|
||||||
"client_security-admin-console": "Security Admin Console",
|
|
||||||
"client_admin-cli": "Admin CLI",
|
|
||||||
"client_realm-management": "Realm Management",
|
|
||||||
"client_broker": "Broker",
|
|
||||||
"requiredFields": "Required fields",
|
|
||||||
"allFieldsRequired": "All fields required",
|
|
||||||
"backToApplication": "« Back to application",
|
|
||||||
"backTo": "Back to {0}",
|
|
||||||
"date": "Date",
|
|
||||||
"event": "Event",
|
|
||||||
"ip": "IP",
|
|
||||||
"client": "Client",
|
|
||||||
"clients": "Clients",
|
|
||||||
"details": "Details",
|
|
||||||
"started": "Started",
|
|
||||||
"lastAccess": "Last Access",
|
|
||||||
"expires": "Expires",
|
|
||||||
"applications": "Applications",
|
|
||||||
"account": "Account",
|
|
||||||
"federatedIdentity": "Federated Identity",
|
|
||||||
"authenticator": "Authenticator",
|
|
||||||
"device-activity": "Device Activity",
|
|
||||||
"sessions": "Sessions",
|
|
||||||
"log": "Log",
|
|
||||||
"application": "Application",
|
|
||||||
"availableRoles": "Available Roles",
|
|
||||||
"grantedPermissions": "Granted Permissions",
|
|
||||||
"grantedPersonalInfo": "Granted Personal Info",
|
|
||||||
"additionalGrants": "Additional Grants",
|
|
||||||
"action": "Action",
|
|
||||||
"inResource": "in",
|
|
||||||
"fullAccess": "Full Access",
|
|
||||||
"offlineToken": "Offline Token",
|
|
||||||
"revoke": "Revoke Grant",
|
|
||||||
"configureAuthenticators": "Configured Authenticators",
|
|
||||||
"mobile": "Mobile",
|
|
||||||
"totpStep1": "Install one of the following applications on your mobile:",
|
|
||||||
"totpStep2": "Open the application and scan the barcode:",
|
|
||||||
"totpStep3": "Enter the one-time code provided by the application and click Save to finish the setup.",
|
|
||||||
"totpStep3DeviceName": "Provide a Device Name to help you manage your OTP devices.",
|
|
||||||
"totpManualStep2": "Open the application and enter the key:",
|
|
||||||
"totpManualStep3": "Use the following configuration values if the application allows setting them:",
|
|
||||||
"totpUnableToScan": "Unable to scan?",
|
|
||||||
"totpScanBarcode": "Scan barcode?",
|
|
||||||
"totp.totp": "Time-based",
|
|
||||||
"totp.hotp": "Counter-based",
|
|
||||||
"totpType": "Type",
|
|
||||||
"totpAlgorithm": "Algorithm",
|
|
||||||
"totpDigits": "Digits",
|
|
||||||
"totpInterval": "Interval",
|
|
||||||
"totpCounter": "Counter",
|
|
||||||
"totpDeviceName": "Device Name",
|
|
||||||
"missingUsernameMessage": "Please specify username.",
|
|
||||||
"missingFirstNameMessage": "Please specify first name.",
|
|
||||||
"invalidEmailMessage": "Invalid email address.",
|
|
||||||
"missingLastNameMessage": "Please specify last name.",
|
|
||||||
"missingEmailMessage": "Please specify email.",
|
|
||||||
"missingPasswordMessage": "Please specify password.",
|
|
||||||
"notMatchPasswordMessage": "Passwords don't match.",
|
|
||||||
"invalidUserMessage": "Invalid user",
|
|
||||||
"missingTotpMessage": "Please specify authenticator code.",
|
|
||||||
"missingTotpDeviceNameMessage": "Please specify device name.",
|
|
||||||
"invalidPasswordExistingMessage": "Invalid existing password.",
|
|
||||||
"invalidPasswordConfirmMessage": "Password confirmation doesn't match.",
|
|
||||||
"invalidTotpMessage": "Invalid authenticator code.",
|
|
||||||
"usernameExistsMessage": "Username already exists.",
|
|
||||||
"emailExistsMessage": "Email already exists.",
|
|
||||||
"readOnlyUserMessage": "You can't update your account as it is read-only.",
|
|
||||||
"readOnlyUsernameMessage": "You can't update your username as it is read-only.",
|
|
||||||
"readOnlyPasswordMessage": "You can't update your password as your account is read-only.",
|
|
||||||
"successTotpMessage": "Mobile authenticator configured.",
|
|
||||||
"successTotpRemovedMessage": "Mobile authenticator removed.",
|
|
||||||
"successGrantRevokedMessage": "Grant revoked successfully.",
|
|
||||||
"accountUpdatedMessage": "Your account has been updated.",
|
|
||||||
"accountPasswordUpdatedMessage": "Your password has been updated.",
|
|
||||||
"missingIdentityProviderMessage": "Identity provider not specified.",
|
|
||||||
"invalidFederatedIdentityActionMessage": "Invalid or missing action.",
|
|
||||||
"identityProviderNotFoundMessage": "Specified identity provider not found.",
|
|
||||||
"federatedIdentityLinkNotActiveMessage": "This identity is not active anymore.",
|
|
||||||
"federatedIdentityRemovingLastProviderMessage": "You can't remove last federated identity as you don't have a password.",
|
|
||||||
"identityProviderRedirectErrorMessage": "Failed to redirect to identity provider.",
|
|
||||||
"identityProviderRemovedMessage": "Identity provider removed successfully.",
|
|
||||||
"identityProviderAlreadyLinkedMessage": "Federated identity returned by {0} is already linked to another user.",
|
|
||||||
"staleCodeAccountMessage": "The page expired. Please try one more time.",
|
|
||||||
"consentDenied": "Consent denied.",
|
|
||||||
"accountDisabledMessage": "Account is disabled, contact your administrator.",
|
|
||||||
"accountTemporarilyDisabledMessage": "Account is temporarily disabled, contact your administrator or try again later.",
|
|
||||||
"invalidPasswordMinLengthMessage": "Invalid password: minimum length {0}.",
|
|
||||||
"invalidPasswordMinLowerCaseCharsMessage": "Invalid password: must contain at least {0} lower case characters.",
|
|
||||||
"invalidPasswordMinDigitsMessage": "Invalid password: must contain at least {0} numerical digits.",
|
|
||||||
"invalidPasswordMinUpperCaseCharsMessage": "Invalid password: must contain at least {0} upper case characters.",
|
|
||||||
"invalidPasswordMinSpecialCharsMessage": "Invalid password: must contain at least {0} special characters.",
|
|
||||||
"invalidPasswordNotUsernameMessage": "Invalid password: must not be equal to the username.",
|
|
||||||
"invalidPasswordRegexPatternMessage": "Invalid password: fails to match regex pattern(s).",
|
|
||||||
"invalidPasswordHistoryMessage": "Invalid password: must not be equal to any of last {0} passwords.",
|
|
||||||
"invalidPasswordBlacklistedMessage": "Invalid password: password is blacklisted.",
|
|
||||||
"invalidPasswordGenericMessage": "Invalid password: new password doesn't match password policies.",
|
|
||||||
"myResources": "My Resources",
|
|
||||||
"myResourcesSub": "My resources",
|
|
||||||
"doDeny": "Deny",
|
|
||||||
"doRevoke": "Revoke",
|
|
||||||
"doApprove": "Approve",
|
|
||||||
"doRemoveSharing": "Remove Sharing",
|
|
||||||
"doRemoveRequest": "Remove Request",
|
|
||||||
"peopleAccessResource": "People with access to this resource",
|
|
||||||
"resourceManagedPolicies": "Permissions granting access to this resource",
|
|
||||||
"resourceNoPermissionsGrantingAccess": "No permissions granting access to this resource",
|
|
||||||
"anyAction": "Any action",
|
|
||||||
"description": "Description",
|
|
||||||
"name": "Name",
|
|
||||||
"scopes": "Scopes",
|
|
||||||
"resource": "Resource",
|
|
||||||
"user": "User",
|
|
||||||
"peopleSharingThisResource": "People sharing this resource",
|
|
||||||
"shareWithOthers": "Share with others",
|
|
||||||
"needMyApproval": "Need my approval",
|
|
||||||
"requestsWaitingApproval": "Your requests waiting approval",
|
|
||||||
"icon": "Icon",
|
|
||||||
"requestor": "Requestor",
|
|
||||||
"owner": "Owner",
|
|
||||||
"resourcesSharedWithMe": "Resources shared with me",
|
|
||||||
"permissionRequestion": "Permission Requestion",
|
|
||||||
"permission": "Permission",
|
|
||||||
"shares": "share(s)",
|
|
||||||
"notBeingShared": "This resource is not being shared.",
|
|
||||||
"notHaveAnyResource": "You don't have any resources",
|
|
||||||
"noResourcesSharedWithYou": "There are no resources shared with you",
|
|
||||||
"havePermissionRequestsWaitingForApproval": "You have {0} permission request(s) waiting for approval.",
|
|
||||||
"clickHereForDetails": "Click here for details.",
|
|
||||||
"resourceIsNotBeingShared": "The resource is not being shared",
|
|
||||||
"locale_ca": "Català",
|
|
||||||
"locale_cs": "Čeština",
|
|
||||||
"locale_de": "Deutsch",
|
|
||||||
"locale_en": "English",
|
|
||||||
"locale_es": "Español",
|
|
||||||
"locale_fr": "Français",
|
|
||||||
"locale_it": "Italian",
|
|
||||||
"locale_ja": "日本語",
|
|
||||||
"locale_nl": "Nederlands",
|
|
||||||
"locale_no": "Norsk",
|
|
||||||
"locale_lt": "Lietuvių",
|
|
||||||
"locale_pt-BR": "Português (Brasil)",
|
|
||||||
"locale_ru": "Русский",
|
|
||||||
"locale_sk": "Slovenčina",
|
|
||||||
"locale_sv": "Svenska",
|
|
||||||
"locale_tr": "Turkish",
|
|
||||||
"locale_zh-CN": "中文简体",
|
|
||||||
"applicaitonName": "Name",
|
|
||||||
"applicationType": "Application Type",
|
|
||||||
"applicationInUse": "In-use app only",
|
|
||||||
"clearAllFilter": "Clear all filters",
|
|
||||||
"activeFilters": "Active filters",
|
|
||||||
"filterByName": "Filter By Name ...",
|
|
||||||
"allApps": "All applications",
|
|
||||||
"internalApps": "Internal applications",
|
|
||||||
"thirdpartyApps": "Third-Party applications",
|
|
||||||
"appResults": "Results",
|
|
||||||
"clientNotFoundMessage": "Client not found.",
|
|
||||||
"authorizedProvider": "Authorized Provider",
|
|
||||||
"authorizedProviderMessage": "Authorized Providers linked with your account",
|
|
||||||
"identityProvider": "Identity Provider",
|
|
||||||
"identityProviderMessage": "To link your account with identity providers you have configured",
|
|
||||||
"socialLogin": "Social Login",
|
|
||||||
"userDefined": "User Defined",
|
|
||||||
"removeAccess": "Remove Access",
|
|
||||||
"removeAccessMessage": "You will need to grant access again, if you want to use this app account.",
|
|
||||||
"authenticatorStatusMessage": "Two-factor authentication is currently",
|
|
||||||
"authenticatorFinishSetUpTitle": "Your Two-Factor Authentication",
|
|
||||||
"authenticatorFinishSetUpMessage":
|
|
||||||
"Each time you sign in to your Keycloak account, you will be asked to provide a two-factor authentication code.",
|
|
||||||
"authenticatorSubTitle": "Set Up Two-Factor Authentication",
|
|
||||||
"authenticatorSubMessage": "To enhance the security of your account, enable at least one of the available two-factor authentication methods.",
|
|
||||||
"authenticatorMobileTitle": "Mobile Authenticator",
|
|
||||||
"authenticatorMobileMessage": "Use mobile Authenticator to get Verification codes as the two-factor authentication.",
|
|
||||||
"authenticatorMobileFinishSetUpMessage": "The authenticator has been bound to your phone.",
|
|
||||||
"authenticatorActionSetup": "Set up",
|
|
||||||
"authenticatorSMSTitle": "SMS Code",
|
|
||||||
"authenticatorSMSMessage": "Keycloak will send the Verification code to your phone as the two-factor authentication.",
|
|
||||||
"authenticatorSMSFinishSetUpMessage": "Text messages are sent to",
|
|
||||||
"authenticatorDefaultStatus": "Default",
|
|
||||||
"authenticatorChangePhone": "Change Phone Number",
|
|
||||||
"authenticatorBackupCodesTitle": "Backup Codes",
|
|
||||||
"authenticatorBackupCodesMessage": "Get your 8-digit backup codes",
|
|
||||||
"authenticatorBackupCodesFinishSetUpMessage": "12 backup codes were generated at this time. Each one can be used once.",
|
|
||||||
"authenticatorMobileSetupTitle": "Mobile Authenticator Setup",
|
|
||||||
"smscodeIntroMessage": "Enter your phone number and a verification code will be sent to your phone.",
|
|
||||||
"mobileSetupStep1": "Install an authenticator application on your phone. The applications listed here are supported.",
|
|
||||||
"mobileSetupStep2": "Open the application and scan the barcode:",
|
|
||||||
"mobileSetupStep3": "Enter the one-time code provided by the application and click Save to finish the setup.",
|
|
||||||
"scanBarCode": "Want to scan the barcode?",
|
|
||||||
"enterBarCode": "Enter the one-time code",
|
|
||||||
"doCopy": "Copy",
|
|
||||||
"doFinish": "Finish",
|
|
||||||
"authenticatorSMSCodeSetupTitle": "SMS Code Setup",
|
|
||||||
"chooseYourCountry": "Choose your country",
|
|
||||||
"enterYourPhoneNumber": "Enter your phone number",
|
|
||||||
"sendVerficationCode": "Send Verification Code",
|
|
||||||
"enterYourVerficationCode": "Enter your verification code",
|
|
||||||
"authenticatorBackupCodesSetupTitle": "Backup Codes Setup",
|
|
||||||
"backupcodesIntroMessage":
|
|
||||||
"If you lose access to your phone, you can still log into your account through backup codes. Keep them somewhere safe and accessible.",
|
|
||||||
"realmName": "Realm",
|
|
||||||
"doDownload": "Download",
|
|
||||||
"doPrint": "Print",
|
|
||||||
"backupCodesTips-1": "Each backup code can be used once.",
|
|
||||||
"backupCodesTips-2": "These codes were generated on",
|
|
||||||
"generateNewBackupCodes": "Generate New Backup Codes",
|
|
||||||
"backupCodesTips-3": "When you generate new backup codes, the current codes will not work anymore.",
|
|
||||||
"backtoAuthenticatorPage": "Back to Authenticator Page",
|
|
||||||
"resources": "Resources",
|
|
||||||
"sharedwithMe": "Shared with Me",
|
|
||||||
"share": "Share",
|
|
||||||
"sharedwith": "Shared with",
|
|
||||||
"accessPermissions": "Access Permissions",
|
|
||||||
"permissionRequests": "Permission Requests",
|
|
||||||
"approve": "Approve",
|
|
||||||
"approveAll": "Approve all",
|
|
||||||
"people": "people",
|
|
||||||
"perPage": "per page",
|
|
||||||
"currentPage": "Current Page",
|
|
||||||
"sharetheResource": "Share the resource",
|
|
||||||
"group": "Group",
|
|
||||||
"selectPermission": "Select Permission",
|
|
||||||
"addPeople": "Add people to share your resource with",
|
|
||||||
"addTeam": "Add team to share your resource with",
|
|
||||||
"myPermissions": "My Permissions",
|
|
||||||
"waitingforApproval": "Waiting for approval",
|
|
||||||
"anyPermission": "Any Permission",
|
|
||||||
"openshift.scope.user_info": "User information",
|
|
||||||
"openshift.scope.user_check-access": "User access information",
|
|
||||||
"openshift.scope.user_full": "Full Access",
|
|
||||||
"openshift.scope.list-projects": "List projects",
|
|
||||||
};
|
|
||||||
|
|
||||||
export default messages;
|
|
||||||
/* spell-checker: enable */
|
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user