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

Changes for Features V2 #6

Merged
merged 33 commits into from
Jun 28, 2022
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
33 commits
Select commit Hold shift + click to select a range
eebacc5
First changes for features
edgonmsft May 9, 2022
e825692
Missing reference.
edgonmsft May 9, 2022
1047e01
Fix test
edgonmsft May 9, 2022
40aa793
Fixes.
edgonmsft May 9, 2022
83afa1b
Run config
edgonmsft May 9, 2022
39cff54
Comment out test
edgonmsft May 9, 2022
12d39a2
Fixes for absolute paths
edgonmsft May 10, 2022
e769e66
Comments on pr.
edgonmsft May 10, 2022
b35174c
Fix for github features.
edgonmsft May 12, 2022
6b59032
Merge remote-tracking branch 'origin/main' into edgon/features-1
edgonmsft May 23, 2022
1e9a5bb
Fixes for locally cacched features.
edgonmsft May 23, 2022
9b98c33
Merge remote-tracking branch 'origin' into edgon/features-1
edgonmsft May 24, 2022
5923ef7
Fix old features.
edgonmsft May 24, 2022
23e87ff
Other fixes.
edgonmsft May 24, 2022
9bce15f
Some comments and formating.
edgonmsft May 24, 2022
91edf38
Test fix
edgonmsft May 24, 2022
971bd56
Fix
edgonmsft May 24, 2022
4e0fc7e
Fix test.
edgonmsft May 24, 2022
4842ec5
Refactor changes and read devcontainer collections.
edgonmsft May 25, 2022
ebbbc4e
Fix test.
edgonmsft May 25, 2022
26f3f35
Fix test.
edgonmsft May 25, 2022
11fb36c
Fix test 3
edgonmsft May 25, 2022
934ae4a
fix test 4
edgonmsft May 25, 2022
63a5ed2
fix test 5
edgonmsft May 25, 2022
b304f90
Comments from PR.
edgonmsft May 25, 2022
5e29707
Fix cleanup
edgonmsft May 25, 2022
ec88721
Remove prefix for environment variables in features v2
edgonmsft Jun 2, 2022
267ef67
support fetching v2 features from github repo (#48)
joshspicer Jun 3, 2022
df7575b
clean up tracing and formatting
joshspicer Jun 8, 2022
1b42a63
Merge remote-tracking branch 'origin/main' into edgon/features-1
edgonmsft Jun 14, 2022
68e68b1
Changes from the last spec proposal changes.
edgonmsft Jun 24, 2022
39a1b22
Comments and fixes from PR.
edgonmsft Jun 27, 2022
e92dd24
Unit test fixes.
edgonmsft Jun 27, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 7 additions & 1 deletion .vscode/launch.json
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,13 @@
"name": "Launch cli - up",
"program": "${workspaceFolder}/src/spec-node/devContainersSpecCLI.ts",
"cwd": "${workspaceFolder}",
"args": ["up", "--workspace-folder", "../features-playground", "--log-level", "debug", ],
"args": [
"up",
"--workspace-folder",
"../devcontainers-features",
"--log-level",
"debug",
],
"console": "integratedTerminal",
}
]
Expand Down
7 changes: 1 addition & 6 deletions src/spec-common/injectHeadless.ts
Original file line number Diff line number Diff line change
Expand Up @@ -105,18 +105,13 @@ export type DevContainerConfigCommand = 'initializeCommand' | 'onCreateCommand'

const defaultWaitFor: DevContainerConfigCommand = 'updateContentCommand';

export interface DevContainerFeature {
id: string;
options: boolean | string | Record<string, boolean | string | undefined>;
}

export interface CommonDevContainerConfig {
configFilePath?: URI;
remoteEnv?: Record<string, string | null>;
forwardPorts?: (number | string)[];
portsAttributes?: Record<string, PortAttributes>;
otherPortsAttributes?: PortAttributes;
features?: DevContainerFeature[] | Record<string, string | boolean | Record<string, string | boolean>>;
features?: Record<string, string | boolean | Record<string, string | boolean>>;
onCreateCommand?: string | string[];
updateContentCommand?: string | string[];
postCreateCommand?: string | string[];
Expand Down
9 changes: 6 additions & 3 deletions src/spec-configuration/configuration.ts
Original file line number Diff line number Diff line change
Expand Up @@ -59,7 +59,8 @@ export interface DevContainerFromImageConfig {
remoteUser?: string;
updateRemoteUserUID?: boolean;
userEnvProbe?: UserEnvProbe;
features?: DevContainerFeature[] | Record<string, string | boolean | Record<string, string | boolean>>;
features?: Record<string, string | boolean | Record<string, string | boolean>>;
overrideFeatureInstallOrder?: string[];
hostRequirements?: HostRequirements;
}

Expand Down Expand Up @@ -90,7 +91,8 @@ export type DevContainerFromDockerfileConfig = {
remoteUser?: string;
updateRemoteUserUID?: boolean;
userEnvProbe?: UserEnvProbe;
features?: DevContainerFeature[] | Record<string, string | boolean | Record<string, string | boolean>>;
features?: Record<string, string | boolean | Record<string, string | boolean>>;
overrideFeatureInstallOrder?: string[];
hostRequirements?: HostRequirements;
} & (
{
Expand Down Expand Up @@ -137,7 +139,8 @@ export interface DevContainerFromDockerComposeConfig {
remoteUser?: string;
updateRemoteUserUID?: boolean;
userEnvProbe?: UserEnvProbe;
features?: DevContainerFeature[] | Record<string, string | boolean | Record<string, string | boolean>>;
features?: Record<string, string | boolean | Record<string, string | boolean>>;
overrideFeatureInstallOrder?: string[];
hostRequirements?: HostRequirements;
}

Expand Down
15 changes: 14 additions & 1 deletion src/spec-configuration/containerFeaturesConfiguration.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,11 +7,13 @@ import * as jsonc from 'jsonc-parser';
import * as path from 'path';
import * as URL from 'url';
import * as tar from 'tar';
import { existsSync } from 'fs';
import { DevContainerConfig, DevContainerFeature } from './configuration';
import { mkdirpLocal, readLocalFile, rmLocal, writeLocalFile, cpDirectoryLocal } from '../spec-utils/pfs';
import { Log, LogLevel } from '../spec-utils/log';
import { request } from '../spec-utils/httpRequest';
import { existsSync } from 'fs';
import { computeFeatureInstallationOrder } from './containerFeaturesOrder';


const V1_ASSET_NAME = 'devcontainer-features.tgz';

Expand All @@ -38,6 +40,7 @@ export interface Feature {
capAdd?: string[];
securityOpt?: string[];
entrypoint?: string;
installAfter?: string[];
include?: string[];
exclude?: string[];
value: boolean | string | Record<string, boolean | string | undefined>; // set programmatically
Expand Down Expand Up @@ -414,6 +417,15 @@ export async function generateFeaturesConfig(params: { extensionPath: string; cw
output.write('--- Fetching User Features ----', LogLevel.Trace);
await fetchFeatures(params, featuresConfig, locallyCachedFeatureSet, dstFolder);

const ordererdFeatures = computeFeatureInstallationOrder(config, featuresConfig.featureSets);

output.write('--- Computed order ----', LogLevel.Trace);
for (const feature of ordererdFeatures) {
output.write(`${feature.features[0].id}`, LogLevel.Trace);
}

featuresConfig.featureSets = ordererdFeatures;

return featuresConfig;
}

Expand Down Expand Up @@ -826,6 +838,7 @@ async function parseDevContainerFeature(featureSet: FeatureSet, feature: Feature
featureSet.internalVersion = '2';
feature.buildArg = featureJson.buildArg;
feature.options = featureJson.options;
feature.installAfter = featureJson.installAfter;
} else {
featureSet.internalVersion = '1';
}
Expand Down
108 changes: 108 additions & 0 deletions src/spec-configuration/containerFeaturesOrder.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,108 @@
/*---------------------------------------------------------------------------------------------
* Copyright (c) Microsoft Corporation. All rights reserved.
* Licensed under the MIT License. See License.txt in the project root for license information.
*--------------------------------------------------------------------------------------------*/


import { ContainerError } from '../spec-common/errors';
import { FeatureSet } from '../spec-configuration/containerFeaturesConfiguration';
import { DevContainerConfig } from './configuration';

interface FeatureNode {
feature: FeatureSet;
before: Set<FeatureNode>;
after: Set<FeatureNode>;
}

export function computeFeatureInstallationOrder(config: DevContainerConfig, features: FeatureSet[]) {

if (!config.overrideFeatureInstallOrder) {
edgonmsft marked this conversation as resolved.
Show resolved Hide resolved
return computeInstallationOrder(features);
}
else {
return computeOverrideInstallationOrder(config, features);
}
}

function computeOverrideInstallationOrder(config: DevContainerConfig, features: FeatureSet[]) {
const automaticOrder = computeInstallationOrder(features);

const orderedFeatures = [];

for (const featureId of config.overrideFeatureInstallOrder!) {
const feature = automaticOrder.find(feature => feature.features[0].name === featureId);
if (!feature) {
throw new ContainerError({ description: `Feature ${featureId} not found` });
}
orderedFeatures.push(feature);
features.splice(features.indexOf(feature), 1);
}

return orderedFeatures.concat(features);
}

function computeInstallationOrder(features: FeatureSet[]) {
joshspicer marked this conversation as resolved.
Show resolved Hide resolved
const nodesById = features.map<FeatureNode>(feature => ({
feature,
before: new Set(),
after: new Set(),
})).reduce((map, feature) => map.set(feature.feature.features[0].id, feature), new Map<string, FeatureNode>());

const nodes = [...nodesById.values()];
for (const later of nodes) {
for (const firstId of later.feature.features[0].installAfter || []) {
const first = nodesById.get(firstId);
// soft dependencies
if (first) {
later.after.add(first);
first.before.add(later);
}
}
}

const { roots, islands } = nodes.reduce((prev, node) => {
if (node.after.size === 0) {
if (node.before.size === 0) {
prev.islands.push(node);
} else {
prev.roots.push(node);
}
}
return prev;
}, { roots: [] as FeatureNode[], islands: [] as FeatureNode[] });

const orderedFeatures = [];
let current = roots;
while (current.length) {
const next = [];
for (const first of current) {
for (const later of first.before) {
later.after.delete(first);
if (later.after.size === 0) {
next.push(later);
}
}
}
orderedFeatures.push(
...current.map(node => node.feature)
.sort((a, b) => a.features[0].id.localeCompare(b.features[0].id)) // stable order
);
current = next;
}

orderedFeatures.push(
...islands.map(node => node.feature)
.sort((a, b) => a.features[0].id.localeCompare(b.features[0].id)) // stable order
);

const missing = new Set(nodesById.keys());
for (const feature of orderedFeatures) {
missing.delete(feature.features[0].id);
}

if (missing.size !== 0) {
throw new ContainerError({ description: `Features declare cyclic dependency: ${[...missing].join(', ')}` });
}

return orderedFeatures;
}
3 changes: 3 additions & 0 deletions src/spec-configuration/tsconfig.json
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,9 @@
"references": [
{
"path": "../spec-utils"
},
{
"path": "../spec-common"
}
]
}