2
0

23 Commits

Author SHA1 Message Date
Sergio Padrino
a6223e756e Create codeql-analysis.yml 2022-02-08 10:33:36 +01:00
Sergio Padrino
ddccbacf81 Merge pull request #19 from desktop/bump-dependencies
Bump dependencies
2022-02-08 10:32:35 +01:00
Sergio Padrino
824c6f1364 Use latest macOS runner 2022-02-08 10:27:18 +01:00
Sergio Padrino
2386f9980f Bump dependencies 2022-02-08 10:27:07 +01:00
Sergio Padrino
ee3e96e70a Merge pull request #14 from desktop/releases/0.9.8
Release v0.9.8
2021-08-18 15:35:28 +02:00
Sergio Padrino
cbd3dbb31d Fix prebuilds to include ssh-wrapper
Aside from changing `--include-regex`, I also needed to fix `prebuild` itself to support stripping debug symbols of multiple files: https://github.com/prebuild/prebuild/pull/280

Instead of waiting until that PR is merged, I went ahead and switched our `prebuild` dependency with my fork, containing my changes.
2021-08-17 12:00:02 +02:00
Sergio Padrino
a2cfb8a8f3 Bump version to v0.9.8 2021-08-16 16:18:10 +02:00
Sergio Padrino
8b38cd56a1 Merge pull request #13 from desktop/ssh-wrapper
Add SSH wrapper for macOS
2021-08-16 15:30:06 +02:00
Sergio Padrino
6d672bd70f Skip ssh-wrapper tests on Linux too 2021-08-16 12:23:32 +02:00
Sergio Padrino
c16ca4bad7 Make sure we always build in CI 2021-08-16 11:54:08 +02:00
Sergio Padrino
158101ca4d Fix index.js style 2021-08-16 11:50:53 +02:00
Sergio Padrino
ac5ec3b3d2 Update README.md 2021-08-16 11:37:29 +02:00
Sergio Padrino
6bab396b51 Fix build & test on Windows 2021-08-16 11:32:50 +02:00
Sergio Padrino
0165c9bc33 Add ssh-wrapper with some tests 2021-08-16 11:03:10 +02:00
Sergio Padrino
c39c55721f Merge pull request #11 from desktop/releases/0.9.6 2021-04-29 08:45:33 +02:00
Sergio Padrino
38c590851f Bump version to 0.9.7 2021-04-29 08:39:30 +02:00
Sergio Padrino
ab9cabd08f Bump node version to fix arm64 prebuilds 😒 2021-04-29 08:39:03 +02:00
Sergio Padrino
0f81f6ec9a Bump version to 0.9.6 2021-04-22 15:27:13 +02:00
Sergio Padrino
0ca3d5caef Allow to actually install prebuilt binaries 2021-04-22 15:26:51 +02:00
Sergio Padrino
1dfb44afbe Merge pull request #10 from desktop/releases/0.9.5
Bump version to 0.9.5
2021-04-22 15:25:49 +02:00
Sergio Padrino
ee5cc6e589 Bump version to 0.9.5 2021-04-22 11:36:06 +02:00
Sergio Padrino
6ad1a5079e Merge pull request #9 from desktop/n-api-prebuilds
Make prebuilds using N-API
2021-04-22 11:34:36 +02:00
Sergio Padrino
d66fb5cf16 Make prebuilds using N-API 2021-04-22 11:24:16 +02:00
12 changed files with 1851 additions and 1602 deletions

View File

@@ -13,15 +13,16 @@ jobs:
name: ${{ matrix.friendlyName }}
runs-on: ${{ matrix.os }}
timeout-minutes: 10
strategy:
fail-fast: false
matrix:
node: [12.14.1]
os: [macos-10.14, windows-2019, ubuntu-18.04]
node: [14.15.4]
os: [macos-latest, windows-latest, ubuntu-18.04]
include:
- os: macos-10.14
- os: macos-latest
friendlyName: macOS
- os: windows-2019
- os: windows-latest
friendlyName: Windows
- os: ubuntu-18.04
friendlyName: Linux
@@ -33,19 +34,33 @@ jobs:
uses: actions/setup-node@v1
with:
node-version: ${{ matrix.node }}
# This step can be removed as soon as official Windows arm64 builds are published:
# https://github.com/nodejs/build/issues/2450#issuecomment-705853342
- run: |
$NodeVersion = (node --version) -replace '^.'
$NodeFallbackVersion = "15.8.0"
& .\script\download-node-lib-win-arm64.ps1 $NodeVersion $NodeFallbackVersion
if: ${{ matrix.os == 'windows-latest' }}
name: Install Windows arm64 node.lib
- name: Install and build
run: yarn
run: |
yarn install
yarn build
- name: Lint
run: yarn lint
- name: Test
run: yarn test
shell: bash
- name: Prebuild Node x64
run: yarn prebuild-node
- name: Prebuild Electron x64
run: yarn prebuild-electron
- name: Prebuild Electron arm64
run: yarn prebuild-electron-arm64
- name: Prebuild (x64)
run: npm run prebuild-napi-x64
- name: Prebuild (arm64)
run: npm run prebuild-napi-arm64
if: ${{ matrix.os != 'ubuntu-18.04' }}
- name: Prebuild (Windows x86)
run: npm run prebuild-napi-ia32
if: ${{ matrix.os == 'windows-latest' }}
- name: Publish
if: github.event_name == 'push' && startsWith(github.ref, 'refs/tags/')
run: yarn upload

70
.github/workflows/codeql-analysis.yml vendored Normal file
View File

@@ -0,0 +1,70 @@
# For most projects, this workflow file will not need changing; you simply need
# to commit it to your repository.
#
# You may wish to alter this file to override the set of languages analyzed,
# or to provide custom queries or build logic.
#
# ******** NOTE ********
# We have attempted to detect the languages in your repository. Please check
# the `language` matrix defined below to confirm you have the correct set of
# supported CodeQL languages.
#
name: "CodeQL"
on:
push:
branches: [ main ]
pull_request:
# The branches below must be a subset of the branches above
branches: [ main ]
schedule:
- cron: '45 23 * * 1'
jobs:
analyze:
name: Analyze
runs-on: ubuntu-latest
permissions:
actions: read
contents: read
security-events: write
strategy:
fail-fast: false
matrix:
language: [ 'cpp', 'javascript', 'python' ]
# CodeQL supports [ 'cpp', 'csharp', 'go', 'java', 'javascript', 'python', 'ruby' ]
# Learn more about CodeQL language support at https://git.io/codeql-language-support
steps:
- name: Checkout repository
uses: actions/checkout@v2
# Initializes the CodeQL tools for scanning.
- name: Initialize CodeQL
uses: github/codeql-action/init@v1
with:
languages: ${{ matrix.language }}
# If you wish to specify custom queries, you can do so here or in a config file.
# By default, queries listed here will override any specified in a config file.
# Prefix the list here with "+" to use these queries and those in the config file.
# queries: ./path/to/local/query, your-org/your-repo/queries@main
# Autobuild attempts to build any compiled languages (C/C++, C#, or Java).
# If this step fails, then you should remove it and run the build manually (see below)
- name: Autobuild
uses: github/codeql-action/autobuild@v1
# Command-line programs to run using the OS shell.
# 📚 https://git.io/JvXDl
# ✏️ If the Autobuild fails above, remove it and uncomment the following three lines
# and modify them (or add more) to build your code if your project
# uses a compiled language
#- run: |
# make bootstrap
# make release
- name: Perform CodeQL Analysis
uses: github/codeql-action/analyze@v1

View File

@@ -2,7 +2,7 @@
A cross-platform no-dependency C executable trampoline which lets GitHub Desktop
intercede in order to provide Git with any additional info it needs (like
credentials through `GIT_ASKPASS`).
credentials through `GIT_ASKPASS` or `SSH_ASKPASS`).
The intention is to support the same platforms that
[Electron supports](https://www.electronjs.org/docs/tutorial/support#supported-platforms).
@@ -125,3 +125,15 @@ Thanks to this, with only one generic trampoline that forwards everything via
that TCP socket, the implementation for every possible protocol like
`GIT_ASKPASS` can live within the GitHub Desktop codebase instead of having
multiple trampoline executables.
## SSH Wrapper
Along with the trampoline, an SSH wrapper is provided for macOS. The reason for
this is macOS before Monterey include an "old" version of OpenSSH that will
ignore the `SSH_ASKPASS` variable unless it's unable to write to a tty.
This SSH wrapper achieves exactly that: just runs whatever `ssh` exists in the
path in a way that will use `SSH_ASKPASS` when necessary.
More recent versions of OpenSSH (starting with 8.3) don't require this wrapper,
since they added support for a new `SSH_ASKPASS_REQUIRE` environment variable.

View File

@@ -2,12 +2,16 @@
'targets': [
{
'target_name': 'desktop-trampoline',
'defines': [
"NAPI_VERSION=<(napi_build_version)",
],
'type': 'executable',
'sources': [
'src/desktop-trampoline.c',
'src/socket.c'
],
'include_dirs': [
'<!(node -p "require(\'node-addon-api\').include_dir")',
'include'
],
'xcode_settings': {
@@ -27,12 +31,17 @@
'-pie',
'-D_FORTIFY_SOURCE=1',
'-fstack-protector-strong',
'-Werror=format-security'
'-Werror=format-security',
'-fno-exceptions'
],
'cflags_cc!': [ '-fno-exceptions' ],
'ldflags!': [
'-z relro',
'-z now'
],
'msvs_settings': {
'VCCLCompilerTool': { 'ExceptionHandling': 1 },
},
'conditions': [
['OS=="win"', {
'defines': [ 'WINDOWS' ],
@@ -42,5 +51,53 @@
}]
]
},
{
'target_name': 'ssh-wrapper',
'defines': [
"NAPI_VERSION=<(napi_build_version)",
],
'type': 'executable',
'sources': [
'src/ssh-wrapper.c'
],
'include_dirs': [
'<!(node -p "require(\'node-addon-api\').include_dir")',
'include'
],
'xcode_settings': {
'OTHER_CFLAGS': [
'-Wall',
'-Werror',
'-Werror=format-security',
'-fPIC',
'-D_FORTIFY_SOURCE=1',
'-fstack-protector-strong'
]
},
'cflags!': [
'-Wall',
'-Werror',
'-fPIC',
'-pie',
'-D_FORTIFY_SOURCE=1',
'-fstack-protector-strong',
'-Werror=format-security',
'-fno-exceptions'
],
'cflags_cc!': [ '-fno-exceptions' ],
'ldflags!': [
'-z relro',
'-z now'
],
'msvs_settings': {
'VCCLCompilerTool': { 'ExceptionHandling': 1 },
},
'conditions': [
# For now only build it for macOS, since it's not needed on Windows
['OS=="win"', {
'defines': [ 'WINDOWS' ],
}]
]
},
],
}

3
index.d.ts vendored
View File

@@ -1,2 +1,5 @@
export function getDesktopTrampolinePath(): string
export function getDesktopTrampolineFilename(): string
export function getSSHWrapperPath(): string
export function getSSHWrapperFilename(): string

View File

@@ -15,7 +15,17 @@ function getDesktopTrampolineFilename() {
: 'desktop-trampoline'
}
function getSSHWrapperPath() {
return Path.join(__dirname, 'build', 'Release', getSSHWrapperFilename())
}
function getSSHWrapperFilename() {
return process.platform === 'win32' ? 'ssh-wrapper.exe' : 'ssh-wrapper'
}
module.exports = {
getDesktopTrampolinePath,
getDesktopTrampolineFilename,
getSSHWrapperPath,
getSSHWrapperFilename,
}

View File

@@ -1,6 +1,6 @@
{
"name": "desktop-trampoline",
"version": "0.9.4",
"version": "0.9.8",
"main": "index.js",
"keywords": [],
"author": "",
@@ -15,10 +15,10 @@
"test": "jest",
"lint": "prettier -c **/*.js **/*.md",
"lint:fix": "prettier --write **/*.js **/*.md",
"prebuild-node": "prebuild -t 10.11.0 -t 11.9.0 -t 12.0.0 -t 14.8.0 --strip --include-regex \"desktop-trampoline(\\.exe)?$\"",
"prebuild-electron": "prebuild -t 7.0.0 -t 8.0.0 -t 9.0.0 -t 10.0.0 -t 11.0.0 -r electron --strip --include-regex \"desktop-trampoline(\\.exe)?$\"",
"prebuild-electron-arm64": "prebuild -t 7.0.0 -t 8.0.0 -t 9.0.0 -t 10.0.0 -t 11.0.0 -r electron -a arm64 --strip --include-regex \"desktop-trampoline(\\.exe)?$\"",
"prebuild-all": "yarn prebuild-node && yarn prebuild-electron && yarn prebuild-electron-arm64",
"prebuild-napi-x64": "prebuild -t 3 -r napi -a x64 --strip --include-regex \"(desktop-trampoline|ssh-wrapper)(\\.exe)?$\"",
"prebuild-napi-ia32": "prebuild -t 3 -r napi -a ia32 --strip --include-regex \"(desktop-trampoline|ssh-wrapper)(\\.exe)?$\"",
"prebuild-napi-arm64": "prebuild -t 3 -r napi -a arm64 --strip --include-regex \"(desktop-trampoline|ssh-wrapper)(\\.exe)?$\"",
"prebuild-all": "yarn prebuild-napi-x64 && yarn prebuild-napi-ia32 && yarn prebuild-napi-arm64",
"upload": "node ./script/upload.js"
},
"repository": {
@@ -29,11 +29,24 @@
"url": "https://github.com/desktop/desktop-trampoline/issues"
},
"homepage": "https://github.com/desktop/desktop-trampoline#readme",
"dependencies": {
"node-addon-api": "^4.3.0",
"prebuild-install": "^7.0.1"
},
"devDependencies": {
"jest": "^26.4.2",
"node-gyp": "^7.1.0",
"prebuild": "^10.0.1",
"prettier": "^2.1.2",
"split2": "^3.2.2"
"jest": "^27.5.0",
"node-gyp": "^8.4.1",
"prebuild": "^11.0.3",
"prettier": "^2.5.1",
"split2": "^4.1.0"
},
"binary": {
"napi_versions": [
3
]
},
"config": {
"runtime": "napi",
"target": 3
}
}

View File

@@ -0,0 +1,36 @@
# This script can be removed as soon as official Windows arm64 builds are published:
# https://github.com/nodejs/build/issues/2450#issuecomment-705853342
$nodeVersion = $args[0]
$fallbackVersion = $args[1]
If ($null -eq $nodeVersion -Or $null -eq $fallbackVersion) {
Write-Error "No NodeJS version given as argument to this file. Run it like download-nodejs-win-arm64.ps1 NODE_VERSION NODE_FALLBACK_VERSION"
exit 1
}
$url = "https://unofficial-builds.nodejs.org/download/release/v$nodeVersion/win-arm64/node.lib"
$fallbackUrl = "https://unofficial-builds.nodejs.org/download/release/v$fallbackVersion/win-arm64/node.lib"
# Always write to the $nodeVersion cache folder, even if we're using the fallbackVersion
$cacheFolder = "$env:TEMP\prebuild\napi\$nodeVersion\arm64"
If (!(Test-Path $cacheFolder)) {
New-Item -ItemType Directory -Force -Path $cacheFolder
}
$output = "$cacheFolder\node.lib"
$start_time = Get-Date
Try {
Invoke-WebRequest -Uri $url -OutFile $output
$downloadedNodeVersion = $nodeVersion
} Catch {
If ($_.Exception.Response -And $_.Exception.Response.StatusCode -eq "NotFound") {
Write-Output "No arm64 node.lib found for Node Windows $nodeVersion, trying fallback version $fallbackVersion..."
Invoke-WebRequest -Uri $fallbackUrl -OutFile $output
$downloadedNodeVersion = $fallbackVersion
}
}
Write-Output "Downloaded arm64 NodeJS lib v$downloadedNodeVersion to $output in $((Get-Date).Subtract($start_time).Seconds) second(s)"

37
src/ssh-wrapper.c Normal file
View File

@@ -0,0 +1,37 @@
#ifdef WINDOWS
int main(int argc, char **argv) {
// Not needed on Windows, this will just create a dummy executable
return -1;
}
#else
#include <unistd.h>
#include <stdio.h>
/**
* This is a wrapper for the ssh command. It is used to make sure ssh runs without
* a tty on macOS, allowing GitHub Desktop to intercept different prompts from
* ssh (e.g. passphrase, adding a host to the list of known hosts...).
* This is not necessary on more recent versions of OpenSSH (starting with v8.3)
* which include support for the SSH_ASKPASS_REQUIRE environment variable.
*/
int main(int argc, char **argv) {
pid_t child = fork();
if (child < 0) {
fprintf(stderr, "Failed to fork\n");
return -1;
}
if (child != 0) {
// This is the parent process. Just exit.
return 0;
}
setsid();
return execvp("ssh", argv);
}
#endif

42
test/ssh-wrapper.test.js Normal file
View File

@@ -0,0 +1,42 @@
const { stat, access } = require('fs').promises
const { constants } = require('fs')
const { execFile } = require('child_process')
const { promisify } = require('util')
const { getSSHWrapperPath } = require('../index')
const sshWrapperPath = getSSHWrapperPath()
const run = promisify(execFile)
describe('ssh-wrapper', () => {
it('exists and is a regular file', async () =>
expect((await stat(sshWrapperPath)).isFile()).toBe(true))
// On Windows, the binary generated is just useless, so no point to test it.
// Also, this won't be used on Linux (for now at least), so don't bother to
// run the tests there.
if (process.platform !== 'darwin') {
return
}
it('can be executed by current process', () =>
access(sshWrapperPath, constants.X_OK))
it('attempts to use ssh-askpass program', async () => {
// Try to connect to github.com with a non-existent known_hosts file to force
// ssh to prompt the user and use askpass.
const result = await run(
sshWrapperPath,
['-o', 'UserKnownHostsFile=/path/to/fake/known_hosts', 'git@github.com'],
{
env: {
SSH_ASKPASS: '/path/to/fake/ssh-askpass',
DISPLAY: '.',
},
}
)
expect(result.stderr).toMatch(
/ssh_askpass: exec\(\/path\/to\/fake\/ssh-askpass\): No such file or directory/
)
})
})

3112
yarn.lock
View File

File diff suppressed because it is too large Load Diff