Skip to content

Commit

Permalink
Merge pull request #29 from rundeck/typescript-client-fix
Browse files Browse the repository at this point in the history
TypeScript Fixes
  • Loading branch information
fdevans authored Jan 11, 2023
2 parents fea4b95 + 39b69a0 commit 770e7da
Show file tree
Hide file tree
Showing 7 changed files with 152 additions and 46 deletions.
4 changes: 3 additions & 1 deletion .env
Original file line number Diff line number Diff line change
@@ -1,2 +1,4 @@
## Set pro options if applicable
RUNDECK_IMAGE=rundeck/rundeck:4.4.0
RUNDECK_IMAGE=rundeck/rundeck:4.8.0
RUNDECK_USER=admin
RUNDECK_PASSWORD=admin
32 changes: 25 additions & 7 deletions client/rundeck-cli/src/commands/addUsers.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import {Argv} from'yargs'
import {waitForRundeckReady, createStoragePassword, createProject, asyncForEach, createStoragePrivateKey, createAcl} from '../lib/util'
import {createWaitForRundeckReady, runeckLoginToken, asyncForEach, loadConfigYaml} from '../lib/util'

import { Rundeck, PasswordCredentialProvider}from 'ts-rundeck'
import { Rundeck, PasswordCredentialProvider} from 'ts-rundeck'
import Path from 'path'
import * as FS from '../async/fs'
import { JobUuidOption } from 'ts-rundeck/dist/lib/models'
Expand All @@ -12,6 +12,8 @@ import { collapseTextChangeRangesAcrossMultipleVersions } from 'typescript'

interface Opts {
rundeck_url: string,
username: string,
password: string,
config_file: string,
path: string,
debug: boolean
Expand All @@ -36,6 +38,20 @@ builder(yargs: Argv) {
default: false,
require: true
})
.option("ru", {
alias: "username",
describe: "Rundeck Username",
type: 'string',
default: false,
require: true
})
.option("rp", {
alias: "password",
describe: "Rundeck password",
type: 'string',
default: false,
require: true
})
.option("f", {
alias: "config_file",
describe: "Config file",
Expand Down Expand Up @@ -69,12 +85,14 @@ builder(yargs: Argv) {

const users: User[] = config.users;

const username = 'admin'
const password = 'admin'
const client = new Rundeck(new PasswordCredentialProvider(rundeckUrl, username, password), {baseUri: rundeckUrl})
const username = opts.username
const password = opts.password
const client = new Rundeck(new PasswordCredentialProvider(rundeckUrl, username, password), {noRetryPolicy: true, baseUri: rundeckUrl})

console.log("Waiting for Rundeck");
await waitForRundeckReady(client);
console.log("Waiting for Rundeck");
await createWaitForRundeckReady(
() => client,5 * 60 * 1000)

console.log("Rundeck started!!!");

console.log("----------------------------------");
Expand Down
54 changes: 31 additions & 23 deletions client/rundeck-cli/src/commands/loadProject.ts
Original file line number Diff line number Diff line change
@@ -1,17 +1,20 @@
import {Argv} from'yargs'
import {waitForRundeckReady, createStoragePassword, createProject, asyncForEach, createStoragePrivateKey, createAcl} from '../lib/util'
import {createWaitForRundeckReady, createStoragePassword, createProject, asyncForEach, createStoragePrivateKey, loadConfigYaml, createAcl, runeckLoginToken} from '../lib/util'

import { Rundeck, PasswordCredentialProvider}from 'ts-rundeck'
import { Rundeck, PasswordCredentialProvider, TokenCredentialProvider,}from 'ts-rundeck'
import Path from 'path'
import * as FS from '../async/fs'
import { JobUuidOption } from 'ts-rundeck/dist/lib/models'
import YAML from 'yaml'
import fetch from 'node-fetch';
import FormData from 'form-data';
import { collapseTextChangeRangesAcrossMultipleVersions } from 'typescript'
import { RequestPolicyFactory } from "@azure/ms-rest-js"

interface Opts {
rundeck_url: string,
username: string,
password: string,
config_file: string,
path: string,
debug: boolean
Expand Down Expand Up @@ -47,6 +50,20 @@ builder(yargs: Argv) {
default: false,
require: true
})
.option("ru", {
alias: "username",
describe: "Rundeck Username",
type: 'string',
default: false,
require: true
})
.option("rp", {
alias: "password",
describe: "Rundeck password",
type: 'string',
default: false,
require: true
})
.option("f", {
alias: "config_file",
describe: "Config file",
Expand Down Expand Up @@ -83,12 +100,19 @@ builder(yargs: Argv) {
const keys: Key[] = config.keys;
const acls: Acl[] = config.acls;

const username = opts.username
const password = opts.password

const rundeckAuth = await runeckLoginToken(rundeckUrl, username, password)
const token = rundeckAuth["token"]
const client = rundeckAuth["client"]

const username = 'admin'
const password = 'admin'
const client = new Rundeck(new PasswordCredentialProvider(rundeckUrl, username, password), {baseUri: rundeckUrl})
console.log("Waiting for Rundeck");
await waitForRundeckReady(client);
await createWaitForRundeckReady(
() => new Rundeck(new PasswordCredentialProvider(rundeckUrl, username, password), {noRetryPolicy: true, baseUri: rundeckUrl}),
5 * 60 * 1000
);

console.log("Rundeck started!!!");

console.log("----------------------------------");
Expand Down Expand Up @@ -182,22 +206,6 @@ builder(yargs: Argv) {
if(projectImport == true){
const importFileName = Path.join(path, project.archive);

const tokenResponse = await client.sendRequest({
headers: {'Content-Type': 'application/json'},
pathTemplate: `/api/36/tokens/{username}`,
pathParameters: {username: username},
baseUrl: rundeckUrl,
method: 'POST',
body: {
"user": username,
"roles": [
"admin",
],
"duration": "30d"
}
});

let token = tokenResponse.parsedBody.token
console.log("import project");

try{
Expand Down Expand Up @@ -225,4 +233,4 @@ builder(yargs: Argv) {

}

module.exports = new LoadProjectCommand()
module.exports = new LoadProjectCommand()
38 changes: 30 additions & 8 deletions client/rundeck-cli/src/commands/updateProject.ts
Original file line number Diff line number Diff line change
@@ -1,9 +1,11 @@
import {Argv} from'yargs'
import {waitForRundeckReady, updateProperty} from '../lib/util'
import { Rundeck, PasswordCredentialProvider}from 'ts-rundeck'
import {createWaitForRundeckReady, updateProperty, asyncForEach, loadConfigYaml, runeckLoginToken} from '../lib/util'
import {Rundeck, PasswordCredentialProvider} from 'ts-rundeck'

interface Opts {
rundeck_url: string,
username: string,
password: string,
project_name: string,
input_value: string,
debug: boolean
Expand All @@ -22,6 +24,20 @@ builder(yargs: Argv) {
default: false,
require: true
})
.option("ru", {
alias: "username",
describe: "Rundeck Username",
type: 'string',
default: false,
require: true
})
.option("rp", {
alias: "password",
describe: "Rundeck password",
type: 'string',
default: false,
require: true
})
.option("p", {
alias: "project_name",
describe: "Project List",
Expand All @@ -48,13 +64,19 @@ builder(yargs: Argv) {
const project_name = opts.project_name
const input_value = opts.input_value
const rundeckUrl = opts.rundeck_url

const username = opts.username
const password = opts.password
const client = new Rundeck(new PasswordCredentialProvider(rundeckUrl, username, password), {baseUri: rundeckUrl})


const client = new Rundeck(new PasswordCredentialProvider(rundeckUrl, 'admin', 'admin'), {baseUri: rundeckUrl})
console.log("----------++++++++++++------------")
console.log("Update Project Script Start")
console.log("Waiting for rundeck")
await waitForRundeckReady(client)
console.log("----------++++++++++++------------");
console.log("Update Project Script Start");

console.log("Waiting for rundeck");
await createWaitForRundeckReady(
() => client,5 * 60 * 1000)


console.log("Rundeck started!!!")

console.log("----------------------------------");
Expand Down
62 changes: 58 additions & 4 deletions client/rundeck-cli/src/lib/util.ts
Original file line number Diff line number Diff line change
@@ -1,14 +1,17 @@
import {Rundeck} from 'ts-rundeck'
import * as dotenv from "dotenv";
import YAML from 'yaml'
import {WebResource, RequestPolicy, RequestPolicyFactory, RequestPolicyOptions, BaseRequestPolicy, HttpOperationResponse} from '@azure/ms-rest-js'
import { Rundeck, PasswordCredentialProvider, TokenCredentialProvider}from 'ts-rundeck'

import {combineCookies} from 'ts-rundeck/dist/util'

export function sleep(ms: number): Promise<{}> {
return new Promise( res => {
setTimeout(res, ms)
})
}


export async function waitForRundeckReady(client: Rundeck, timeout = 500000) {
await createWaitForRundeckReady(() => client, timeout)
}
Expand Down Expand Up @@ -129,10 +132,10 @@ export function loadConfigYaml(importYaml: string): any{

const env = process.env;

console.log(importYaml);
//console.log(importYaml);

Object.keys(env).forEach(function(key) {
console.log('export ' + key + '="' + env[key] +'"');
//console.log('export ' + key + '="' + env[key] +'"');
let value = env[key] as string;

if(importYaml.includes(key)){
Expand All @@ -141,8 +144,59 @@ export function loadConfigYaml(importYaml: string): any{
}

});
console.log(importYaml);
//console.log(importYaml);
const config = YAML.parse(importYaml)
return config

}


export function cookieEnrichPolicy(cookies: string[]): RequestPolicyFactory {
return {
create: (nextPolicy: RequestPolicy, options: RequestPolicyOptions) => {
return new CookieEnrichPolicy(nextPolicy, options, cookies)
}
}
}

/** Enriches each request with a set of cookies */
export class CookieEnrichPolicy extends BaseRequestPolicy {
constructor(nextPolicy: RequestPolicy, options: RequestPolicyOptions, readonly cookies: string[]) {
super(nextPolicy, options)
}

async sendRequest(webResource: WebResource): Promise<HttpOperationResponse> {
const reqCookies = webResource.headers.get('cookie')
const combinedCookies = combineCookies(reqCookies, this.cookies)

webResource.headers.set('cookie', combinedCookies.join(';'))

return await this._nextPolicy.sendRequest(webResource)
}
}


export async function runeckLoginToken(rundeckUrl: string, username: string, password: string ){
const clientPasswordAuth = new Rundeck(new PasswordCredentialProvider(rundeckUrl, username, password), {noRetryPolicy: true, baseUri: rundeckUrl})

const tokenResponse = await clientPasswordAuth.sendRequest({
headers: {'Content-Type': 'application/json'},
pathTemplate: `/api/36/tokens/{username}`,
pathParameters: {username: username},
baseUrl: rundeckUrl,
method: 'POST',
body: {
"user": username,
"roles": [
"admin",
],
"duration": "30d"
}
});

let token = tokenResponse.parsedBody.token

const client = new Rundeck(new TokenCredentialProvider(token),{baseUri: rundeckUrl})
return {token: token, client: client}

}
6 changes: 3 additions & 3 deletions client/scripts/init.sh
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ echo "Init Script Starting"
## Place Init Script Code here as needed. This is run as part of DockerFile steps for Client node.
ssh-keygen -q -t rsa -N '' -f /rundeck-cli/data/keys/id_rsa

cat $CONFIG_FILE
#cat $CONFIG_FILE

chmod +x bin/cli

Expand All @@ -11,8 +11,8 @@ ls -last /rundeck-cli/data
echo "-----"

# load project
./bin/cli load --rundeck_url $RUNDECK_URL --config_file "$CONFIG_FILE" --path /rundeck-cli
./bin/cli addUsers --rundeck_url $RUNDECK_URL --config_file "$CONFIG_FILE" --path /rundeck-cli
./bin/cli load --rundeck_url $RUNDECK_URL --username $RUNDECK_USER --password $RUNDECK_PASSWORD --config_file "$CONFIG_FILE" --path /rundeck-cli
./bin/cli addUsers --rundeck_url $RUNDECK_URL --username $RUNDECK_USER --password $RUNDECK_PASSWORD --config_file "$CONFIG_FILE" --path /rundeck-cli

# update a value in a specified project
# See rundeckpro/sensu-demo repo for detailed example use
Expand Down
2 changes: 2 additions & 0 deletions docker-compose.yml
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,8 @@ services:
RUNDECK_URL: http://rundeck:4440
RUNDECK_API_URL: http://localhost:4440
RUNDECK_TOKEN: u5gJoKU3NpS6IC4lFmPQFfDkUlZnQOXp
RUNDECK_USER: ${RUNDECK_USER:-admin}
RUNDECK_PASSWORD: ${RUNDECK_PASSWORD:-admin}
volumes:
- shared-volume:/rundeck-cli/data/keys/
- ./data/user.aclpolicy:/rundeck-cli/data/user.aclpolicy
Expand Down

0 comments on commit 770e7da

Please sign in to comment.