Skip to content

Commit 9bbf53f

Browse files
committed
fix: make AZURE_DEVOPS_AUTH_METHOD parameter case-insensitive
1 parent 91c349c commit 9bbf53f

File tree

4 files changed

+132
-4
lines changed

4 files changed

+132
-4
lines changed

README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -120,7 +120,7 @@ Key environment variables include:
120120

121121
| Variable | Description | Required | Default |
122122
| ------------------------------ | --------------------------------------------------------------- | ------------------ | ------------------ |
123-
| `AZURE_DEVOPS_AUTH_METHOD` | Authentication method (`pat`, `azure-identity`, or `azure-cli`) | No | `azure-identity` |
123+
| `AZURE_DEVOPS_AUTH_METHOD` | Authentication method (`pat`, `azure-identity`, or `azure-cli`) - case-insensitive | No | `azure-identity` |
124124
| `AZURE_DEVOPS_ORG` | Azure DevOps organization name | No | Extracted from URL |
125125
| `AZURE_DEVOPS_ORG_URL` | Full URL to your Azure DevOps organization | Yes | - |
126126
| `AZURE_DEVOPS_PAT` | Personal Access Token (for PAT auth) | Only with PAT auth | - |

docs/authentication.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -135,7 +135,7 @@ Azure CLI authentication uses the `AzureCliCredential` class from the `@azure/id
135135

136136
| Environment Variable | Description | Required | Default |
137137
| ------------------------------ | --------------------------------------------------------------- | ---------------------------- | ---------------- |
138-
| `AZURE_DEVOPS_AUTH_METHOD` | Authentication method (`pat`, `azure-identity`, or `azure-cli`) | No | `azure-identity` |
138+
| `AZURE_DEVOPS_AUTH_METHOD` | Authentication method (`pat`, `azure-identity`, or `azure-cli`) - case-insensitive | No | `azure-identity` |
139139
| `AZURE_DEVOPS_ORG_URL` | Full URL to your Azure DevOps organization | Yes | - |
140140
| `AZURE_DEVOPS_PAT` | Personal Access Token (for PAT auth) | Only with PAT auth | - |
141141
| `AZURE_DEVOPS_DEFAULT_PROJECT` | Default project if none specified | No | - |

src/index.spec.unit.ts

+100
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,100 @@
1+
import { normalizeAuthMethod } from './index';
2+
import { AuthenticationMethod } from './shared/auth/auth-factory';
3+
4+
describe('index', () => {
5+
describe('normalizeAuthMethod', () => {
6+
it('should return AzureIdentity when authMethodStr is undefined', () => {
7+
// Arrange
8+
const authMethodStr = undefined;
9+
10+
// Act
11+
const result = normalizeAuthMethod(authMethodStr);
12+
13+
// Assert
14+
expect(result).toBe(AuthenticationMethod.AzureIdentity);
15+
});
16+
17+
it('should return AzureIdentity when authMethodStr is empty', () => {
18+
// Arrange
19+
const authMethodStr = '';
20+
21+
// Act
22+
const result = normalizeAuthMethod(authMethodStr);
23+
24+
// Assert
25+
expect(result).toBe(AuthenticationMethod.AzureIdentity);
26+
});
27+
28+
it('should handle PersonalAccessToken case-insensitively', () => {
29+
// Arrange
30+
const variations = [
31+
'pat',
32+
'PAT',
33+
'Pat',
34+
'pAt',
35+
'paT',
36+
];
37+
38+
// Act & Assert
39+
variations.forEach(variant => {
40+
expect(normalizeAuthMethod(variant)).toBe(AuthenticationMethod.PersonalAccessToken);
41+
});
42+
});
43+
44+
it('should handle AzureIdentity case-insensitively', () => {
45+
// Arrange
46+
const variations = [
47+
'azure-identity',
48+
'AZURE-IDENTITY',
49+
'Azure-Identity',
50+
'azure-Identity',
51+
'Azure-identity',
52+
];
53+
54+
// Act & Assert
55+
variations.forEach(variant => {
56+
expect(normalizeAuthMethod(variant)).toBe(AuthenticationMethod.AzureIdentity);
57+
});
58+
});
59+
60+
it('should handle AzureCli case-insensitively', () => {
61+
// Arrange
62+
const variations = [
63+
'azure-cli',
64+
'AZURE-CLI',
65+
'Azure-Cli',
66+
'azure-Cli',
67+
'Azure-cli',
68+
];
69+
70+
// Act & Assert
71+
variations.forEach(variant => {
72+
expect(normalizeAuthMethod(variant)).toBe(AuthenticationMethod.AzureCli);
73+
});
74+
});
75+
76+
it('should return AzureIdentity for unrecognized values', () => {
77+
// Arrange
78+
const unrecognized = [
79+
'unknown',
80+
'azureCli', // no hyphen
81+
'azureIdentity', // no hyphen
82+
'personal-access-token', // not matching enum value
83+
'cli',
84+
'identity',
85+
];
86+
87+
// Act & Assert (mute stderr for warning messages)
88+
const originalStderrWrite = process.stderr.write;
89+
process.stderr.write = jest.fn();
90+
91+
try {
92+
unrecognized.forEach(value => {
93+
expect(normalizeAuthMethod(value)).toBe(AuthenticationMethod.AzureIdentity);
94+
});
95+
} finally {
96+
process.stderr.write = originalStderrWrite;
97+
}
98+
});
99+
});
100+
});

src/index.ts

+30-2
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,35 @@ import dotenv from 'dotenv';
99
import { AzureDevOpsConfig } from './shared/types';
1010
import { AuthenticationMethod } from './shared/auth/auth-factory';
1111

12+
/**
13+
* Normalize auth method string to a valid AuthenticationMethod enum value
14+
* in a case-insensitive manner
15+
*
16+
* @param authMethodStr The auth method string from environment variable
17+
* @returns A valid AuthenticationMethod value
18+
*/
19+
export function normalizeAuthMethod(authMethodStr?: string): AuthenticationMethod {
20+
if (!authMethodStr) {
21+
return AuthenticationMethod.AzureIdentity; // Default
22+
}
23+
24+
// Convert to lowercase for case-insensitive comparison
25+
const normalizedMethod = authMethodStr.toLowerCase();
26+
27+
// Check against known enum values (as lowercase strings)
28+
if (normalizedMethod === AuthenticationMethod.PersonalAccessToken.toLowerCase()) {
29+
return AuthenticationMethod.PersonalAccessToken;
30+
} else if (normalizedMethod === AuthenticationMethod.AzureIdentity.toLowerCase()) {
31+
return AuthenticationMethod.AzureIdentity;
32+
} else if (normalizedMethod === AuthenticationMethod.AzureCli.toLowerCase()) {
33+
return AuthenticationMethod.AzureCli;
34+
}
35+
36+
// If not recognized, log a warning and use the default
37+
process.stderr.write(`WARNING: Unrecognized auth method '${authMethodStr}'. Using default (${AuthenticationMethod.AzureIdentity}).\n`);
38+
return AuthenticationMethod.AzureIdentity;
39+
}
40+
1241
// Load environment variables
1342
dotenv.config();
1443

@@ -25,8 +54,7 @@ function getConfig(): AzureDevOpsConfig {
2554

2655
return {
2756
organizationUrl: process.env.AZURE_DEVOPS_ORG_URL || '',
28-
authMethod: (process.env.AZURE_DEVOPS_AUTH_METHOD ||
29-
AuthenticationMethod.AzureIdentity) as AuthenticationMethod,
57+
authMethod: normalizeAuthMethod(process.env.AZURE_DEVOPS_AUTH_METHOD),
3058
personalAccessToken: process.env.AZURE_DEVOPS_PAT,
3159
defaultProject: process.env.AZURE_DEVOPS_DEFAULT_PROJECT,
3260
apiVersion: process.env.AZURE_DEVOPS_API_VERSION,

0 commit comments

Comments
 (0)