n8cn / packages /cli /src /sso.ee /saml /saml-validator.ts
gallyga's picture
Add n8n Chinese version
aec3094
import { Logger } from '@n8n/backend-common';
import { Service } from '@n8n/di';
import { Constants, IdentityProvider } from 'samlify';
import type { IdentityProviderInstance } from 'samlify';
import type { XMLFileInfo, XMLLintOptions, XMLValidationResult } from 'xmllint-wasm';
import { InvalidSamlMetadataError } from './errors/invalid-saml-metadata.error';
@Service()
export class SamlValidator {
private xmlMetadata: XMLFileInfo;
private xmlProtocol: XMLFileInfo;
private preload: XMLFileInfo[] = [];
constructor(private readonly logger: Logger) {}
private xmllint: {
validateXML: (options: XMLLintOptions) => Promise<XMLValidationResult>;
};
async init() {
await this.loadSchemas();
this.xmllint = await import('xmllint-wasm');
}
validateIdentiyProvider(idp: IdentityProviderInstance) {
const binding = idp.entityMeta.getSingleSignOnService(Constants.wording.binding.redirect);
if (typeof binding !== 'string') {
throw new InvalidSamlMetadataError('only SAML redirect binding is supported.');
}
}
async validateMetadata(metadata: string): Promise<boolean> {
const validXML = await this.validateXml('metadata', metadata);
if (validXML) {
const idp = IdentityProvider({
metadata,
});
this.validateIdentiyProvider(idp);
}
return validXML;
}
async validateResponse(response: string): Promise<boolean> {
return await this.validateXml('response', response);
}
// dynamically load schema files
private async loadSchemas(): Promise<void> {
this.xmlProtocol = (await import('./schema/saml-schema-protocol-2.0.xsd')).xmlFileInfo;
this.xmlMetadata = (await import('./schema/saml-schema-metadata-2.0.xsd')).xmlFileInfo;
this.preload = (
await Promise.all([
// SAML
import('./schema/saml-schema-assertion-2.0.xsd'),
import('./schema/xmldsig-core-schema.xsd'),
import('./schema/xenc-schema.xsd'),
import('./schema/xml.xsd'),
// WS-Federation
import('./schema/ws-federation.xsd'),
import('./schema/oasis-200401-wss-wssecurity-secext-1.0.xsd'),
import('./schema/oasis-200401-wss-wssecurity-utility-1.0.xsd'),
import('./schema/ws-addr.xsd'),
import('./schema/metadata-exchange.xsd'),
import('./schema/ws-securitypolicy-1.2.xsd'),
import('./schema/ws-authorization.xsd'),
])
).map((m) => m.xmlFileInfo);
}
private async validateXml(type: 'metadata' | 'response', contents: string): Promise<boolean> {
const fileName = `${type}.xml`;
const schema = type === 'metadata' ? [this.xmlMetadata] : [this.xmlProtocol];
const preload = [type === 'metadata' ? this.xmlProtocol : this.xmlMetadata, ...this.preload];
try {
const validationResult = await this.xmllint.validateXML({
xml: [{ fileName, contents }],
extension: 'schema',
schema,
preload,
});
if (validationResult?.valid) {
this.logger.debug(`SAML ${type} is valid`);
return true;
} else {
this.logger.debug(`SAML ${type} is invalid`);
this.logger.warn(
validationResult
? validationResult.errors
.map((error) => `${error.message} - ${error.rawMessage}`)
.join('\n')
: '',
);
}
} catch (error) {
// eslint-disable-next-line @typescript-eslint/no-unsafe-argument
this.logger.warn(error);
}
return false;
}
}