-openwork
This commit is contained in:
@@ -12,7 +12,7 @@
|
|||||||
|
|
||||||
# opencode needs newer bun from master
|
# opencode needs newer bun from master
|
||||||
opencode = {
|
opencode = {
|
||||||
url = "github:anomalyco/opencode/v1.3.6";
|
url = "github:anomalyco/opencode/v1.3.13";
|
||||||
inputs.nixpkgs.follows = "nixpkgs-master";
|
inputs.nixpkgs.follows = "nixpkgs-master";
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -1,211 +0,0 @@
|
|||||||
{
|
|
||||||
lib,
|
|
||||||
stdenv,
|
|
||||||
fetchFromGitHub,
|
|
||||||
rustPlatform,
|
|
||||||
pkg-config,
|
|
||||||
cargo-tauri,
|
|
||||||
bun,
|
|
||||||
nodejs,
|
|
||||||
pnpm_10,
|
|
||||||
fetchPnpmDeps,
|
|
||||||
pnpmConfigHook,
|
|
||||||
cargo,
|
|
||||||
rustc,
|
|
||||||
wrapGAppsHook3,
|
|
||||||
makeWrapper,
|
|
||||||
dbus,
|
|
||||||
glib,
|
|
||||||
gtk3,
|
|
||||||
libsoup_3,
|
|
||||||
librsvg,
|
|
||||||
libayatana-appindicator,
|
|
||||||
glib-networking,
|
|
||||||
openssl,
|
|
||||||
webkitgtk_4_1,
|
|
||||||
gst_all_1,
|
|
||||||
inputs ? null,
|
|
||||||
}: let
|
|
||||||
# Reuse the anomalyco/opencode binary already pinned in the flake
|
|
||||||
opencode = inputs.opencode.packages.${stdenv.hostPlatform.system}.default;
|
|
||||||
# NOTE: bun build --compile is run WITHOUT --target inside the Nix sandbox.
|
|
||||||
# Specifying a cross-target would cause Bun to download a remote runtime,
|
|
||||||
# which is forbidden in sandbox mode. We build for the host and rename the
|
|
||||||
# output binary to the Rust target triple that Tauri expects.
|
|
||||||
in
|
|
||||||
rustPlatform.buildRustPackage (finalAttrs: {
|
|
||||||
pname = "openwork";
|
|
||||||
version = "0.11.199";
|
|
||||||
|
|
||||||
src = fetchFromGitHub {
|
|
||||||
owner = "different-ai";
|
|
||||||
repo = "openwork";
|
|
||||||
rev = "v${finalAttrs.version}";
|
|
||||||
hash = "sha256-ErON7clClL0iFFbko4cAKgZROt/bJ8k9FTX8eivrDnA=";
|
|
||||||
};
|
|
||||||
|
|
||||||
# Rust crate lives inside the monorepo under apps/desktop/src-tauri
|
|
||||||
cargoRoot = "apps/desktop/src-tauri";
|
|
||||||
cargoLock.lockFile = finalAttrs.src + "/apps/desktop/src-tauri/Cargo.lock";
|
|
||||||
buildAndTestSubdir = finalAttrs.cargoRoot;
|
|
||||||
|
|
||||||
# Prefetch the entire pnpm workspace store as a fixed-output derivation.
|
|
||||||
# After the first successful build, replace lib.fakeHash with the real hash
|
|
||||||
# from the error output.
|
|
||||||
pnpmDeps = fetchPnpmDeps {
|
|
||||||
inherit (finalAttrs) pname version src;
|
|
||||||
pnpm = pnpm_10;
|
|
||||||
fetcherVersion = 2;
|
|
||||||
hash = "sha256-+fN9h9htN5nGsh4wG4DqCilykmnJn0DDoFtrlF/ajRU=";
|
|
||||||
};
|
|
||||||
|
|
||||||
nativeBuildInputs =
|
|
||||||
[
|
|
||||||
pkg-config
|
|
||||||
cargo-tauri.hook
|
|
||||||
bun
|
|
||||||
nodejs # needed for patchShebangs
|
|
||||||
pnpm_10
|
|
||||||
pnpmConfigHook # installs workspace node_modules offline from pnpmDeps
|
|
||||||
cargo
|
|
||||||
rustc
|
|
||||||
makeWrapper
|
|
||||||
]
|
|
||||||
++ lib.optionals stdenv.hostPlatform.isLinux [wrapGAppsHook3];
|
|
||||||
|
|
||||||
buildInputs = lib.optionals stdenv.hostPlatform.isLinux [
|
|
||||||
dbus
|
|
||||||
glib
|
|
||||||
gtk3
|
|
||||||
libsoup_3
|
|
||||||
librsvg
|
|
||||||
libayatana-appindicator
|
|
||||||
glib-networking
|
|
||||||
openssl
|
|
||||||
webkitgtk_4_1
|
|
||||||
gst_all_1.gstreamer
|
|
||||||
gst_all_1.gst-plugins-base
|
|
||||||
gst_all_1.gst-plugins-good
|
|
||||||
gst_all_1.gst-plugins-bad
|
|
||||||
];
|
|
||||||
|
|
||||||
strictDeps = true;
|
|
||||||
|
|
||||||
# These two tests require docker or rely on OS-level process/timeout
|
|
||||||
# behaviour that differs inside the Nix sandbox:
|
|
||||||
# - docker_command_falls_back_after_timeout: expects docker (exit 127 = not found)
|
|
||||||
# - local_command_timeout_returns_when_descendant_keeps_pipe_open: sandbox kills
|
|
||||||
# descendant processes differently, so the expected error never arrives
|
|
||||||
checkFlags = [
|
|
||||||
"--skip=commands::orchestrator::tests::docker_command_falls_back_after_timeout"
|
|
||||||
"--skip=commands::orchestrator::tests::local_command_timeout_returns_when_descendant_keeps_pipe_open"
|
|
||||||
];
|
|
||||||
|
|
||||||
# cargo-tauri.hook: run `cargo tauri build` from this subdirectory
|
|
||||||
tauriRoot = "apps/desktop";
|
|
||||||
|
|
||||||
# Override tauri config at build time:
|
|
||||||
# - clear beforeBuildCommand (we run frontend + sidecars manually in preBuild)
|
|
||||||
# - disable updater artifact generation (no auto-update in Nix packages)
|
|
||||||
tauriBuildFlags = [
|
|
||||||
"--config"
|
|
||||||
(builtins.toJSON {
|
|
||||||
build.beforeBuildCommand = "";
|
|
||||||
bundle.createUpdaterArtifacts = false;
|
|
||||||
})
|
|
||||||
"--no-sign"
|
|
||||||
];
|
|
||||||
|
|
||||||
preBuild = let
|
|
||||||
target = stdenv.hostPlatform.rust.rustcTarget;
|
|
||||||
in ''
|
|
||||||
sidecarDir="$(pwd)/apps/desktop/src-tauri/sidecars"
|
|
||||||
mkdir -p "$sidecarDir"
|
|
||||||
|
|
||||||
# ── 1. openwork-server ────────────────────────────────────────────────
|
|
||||||
# Bun compiles TypeScript → self-contained native binary (no Bun runtime needed at runtime)
|
|
||||||
(
|
|
||||||
cd apps/server
|
|
||||||
bun ./script/build.ts --outdir "$sidecarDir" --filename openwork-server
|
|
||||||
)
|
|
||||||
# script/build.ts names the output after the bun target when --target is given;
|
|
||||||
# without --target it outputs just the filename. Rename to Tauri's expected triple format.
|
|
||||||
mv "$sidecarDir/openwork-server" "$sidecarDir/openwork-server-${target}"
|
|
||||||
|
|
||||||
# ── 2. opencode-router ────────────────────────────────────────────────
|
|
||||||
(
|
|
||||||
cd apps/opencode-router
|
|
||||||
bun ./script/build.ts --outdir "$sidecarDir" --filename opencode-router
|
|
||||||
)
|
|
||||||
mv "$sidecarDir/opencode-router" "$sidecarDir/opencode-router-${target}"
|
|
||||||
|
|
||||||
# ── 3. openwork-orchestrator ──────────────────────────────────────────
|
|
||||||
(
|
|
||||||
cd apps/orchestrator
|
|
||||||
bun ./script/build.ts --outdir "$sidecarDir" --filename openwork-orchestrator
|
|
||||||
)
|
|
||||||
mv "$sidecarDir/openwork-orchestrator" "$sidecarDir/openwork-orchestrator-${target}"
|
|
||||||
|
|
||||||
# ── 4. chrome-devtools-mcp shim ───────────────────────────────────────
|
|
||||||
# This is a tiny TypeScript shim that proxies to `npm exec chrome-devtools-mcp`.
|
|
||||||
# We compile it the same way as the other sidecars.
|
|
||||||
bun build --compile apps/desktop/scripts/chrome-devtools-mcp-shim.ts \
|
|
||||||
--outfile "$sidecarDir/chrome-devtools-mcp-${target}"
|
|
||||||
|
|
||||||
# ── 5. opencode binary (from Nix store) ───────────────────────────────
|
|
||||||
cp ${opencode}/bin/opencode "$sidecarDir/opencode-${target}"
|
|
||||||
chmod +x "$sidecarDir/opencode-${target}"
|
|
||||||
|
|
||||||
# ── 6. versions.json (required by Tauri; sha256 computed at build time) ─
|
|
||||||
oc_sha=$(sha256sum "$sidecarDir/opencode-${target}" | awk '{print $1}')
|
|
||||||
sv_sha=$(sha256sum "$sidecarDir/openwork-server-${target}" | awk '{print $1}')
|
|
||||||
rt_sha=$(sha256sum "$sidecarDir/opencode-router-${target}" | awk '{print $1}')
|
|
||||||
or_sha=$(sha256sum "$sidecarDir/openwork-orchestrator-${target}" | awk '{print $1}')
|
|
||||||
cd_sha=$(sha256sum "$sidecarDir/chrome-devtools-mcp-${target}" | awk '{print $1}')
|
|
||||||
|
|
||||||
cat > "$sidecarDir/versions.json" <<VERSIONS_EOF
|
|
||||||
{
|
|
||||||
"opencode": { "version": "${opencode.version}", "sha256": "$oc_sha" },
|
|
||||||
"openwork-server": { "version": "${finalAttrs.version}", "sha256": "$sv_sha" },
|
|
||||||
"opencodeRouter": { "version": "${finalAttrs.version}", "sha256": "$rt_sha" },
|
|
||||||
"openwork-orchestrator": { "version": "${finalAttrs.version}", "sha256": "$or_sha" },
|
|
||||||
"chrome-devtools-mcp": { "version": "0.17.0", "sha256": "$cd_sha" }
|
|
||||||
}
|
|
||||||
VERSIONS_EOF
|
|
||||||
cp "$sidecarDir/versions.json" "$sidecarDir/versions.json-${target}"
|
|
||||||
|
|
||||||
# ── 7. SolidJS frontend ───────────────────────────────────────────────
|
|
||||||
# Builds apps/app/ → apps/app/dist/ which tauri.conf.json references as frontendDist
|
|
||||||
pnpm --filter @openwork/app build
|
|
||||||
'';
|
|
||||||
|
|
||||||
# Remove internal Tauri sidecar binaries BEFORE fixupPhase so that
|
|
||||||
# wrapGAppsHook3 never sees them and never creates .opencode-wrapped
|
|
||||||
# (or similar) files that would conflict with other packages in buildEnv.
|
|
||||||
preFixup = lib.optionalString stdenv.hostPlatform.isLinux ''
|
|
||||||
for sidecar in opencode opencode-router openwork-server openwork-orchestrator chrome-devtools-mcp versions.json; do
|
|
||||||
rm -f "$out/bin/$sidecar"
|
|
||||||
done
|
|
||||||
'';
|
|
||||||
|
|
||||||
postFixup = lib.optionalString stdenv.hostPlatform.isLinux ''
|
|
||||||
for name in OpenWork "OpenWork-Dev"; do
|
|
||||||
if [ -f "$out/bin/$name" ]; then
|
|
||||||
mv "$out/bin/$name" "$out/bin/openwork"
|
|
||||||
break
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
find "$out/share/applications" -name "*.desktop" \
|
|
||||||
-exec sed -i 's|^Exec=OpenWork.*|Exec=openwork|' {} +
|
|
||||||
wrapProgram $out/bin/openwork \
|
|
||||||
--set WEBKIT_DISABLE_COMPOSITING_MODE 1
|
|
||||||
'';
|
|
||||||
|
|
||||||
meta = {
|
|
||||||
description = "Open-source alternative to Claude Cowork, built for teams";
|
|
||||||
homepage = "https://github.com/different-ai/openwork";
|
|
||||||
license = lib.licenses.mit;
|
|
||||||
mainProgram = "openwork";
|
|
||||||
platforms = lib.platforms.linux;
|
|
||||||
};
|
|
||||||
})
|
|
||||||
Reference in New Issue
Block a user