Skip to content

Commit

Permalink
Merge pull request #9 from HyperPlay-Gaming/feat/generic_platforms
Browse files Browse the repository at this point in the history
[Feat] Support generic platform keys with YML
  • Loading branch information
BrettCleary authored Mar 12, 2024
2 parents 18968a9 + a5e7f0f commit b63d4bd
Show file tree
Hide file tree
Showing 15 changed files with 310 additions and 54 deletions.
7 changes: 6 additions & 1 deletion .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -6,4 +6,9 @@
/tmp
/yarn.lock
node_modules
oclif.manifest.json
oclif.manifest.json
mac_amd64.zip
mac_arm64.zip
mac_x64.zip
web.zip
windows_amd64.zip
10 changes: 0 additions & 10 deletions hyperplay.yml

This file was deleted.

42 changes: 36 additions & 6 deletions package-lock.json

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

4 changes: 2 additions & 2 deletions package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@hyperplay/cli",
"version": "2.9.2",
"version": "2.10.0",
"description": "Hyperplay CLI",
"author": "HyperPlay Labs, Inc.",
"bin": {
Expand All @@ -19,7 +19,7 @@
"dependencies": {
"@oclif/core": "^1.8.0",
"@oclif/plugin-help": "^5",
"@valist/sdk": "^2.9.3",
"@valist/sdk": "^2.9.4",
"archiver": "^7.0.0",
"axios": "^1.6.7",
"axios-cookiejar-support": "^5.0.0",
Expand Down
41 changes: 15 additions & 26 deletions src/commands/publish.ts
Original file line number Diff line number Diff line change
@@ -1,15 +1,15 @@
import { Command, CliUx } from '@oclif/core';
import { ethers } from 'ethers';
import { create, ReleaseConfig, SupportedPlatform } from '@valist/sdk';
import YAML from 'yaml';
import * as fs from 'node:fs';
import { create } from '@valist/sdk';
import * as flags from '../flags';
import { select } from '../keys';
import { CookieJar } from 'tough-cookie';
import axios from 'axios';
import { wrapper } from 'axios-cookiejar-support';
import { loginAndPublish } from '../api';
import { uploadRelease } from '../releases';
import { parseYml } from '../yml';
import { FlagOutput } from '@oclif/core/lib/interfaces';

export default class Publish extends Command {
static provider?: ethers.Signer;
Expand All @@ -31,7 +31,9 @@ export default class Publish extends Command {
'darwin_arm64': flags.darwin_arm64,
'windows_amd64': flags.windows_amd64,
'skip_hyperplay_publish': flags.skip_hyperplay_publish,
'channel': flags.channel
'channel': flags.channel,
'use-yml': flags.useYml,
'yml-path': flags.ymlPath
}

static args = [
Expand Down Expand Up @@ -59,39 +61,26 @@ export default class Publish extends Command {
return new ethers.Wallet(privateKey, provider);
}

async parseConfig(){
// ts having issues passing these as args so duplicating the parse here
const { args, flags } = await this.parse(Publish);

let config: ReleaseConfig;
if (args.account && args.project && args.release) {
config = new ReleaseConfig(args.account, args.project, args.release);
const platformFlags: SupportedPlatform[] = ["web", "darwin_amd64", "darwin_arm64", "linux_amd64", "linux_arm64", "windows_amd64", "windows_arm64", "android_arm64"]
for (const platform of platformFlags){
if (flags[platform])
config.platforms[platform] = flags[platform]
}

} else if(fs.existsSync('hyperplay.yml')){
const data = fs.readFileSync('hyperplay.yml', 'utf8');
config = YAML.parse(data);
}
else {
/* eslint-disable-next-line */
async parseConfig(args: {[name: string]: any;}, flags: FlagOutput){
const parsed = parseYml(args, flags)
if (parsed === undefined){
this.error('Account, project, and release were not supplied and hyperplay.yml does not exist')
}

const {config, yamlConfig} = parsed;
if (!config.account) this.error('invalid account name');
if (!config.project) this.error('invalid project name');
if (!config.release) this.error('invalid release name');
if (!config.platforms) this.error('no platforms configured');

return config
return {config, yamlConfig}
}

// if no args are passed, use the yml
public async run(): Promise<void> {
const { flags } = await this.parse(Publish);
const config = await this.parseConfig()
const { args, flags } = await this.parse(Publish);
const {config, yamlConfig} = await this.parseConfig(args, flags)

const fullReleaseName = `${config.account}/${config.project}/${config.release}`;
console.log('Publishing', { platforms: config.platforms }, `as ${fullReleaseName}`);
Expand Down Expand Up @@ -125,7 +114,7 @@ export default class Publish extends Command {
}

CliUx.ux.action.start('uploading files');
const release = await uploadRelease(valist, config);
const release = await uploadRelease(valist, config, yamlConfig);
CliUx.ux.action.stop();
CliUx.ux.log(`successfully uploaded files to IPFS: ${release.external_url}`);

Expand Down
12 changes: 12 additions & 0 deletions src/flags.ts
Original file line number Diff line number Diff line change
Expand Up @@ -90,3 +90,15 @@ export const channel = Flags.string({
default: 'main',
env: 'HYPERPLAY_TARGET_CHANNEL'
});

export const useYml = Flags.boolean({
description: 'Use hyperplay.yml to get platform config',
default: false,
env: 'HYPERPLAY_USE_YML'
})

export const ymlPath = Flags.string({
description: 'Path to yml file containing publish args',
default: '',
env: 'HYPERPLAY_YML_PATH'
})
6 changes: 5 additions & 1 deletion src/releases.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,12 +4,16 @@ import { PlatformsMetaInterface } from '@valist/sdk/dist/typesShared';
import fs from 'fs';
import path from 'path';
import { zipDirectory } from './zip';
import { YamlConfig } from './types';

export async function uploadRelease(valist: Client, config: ReleaseConfig) {
export async function uploadRelease(valist: Client, config: ReleaseConfig, yamlConfig?: YamlConfig) {
/* eslint-disable-next-line */
const platformEntries = Object.entries(config.platforms).filter(([_key, value]) => value !== "");

const updatedPlatformEntries = await Promise.all(platformEntries.map(async ([platform, filePath]) => {
if (yamlConfig && yamlConfig.platforms[platform] && !yamlConfig.platforms[platform].zip) {
return [platform, filePath]
}
const zipPath = `./${path.basename(filePath)}.zip`;
await zipDirectory(filePath, zipPath);
return [platform, zipPath] as [string, string];
Expand Down
11 changes: 11 additions & 0 deletions src/types.d.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
export interface PlatformYamlConfig {
path: string,
zip: boolean
}

export interface YamlConfig {
account: string,
project: string,
release: string,
platforms: Record<string, PlatformYamlConfig>
}
51 changes: 51 additions & 0 deletions src/yml.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
import { ReleaseConfig, SupportedPlatform } from "@valist/sdk";
import { YamlConfig } from "./types";
import fs from 'fs';
import YAML from 'yaml';
import { FlagOutput } from "@oclif/core/lib/interfaces";

/* eslint-disable-next-line */
export function parseYml(args: {[name: string]: any;}, flags: FlagOutput): {config: ReleaseConfig, yamlConfig: YamlConfig | undefined} | undefined{
let config: ReleaseConfig;
let yamlConfig: YamlConfig | undefined = undefined;
const flagPath = flags['yml-path']
const ymlPath = flagPath ? flagPath : 'hyperplay.yml'
// cli args and flags
if (args.account && args.project && args.release && !flags['use-yml']) {
config = new ReleaseConfig(args.account, args.project, args.release);
const platformFlags: SupportedPlatform[] = ["web", "darwin_amd64", "darwin_arm64", "linux_amd64", "linux_arm64", "windows_amd64", "windows_arm64", "android_arm64"]
for (const platform of platformFlags){
if (flags[platform])
config.platforms[platform] = flags[platform]
}

// using hyperplay.yml
} else if(fs.existsSync(ymlPath)){
const data = fs.readFileSync(ymlPath, 'utf8');
yamlConfig = YAML.parse(data);

const configPlatforms: Record<string, string> = {}
for (const [key, value] of Object.entries(yamlConfig!.platforms)){
configPlatforms[key] = value.path
}
if (yamlConfig === undefined){
return undefined
}
config = {...yamlConfig, platforms: configPlatforms}

// override yaml if cli args are passed for acct, project, or release
if (args.account) {
config.account = args.account
}
if (args.project) {
config.project = args.project
}
if (args.release) {
config.release = args.release
}
}
else {
return undefined
}
return {config, yamlConfig}
}
34 changes: 26 additions & 8 deletions test/commands/publish.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ const signer = provider.getSigner();
const Registry = new ethers.ContractFactory(contracts.registryABI, contracts.registryBytecode, signer);
const License = new ethers.ContractFactory(contracts.licenseABI, contracts.licenseBytecode, signer);

// publishing unzipped folder is not supported only zipped or unzipped files
describe('publish CLI command', () => {
let valist: Client
let members: string[] = []
Expand Down Expand Up @@ -90,11 +91,7 @@ describe('publish CLI command', () => {
const releaseExists = await valist.releaseExists(releaseID);
expect(releaseExists).to.be.true;
const releaseMeta = await valist.getReleaseMeta(releaseID);
const platformKeys = Object.keys(releaseMeta.platforms)
expect(platformKeys.includes('web')).true
expect(platformKeys.includes('darwin_amd64')).true
expect(platformKeys.includes('darwin_arm64')).true
expect(platformKeys.includes('windows_amd64')).true
return releaseMeta
}

it('should create a release with the publish command cli args', async function () {
Expand All @@ -111,14 +108,35 @@ describe('publish CLI command', () => {
'--private-key=4f3edf983ac636a65a842ce7c78d9aa706d3b113bce9c46f30d7d21715b23b1d',
'--no-meta-tx'
]
await runPublishCommandWithMockData(releaseVersion, publishArgs)
const releaseMeta = await runPublishCommandWithMockData(releaseVersion, publishArgs)
const platformKeys = Object.keys(releaseMeta.platforms)
expect(platformKeys.includes('web')).true
expect(platformKeys.includes('darwin_amd64')).true
expect(platformKeys.includes('darwin_arm64')).true
expect(platformKeys.includes('windows_amd64')).true
});

it('should create a release with the publish command and the hyperplay.yml file', async function () {
const publishArgs = [
'--private-key=4f3edf983ac636a65a842ce7c78d9aa706d3b113bce9c46f30d7d21715b23b1d',
'--no-meta-tx'
'--no-meta-tx',
'--yml-path=./test/mock_data/hyperplay.yml'
]
await runPublishCommandWithMockData('v0.0.2', publishArgs)
const releaseMeta = await runPublishCommandWithMockData('v0.0.2', publishArgs)
const platformKeys = Object.keys(releaseMeta.platforms)
expect(platformKeys.includes('web')).true
expect(platformKeys.includes('darwin_amd64')).true
expect(platformKeys.includes('darwin_arm64')).true
expect(platformKeys.includes('windows_amd64')).true
});

it('should create a release with custom keys and some files and folders not zipped', async function(){
const publishArgs = [
'--private-key=4f3edf983ac636a65a842ce7c78d9aa706d3b113bce9c46f30d7d21715b23b1d',
'--no-meta-tx',
'--yml-path=./test/mock_data/hyperplay_publish.yml'
]
const releaseMeta = await runPublishCommandWithMockData('v0.0.3', publishArgs)
console.log('release meta ', releaseMeta)
})
})
1 change: 1 addition & 0 deletions test/mock_data/dmg.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
this is an unzipped text file
19 changes: 19 additions & 0 deletions test/mock_data/hyperplay.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
# This is just used for testing
account: valist
project: cli
release: v0.0.2

platforms:
darwin_amd64:
path: ./mock_data/mac_x64
zip: true
darwin_arm64:
path: ./mock_data/mac_arm64
zip: true
windows_amd64:
path: ./mock_data/windows_amd64
zip: true
web:
path: ./mock_data/web
zip: true

Loading

0 comments on commit b63d4bd

Please sign in to comment.