1
0
mirror of https://github.com/lensapp/lens.git synced 2025-05-20 05:10:56 +00:00

Unify Age column output with kubectl (#1822)

Signed-off-by: Lauri Nevala <lauri.nevala@gmail.com>
This commit is contained in:
Lauri Nevala 2020-12-28 09:32:20 +02:00 committed by GitHub
parent ffa48b32d1
commit 2643f9939b
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 142 additions and 58 deletions

View File

@ -1,57 +1,89 @@
import moment from "moment";
import { formatDuration } from "../formatDuration"; import { formatDuration } from "../formatDuration";
const second = 1000; const second = 1000;
const minute = 60 * second; const minute = 60 * second;
const hour = 60 * minute; const hour = 60 * minute;
const day = 24 * hour; const day = 24 * hour;
const week = 7 * day; const year = 365 * day;
describe("human format durations", () => { describe("human format durations", () => {
test("long formatted durations less than 24 hours long shouldn't have a 'd' component", () => { test("small duration should output something", () => {
const res = formatDuration(19 * 60 * 60 * 1000, false); expect(formatDuration(1)).toBe("0s");
expect(formatDuration(3)).toBe("0s");
expect(res).not.toContain("d");
expect(res).toBe("19h");
}); });
test("long formatted durations more than a week have correct day count", () => { test("returns seconds for duration under 1 min", () => {
const res = formatDuration(2 * week + 2 * day, false); const res = formatDuration(8 * second);
expect(res).toBe("2w 2d"); expect(res).toBe("8s");
});
test("durations > 1/2 week shouldn't show 1w has passed", () => {
const res = formatDuration(5 * 24 * 60 * 60 * 1000, false);
expect(res).not.toContain("w");
expect(res).toBe("5d");
});
test("durations shouldn't include zero magnitude parts", () => {
const res = formatDuration(6 * day + 2 * minute, false);
expect(res).not.toContain("h");
expect(res).toBe("6d 2m");
});
test("seconds are ignored unless they are significant (< 1m)", () => {
const insignificant = formatDuration(1 * hour + 2 * minute + 31 * second, false);
expect(insignificant).not.toContain("s");
expect(insignificant).toBe("1h 2m");
const significant = formatDuration(31 * second, false);
expect(significant).toBe("31s");
}); });
test("zero duration should output something", () => { test("zero duration should output something", () => {
expect(formatDuration(0, false)).toBe("0s"); expect(formatDuration(0)).toBe("0s");
expect(formatDuration(0, true)).toBe("0s");
}); });
test("small duration should output something", () => { describe("when compact is true", () => {
expect(formatDuration(1, false)).toBe("0s");
expect(formatDuration(3, true)).toBe("0s"); test("duration under 3 hours return minutes", () => {
const res = formatDuration(1 * hour + 35 * minute);
expect(res).toBe("95m");
});
test("duration under 8 hours return hours and minutes", () => {
const res = formatDuration(6 * hour + 15 * minute + 20 * second);
expect(res).toBe("6h15m");
});
test("duration under 48 hours return hours", () => {
const res = formatDuration(1 * day + 4 * hour + 15 * minute);
expect(res).toBe("28h");
});
test("duration under 2 years return days", () => {
const res = formatDuration(400 * day + 4 * hour + 15 * minute);
expect(res).toBe("400d");
});
test("durations less than 8 years returns years and days", () => {
const timeValue = new Date().getTime() - new Date(moment().subtract(2, "years").subtract(5, "days").subtract(2, "hours").toDate()).getTime();
const res = formatDuration(timeValue);
expect(res).toBe("2y5d");
});
test("durations more than 8 years returns years", () => {
const timeValue = new Date().getTime() - new Date(moment().subtract(9, "years").subtract(5, "days").toDate()).getTime();
const res = formatDuration(timeValue);
expect(res).toBe("9y");
});
test("durations more than 8 years returns years", () => {
const res = formatDuration(10 * year + 25 * day);
expect(res).toBe("10y");
});
test("durations shouldn't include zero magnitude parts", () => {
const zeroSeconds = formatDuration(8 * minute);
expect(zeroSeconds).toBe("8m");
const zeroMinutes = formatDuration(8 * hour + 15 * minute);
expect(zeroMinutes).toBe("8h");
const zeroHours = formatDuration(6 * day + 2 * minute);
expect(zeroHours).toBe("6d");
}); });
}); });
});

View File

@ -1,34 +1,86 @@
import moment from "moment"; import moment from "moment";
const suffixes = ["w", "d", "h", "m", "s"];
/** /**
* This function formats durations in a more human readable form. * This function formats durations in a more human readable form.
* @param timeValue the duration in milliseconds to format * @param timeValue the duration in milliseconds to format
* @param compact when true, only the largest non-zero time frame will be returned
*/ */
export function formatDuration(timeValue: number, compact: boolean) { export function formatDuration(timeValue: number, compact = true) {
const duration = moment.duration(timeValue, "milliseconds"); const duration = moment.duration(timeValue, "milliseconds");
const durationValues = [ const seconds = Math.floor(duration.asSeconds());
Math.floor(duration.asWeeks()), const separator = compact ? "": " ";
Math.floor(duration.asDays()) % 7,
duration.hours(),
duration.minutes(),
duration.seconds(),
];
const meaningfulValues = durationValues
.map((a, i): [number, string] => [a, suffixes[i]])
.filter(([dur]) => dur > 0)
.filter(([, suf], i) => i === 0 || suf !== "s") // remove seconds, unless it is the only one
.map(([dur, suf]) => dur + suf);
if (meaningfulValues.length === 0) { if (seconds < 0) {
return "0s"; return "0s";
} else if (seconds < 60*2 ) {
return `${seconds}s`;
}
const minutes = Math.floor(duration.asMinutes());
if (minutes < 10) {
const seconds = duration.seconds();
return getMeaningfulValues([minutes, seconds], ["m", "s"], separator);
} else if (minutes < 60 * 3) {
if (!compact) {
return getMeaningfulValues([minutes, duration.seconds()], ["m", "s"]);
}
return `${minutes}m`;
}
const hours = Math.floor(duration.asHours());
if(hours < 8) {
const minutes = duration.minutes();
return getMeaningfulValues([hours, minutes], ["h", "m"], separator);
} else if (hours < 48) {
if (compact) {
return `${hours}h`;
} else {
return getMeaningfulValues([hours, duration.minutes()], ["h", "m"]);
}
}
const days = Math.floor(duration.asDays());
if (days < 8) {
const hours = duration.hours();
if (compact) {
return getMeaningfulValues([days, hours], ["d", "h"], separator);
} else {
return getMeaningfulValues([days, hours, duration.minutes()], ["d", "h", "m"]);
}
}
const years = Math.floor(duration.asYears());
if (years < 2) {
if (compact) {
return `${days}d`;
} else {
return getMeaningfulValues([days, duration.hours(), duration.minutes()], ["d", "h", "m"]);
}
} else if (years < 8) {
const days = duration.days();
if (compact) {
return getMeaningfulValues([years, days], ["y", "d"], separator);
}
} }
if (compact) { if (compact) {
return meaningfulValues[0]; return `${years}y`;
} }
return meaningfulValues.join(" "); return getMeaningfulValues([years, duration.days(), duration.hours(), duration.minutes()], ["y", "d", "h", "m"]);
}
function getMeaningfulValues(values: number[], suffixes: string[], separator = " ") {
return values
.map((a, i): [number, string] => [a, suffixes[i]])
.filter(([dur]) => dur > 0)
.map(([dur, suf]) => dur + suf)
.join(separator);
} }