Skip to content

Commit

Permalink
Gzip artifacts
Browse files Browse the repository at this point in the history
Co-authored-by: bjorn3 <bjorn3@users.noreply.github.com>

Override miniz_oxide to build it with optimizations

Building this crate with optimizations decreases the gzipping
part of `cargo xtask dist` from `30-40s` down to `3s`,
the overhead for `rustc` to apply optimizations is miserable on this background
  • Loading branch information
Veetaha committed Jul 1, 2020
1 parent 8943c2c commit 5697716
Show file tree
Hide file tree
Showing 6 changed files with 84 additions and 25 deletions.
39 changes: 38 additions & 1 deletion Cargo.lock

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

5 changes: 5 additions & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,11 @@ opt-level = 0
[profile.release.package.xtask]
opt-level = 0

# Gzipping the artifacts is up to 10 times faster with optimizations (`cargo xtask dist`).
# `miniz_oxide` is the direct dependency of `flate2` which does all the heavy lifting
[profile.dev.package.miniz_oxide]
opt-level = 3

[patch.'crates-io']
# rowan = { path = "../rowan" }

Expand Down
17 changes: 9 additions & 8 deletions editors/code/src/main.ts
Original file line number Diff line number Diff line change
Expand Up @@ -261,13 +261,13 @@ async function getServer(config: Config, state: PersistentState): Promise<string
};
if (config.package.releaseTag === null) return "rust-analyzer";

let binaryName: string | undefined = undefined;
let platform: string | undefined;
if (process.arch === "x64" || process.arch === "ia32") {
if (process.platform === "linux") binaryName = "rust-analyzer-linux";
if (process.platform === "darwin") binaryName = "rust-analyzer-mac";
if (process.platform === "win32") binaryName = "rust-analyzer-windows.exe";
if (process.platform === "linux") platform = "linux";
if (process.platform === "darwin") platform = "mac";
if (process.platform === "win32") platform = "windows";
}
if (binaryName === undefined) {
if (platform === undefined) {
vscode.window.showErrorMessage(
"Unfortunately we don't ship binaries for your platform yet. " +
"You need to manually clone rust-analyzer repository and " +
Expand All @@ -278,8 +278,8 @@ async function getServer(config: Config, state: PersistentState): Promise<string
);
return undefined;
}

const dest = path.join(config.globalStoragePath, binaryName);
const ext = platform === "windows" ? ".exe" : "";
const dest = path.join(config.globalStoragePath, `rust-analyzer-${platform}${ext}`);
const exists = await fs.stat(dest).then(() => true, () => false);
if (!exists) {
await state.updateServerVersion(undefined);
Expand All @@ -296,7 +296,7 @@ async function getServer(config: Config, state: PersistentState): Promise<string
}

const release = await fetchRelease(config.package.releaseTag);
const artifact = release.assets.find(artifact => artifact.name === binaryName);
const artifact = release.assets.find(artifact => artifact.name === `rust-analyzer-${platform}.gz`);
assert(!!artifact, `Bad release: ${JSON.stringify(release)}`);

// Unlinking the exe file before moving new one on its place should prevent ETXTBSY error.
Expand All @@ -308,6 +308,7 @@ async function getServer(config: Config, state: PersistentState): Promise<string
url: artifact.browser_download_url,
dest,
progressTitle: "Downloading rust-analyzer server",
gunzip: true,
mode: 0o755
});

Expand Down
16 changes: 8 additions & 8 deletions editors/code/src/net.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ import * as vscode from "vscode";
import * as stream from "stream";
import * as crypto from "crypto";
import * as fs from "fs";
import * as zlib from "zlib";
import * as util from "util";
import * as path from "path";
import { log, assert } from "./util";
Expand Down Expand Up @@ -65,6 +66,7 @@ interface DownloadOpts {
url: string;
dest: string;
mode?: number;
gunzip?: boolean;
}

export async function download(opts: DownloadOpts) {
Expand All @@ -82,7 +84,7 @@ export async function download(opts: DownloadOpts) {
},
async (progress, _cancellationToken) => {
let lastPercentage = 0;
await downloadFile(opts.url, tempFile, opts.mode, (readBytes, totalBytes) => {
await downloadFile(opts.url, tempFile, opts.mode, !!opts.gunzip, (readBytes, totalBytes) => {
const newPercentage = (readBytes / totalBytes) * 100;
progress.report({
message: newPercentage.toFixed(0) + "%",
Expand All @@ -97,16 +99,11 @@ export async function download(opts: DownloadOpts) {
await fs.promises.rename(tempFile, opts.dest);
}

/**
* Downloads file from `url` and stores it at `destFilePath` with `mode` (unix permissions).
* `onProgress` callback is called on recieveing each chunk of bytes
* to track the progress of downloading, it gets the already read and total
* amount of bytes to read as its parameters.
*/
async function downloadFile(
url: string,
destFilePath: fs.PathLike,
mode: number | undefined,
gunzip: boolean,
onProgress: (readBytes: number, totalBytes: number) => void
): Promise<void> {
const res = await fetch(url);
Expand All @@ -130,7 +127,10 @@ async function downloadFile(
});

const destFileStream = fs.createWriteStream(destFilePath, { mode });
await pipeline(res.body, destFileStream);
const srcStream = gunzip ? res.body.pipe(zlib.createGunzip()) : res.body;

await pipeline(srcStream, destFileStream);

await new Promise<void>(resolve => {
destFileStream.on("close", resolve);
destFileStream.destroy();
Expand Down
1 change: 1 addition & 0 deletions xtask/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -14,3 +14,4 @@ pico-args = "0.3.1"
quote = "1.0.2"
proc-macro2 = "1.0.8"
anyhow = "1.0.26"
flate2 = "1.0"
31 changes: 23 additions & 8 deletions xtask/src/dist.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,10 @@
use std::path::PathBuf;
use flate2::{write::GzEncoder, Compression};
use std::{
env,
fs::File,
io,
path::{Path, PathBuf},
};

use anyhow::Result;

Expand All @@ -16,7 +22,7 @@ pub fn run_dist(nightly: bool, client_version: Option<String>) -> Result<()> {
let release_tag = if nightly { "nightly".to_string() } else { date_iso()? };
dist_client(&version, &release_tag)?;
}
dist_server(nightly)?;
dist_server()?;
Ok(())
}

Expand Down Expand Up @@ -46,17 +52,14 @@ fn dist_client(version: &str, release_tag: &str) -> Result<()> {
Ok(())
}

fn dist_server(nightly: bool) -> Result<()> {
fn dist_server() -> Result<()> {
if cfg!(target_os = "linux") {
std::env::set_var("CC", "clang");
env::set_var("CC", "clang");
run!(
"cargo build --manifest-path ./crates/rust-analyzer/Cargo.toml --bin rust-analyzer --release"
// We'd want to add, but that requires setting the right linker somehow
// --features=jemalloc
)?;
if !nightly {
run!("strip ./target/release/rust-analyzer")?;
}
} else {
run!("cargo build --manifest-path ./crates/rust-analyzer/Cargo.toml --bin rust-analyzer --release")?;
}
Expand All @@ -71,8 +74,20 @@ fn dist_server(nightly: bool) -> Result<()> {
panic!("Unsupported OS")
};

fs2::copy(src, dst)?;
let src = PathBuf::from(src);
let dst = PathBuf::from(dst);

fs2::copy(&src, &dst)?;
gzip(&src, &dst.with_extension("gz"))?;

Ok(())
}

fn gzip(src_path: &Path, dest_path: &Path) -> Result<()> {
let mut encoder = GzEncoder::new(File::create(dest_path)?, Compression::best());
let mut input = io::BufReader::new(File::open(src_path)?);
io::copy(&mut input, &mut encoder)?;
encoder.finish()?;
Ok(())
}

Expand Down

0 comments on commit 5697716

Please sign in to comment.