Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Validate against Banned-Passwords List #9727

Merged
merged 15 commits into from
Sep 26, 2023
Merged
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
Enhancement: Show error if password is on a banned password list

We now show a meaningful error if the user tries to set a public link password,
that's on a server side banned password list.

https://github.com/owncloud/web/pull/9727
https://github.com/owncloud/web/issues/9726
2 changes: 2 additions & 0 deletions dev/docker/ocis/password-policy-banned-passwords.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
password
12345678
2 changes: 2 additions & 0 deletions docker-compose.yml
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ services:
FRONTEND_SEARCH_MIN_LENGTH: "2"
FRONTEND_OCS_ENABLE_DENIALS: "true"
FRONTEND_FULL_TEXT_SEARCH_ENABLED: "true"
FRONTEND_PASSWORD_POLICY_BANNED_PASSWORDS_LIST: '/etc/ocis/password-policy-banned-passwords.txt'
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

added some demo data, can't hurt


# IDM
IDM_CREATE_DEMO_USERS: "${DEMO_USERS:-true}"
Expand Down Expand Up @@ -61,6 +62,7 @@ services:
# make the registry available to the app provider containers
MICRO_REGISTRY: "mdns"
volumes:
- ./dev/docker/ocis/password-policy-banned-passwords.txt:/etc/ocis/password-policy-banned-passwords.txt
- ./dev/docker/ocis.idp.config.yaml:/etc/ocis/idp.yaml
- ./dev/docker/ocis-ca:/var/lib/ocis/proxy
- ./dist:/web/dist
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -273,6 +273,8 @@ export default defineComponent({
if (this.type === 'password') {
additionalAttrs['password-policy'] = this.passwordPolicy
additionalAttrs['generate-password-method'] = this.generatePasswordMethod
additionalAttrs['has-warning'] = !!this.warningMessage
additionalAttrs['has-error'] = !!this.errorMessage
}
// Exclude listeners for events which are handled via methods in this component
// eslint-disable-next-line no-unused-vars
Expand Down
Original file line number Diff line number Diff line change
@@ -1,11 +1,12 @@
<template>
<div class="oc-text-input-password-wrapper">
<input
v-bind="$attrs"
v-model="password"
:type="showPassword ? 'text' : 'password'"
@input="onPasswordEntered"
/>
<div
class="oc-text-input-password-wrapper"
:class="{
'oc-text-input-password-wrapper-warning': hasWarning,
'oc-text-input-password-wrapper-danger': hasError
}"
>
<input v-bind="$attrs" v-model="password" :type="showPassword ? 'text' : 'password'" />
<oc-button
v-if="password"
v-oc-tooltip="$gettext('Show password')"
Expand All @@ -32,7 +33,7 @@
class="oc-text-input-generate-password-button oc-px-s oc-background-default"
appearance="raw"
size="small"
@click="showGeneratedPassword"
@click="generatePassword"
>
<oc-icon size="small" name="refresh" fill-type="line" />
</oc-button>
Expand Down Expand Up @@ -88,10 +89,20 @@ export default defineComponent({
type: Function as PropType<(...args: unknown[]) => string>,
required: false,
default: null
},
hasWarning: {
type: Boolean,
required: false,
default: false
},
hasError: {
type: Boolean,
required: false,
default: false
}
},
emits: ['passwordChallengeCompleted', 'passwordChallengeFailed'],
setup(props, { emit }) {
setup(props, { emit, attrs }) {
const { $gettext } = useGettext()
const password = ref(props.value)
const showPassword = ref(false)
Expand Down Expand Up @@ -123,17 +134,15 @@ export default defineComponent({
setTimeout(() => (copyPasswordIcon.value = copyPasswordIconInitial), 500)
}

const showGeneratedPassword = () => {
const generatePassword = () => {
const generatedPassword = props.generatePasswordMethod()
password.value = generatedPassword
showPassword.value = true
}

const onPasswordEntered = () => {
watch(password, (value) => {
passwordEntered.value = true
}

watch(password, (value) => {
if (!Object.keys(props.passwordPolicy).length) {
return
}
Expand All @@ -149,13 +158,12 @@ export default defineComponent({
$gettext,
password,
showPassword,
copyPasswordIcon,
showPasswordPolicyInformation,
testedPasswordPolicy,
generatePassword,
getPasswordPolicyRuleMessage,
copyPasswordToClipboard,
showGeneratedPassword,
copyPasswordIcon,
onPasswordEntered
copyPasswordToClipboard
}
}
})
Expand All @@ -177,6 +185,18 @@ export default defineComponent({
input:focus {
outline: none;
}

&-warning,
&-warning:focus {
border-color: var(--oc-color-swatch-warning-default) !important;
color: var(--oc-color-swatch-warning-default) !important;
}

&-danger,
&-danger:focus {
border-color: var(--oc-color-swatch-danger-default) !important;
color: var(--oc-color-swatch-danger-default) !important;
}
}
.oc-text-input-password-wrapper:focus-within {
border-color: var(--oc-color-swatch-passive-default);
Expand Down
58 changes: 35 additions & 23 deletions packages/web-app-files/src/components/SideBar/Shares/FileLinks.vue
Original file line number Diff line number Diff line change
Expand Up @@ -350,7 +350,13 @@ export default defineComponent({
},
methods: {
...mapActions('Files', ['addLink', 'updateLink', 'removeLink']),
...mapActions(['showMessage', 'showErrorMessage', 'createModal', 'hideModal']),
...mapActions([
'showMessage',
'showErrorMessage',
'createModal',
'hideModal',
'setModalInputErrorMessage'
]),
...mapMutations('Files', ['REMOVE_FILES']),

toggleLinkListCollapsed() {
Expand Down Expand Up @@ -394,7 +400,7 @@ export default defineComponent({
})
},

checkLinkToCreate({ link, onError = () => {} }) {
checkLinkToCreate({ link }) {
const paramsToCreate = this.getParamsForLink(link)

if (this.isPasswordEnforcedFor(link)) {
Expand All @@ -405,15 +411,15 @@ export default defineComponent({
passwordPolicyService: this.passwordPolicyService
},
(newPassword) => {
this.createLink({ params: { ...paramsToCreate, password: newPassword }, onError })
this.createLink({ params: { ...paramsToCreate, password: newPassword } })
}
)
} else {
this.createLink({ params: paramsToCreate, onError })
this.createLink({ params: paramsToCreate })
}
},

checkLinkToUpdate({ link, onSuccess = () => {} }) {
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I removed all those onSuccess and onError callbacks because they actually weren't used and added to confusion.

checkLinkToUpdate({ link }) {
const params = this.getParamsForLink(link)

if (!link.password && this.isPasswordEnforcedFor(link)) {
Expand All @@ -424,11 +430,11 @@ export default defineComponent({
passwordPolicyService: this.passwordPolicyService
},
(newPassword) => {
this.updatePublicLink({ params: { ...params, password: newPassword }, onSuccess })
this.updatePublicLink({ params: { ...params, password: newPassword } })
}
)
} else {
this.updatePublicLink({ params, onSuccess })
this.updatePublicLink({ params })
}
},

Expand Down Expand Up @@ -481,7 +487,7 @@ export default defineComponent({
}
},

async createLink({ params, onError = (e) => {} }) {
async createLink({ params }) {
let path = this.resource.path
// sharing a share root from the share jail -> use resource name as path
if (this.hasShareJail && path === '/') {
Expand All @@ -494,41 +500,47 @@ export default defineComponent({
storageId: this.resource.fileId || this.resource.id,
params
})
this.hideModal()
this.showMessage({
title: this.$gettext('Link was created successfully')
})
} catch (e) {
onError(e)
console.error(e)

// Human-readable error message is provided, for example when password is on banned list
if (e.status === 400) {
return this.setModalInputErrorMessage(this.$gettext(e.message))
}

this.showErrorMessage({
title: this.$gettext('Failed to create link'),
error: e
})
return
}

this.showMessage({
title: this.$gettext('Link was created successfully')
})
},

async updatePublicLink({ params, onSuccess = () => {}, onError = (e) => {} }) {
async updatePublicLink({ params }) {
try {
await this.updateLink({
id: params.id,
client: this.$client,
params
}).then(onSuccess)
})
this.hideModal()
this.showMessage({
title: this.$gettext('Link was updated successfully')
})
} catch (e) {
onError(e)
console.error(e)
// Human-readable error message is provided, for example when password is on banned list
if (e.statusCode === 400) {
return this.setModalInputErrorMessage(this.$gettext(e.message))
}

this.showErrorMessage({
title: this.$gettext('Failed to update link'),
error: e
})
return
}

this.showMessage({
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Shouldn't be here.

title: this.$gettext('Link was updated successfully')
})
},

deleteLinkConfirmation({ link }) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -472,9 +472,6 @@ export default defineComponent({
link: {
...this.link,
name
},
onSuccess: () => {
this.hideModal()
}
})
}
Expand All @@ -496,6 +493,7 @@ export default defineComponent({
inputPlaceholder: this.link.password ? '●●●●●●●●' : null,
inputType: 'password',
onCancel: this.hideModal,
onInput: () => this.setModalInputErrorMessage(''),
onPasswordChallengeCompleted: () => this.setModalConfirmButtonDisabled(false),
onPasswordChallengeFailed: () => this.setModalConfirmButtonDisabled(true),
onConfirm: (password) => {
Expand Down
18 changes: 4 additions & 14 deletions packages/web-app-files/src/quickActions.ts
Original file line number Diff line number Diff line change
Expand Up @@ -30,22 +30,12 @@ export function showQuickLinkPasswordModal({ $gettext, store, passwordPolicyServ
inputGeneratePasswordMethod: () => passwordPolicyService.generatePassword(),
inputLabel: $gettext('Password'),
inputType: 'password',
onInput: () => store.dispatch('setModalInputErrorMessage', ''),
onPasswordChallengeCompleted: () => store.dispatch('setModalConfirmButtonDisabled', false),
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I missed those in the last pr.

onPasswordChallengeFailed: () => store.dispatch('setModalConfirmButtonDisabled', true),
onCancel: () => store.dispatch('hideModal'),
onConfirm: async (password) => {
if (!password || password.trim() === '') {
store.dispatch('showErrorMessage', {
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not needed anymore as built in default password policy. Also closing the modal instantly was not correct, because we might want to show an error in the modal.

title: $gettext('Password cannot be empty')
})
} else {
await store.dispatch('hideModal')
onConfirm(password)
}
},
onInput: (password) => {
if (password.trim() === '') {
return store.dispatch('setModalInputErrorMessage', $gettext('Password cannot be empty'))
}
return store.dispatch('setModalInputErrorMessage', null)
onConfirm(password)
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -37,9 +37,11 @@ export class PasswordPolicyService {

private useDefaultRules(): boolean {
return (
Object.keys(this.capability).length === 0 ||
(Object.keys(this.capability).length === 1 &&
Object.keys(this.capability)[0] === 'max_characters')
!this.capability.min_characters &&
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

They are always set, defaults to 0, so had to change the logic

!this.capability.min_lowercase_characters &&
!this.capability.min_uppercase_characters &&
!this.capability.min_digits &&
!this.capability.min_special_characters
)
}

Expand Down
2 changes: 1 addition & 1 deletion packages/web-runtime/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@
"luxon": "^2.4.0",
"marked": "^4.0.12",
"oidc-client-ts": "^2.1.0",
"owncloud-sdk": "3.1.0-alpha.8",
"owncloud-sdk": "3.1.0-alpha.9",
"p-queue": "^6.6.2",
"pinia": "^2.1.3",
"portal-vue": "3.0.0",
Expand Down
5 changes: 4 additions & 1 deletion packages/web-runtime/src/helpers/additionalTranslations.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,5 +14,8 @@ export const additionalTranslations = {
'Virus "%{description}" detected. Please contact your administrator for more information.'
),
virusScan: $gettext('Scan for viruses'),
requestErrorDeniedByPolicy: $gettext('Operation denied due to security policies')
requestErrorDeniedByPolicy: $gettext('Operation denied due to security policies'),
ocsErrorPasswordOnBannedList: $gettext(
'Unfortunately, your password is commonly used. please pick a harder-to-guess password for your safety'
)
}
8 changes: 4 additions & 4 deletions pnpm-lock.yaml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.