Skip to content

Commit

Permalink
Merge pull request #139 from odroe/129-dart-run-orm-generate-throws-p…
Browse files Browse the repository at this point in the history
…rocessexception-the-system-cannot-find-the-file-specified

129 dart run orm generate throws processexception the system cannot find the file specified
  • Loading branch information
Seven Du authored Mar 7, 2023
2 parents 88e55b7 + 87de63c commit 98cdfe9
Show file tree
Hide file tree
Showing 9 changed files with 122 additions and 190 deletions.
1 change: 0 additions & 1 deletion .env

This file was deleted.

14 changes: 12 additions & 2 deletions .github/workflows/test.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -3,10 +3,20 @@ on:
- push
- pull_request
jobs:
build:
runs-on: ubuntu-latest
test:
strategy:
matrix:
os:
- ubuntu-latest
- windows-latest
runs-on: ${{ matrix.os }}
steps:
- uses: actions/checkout@v3
- uses: actions/setup-node@v3
with:
node-version: latest
cache: "npm"
- uses: dart-lang/setup-dart@v1
- run: npm ci
- run: dart pub get
- run: dart test
3 changes: 2 additions & 1 deletion .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -10,4 +10,5 @@ docs/.vitepress/*
pubspec.lock

# Development files or directories.
/lib/prisma_client.*
lib/prisma_client.*
prisma/prisma.sqlite
104 changes: 49 additions & 55 deletions bin/generator/prisma_info.dart
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,39 @@ import 'dart:convert' as convert;

import 'package:path/path.dart';

final _separator = Platform.isWindows ? ';' : ':';

Iterable<String> get _globalPaths sync* {
yield Directory.current.path;
yield join(Directory.current.path, 'node_modules', '.bin');
yield* Platform.environment['PATH']?.split(_separator) ?? const <String>[];
}

const Iterable<String> _extensions = <String>[
'cmd', // Windows CMD
'ps1', // Windows PowerShell
'bat', // Windows Batch
'exe', // Windows Executable
'sh', // Shell Script
];

String findNpmExecutable(String executable) {
if (File(executable).existsSync()) {
return executable;
}

return _globalPaths.expand((path) sync* {
// Generate all possible executable paths.
yield* _extensions.map((extension) => join(path, '$executable.$extension'));

// Generate without extension executable paths.
yield join(path, executable);
}).firstWhere((executable) => File(executable).existsSync(), orElse: () {
// If no executable was found, throw an error.
throw StateError('Unable to find NPM executable.');
});
}

class PrismaInfo {
/// Prisma version
final String version;
Expand All @@ -13,61 +46,22 @@ class PrismaInfo {
const PrismaInfo._(this.version, this.platform);

/// Lookup the Prisma version and platform.
factory PrismaInfo.lookup(String packageManager) {
final pm = NodePackageManager(packageManager.trim());

final result = Process.runSync(
pm.toExecutable(),
['exec', if (pm.isNpm) '--', 'prisma', 'version', '--json'],
factory PrismaInfo.lookup(String excutable) {
final String result = Process.runSync(
findNpmExecutable(excutable),
['exec', 'prisma', 'version'],
stdoutEncoding: convert.utf8,
);

// Find JSON block.
final json = result.stdout.toString().split('{').last.split('}').first;
final map = convert.json.decode('{$json}');

return PrismaInfo._(map['prisma'], map['current-platform']);
}
}

class NodePackageManager {
static final separator = Platform.isWindows ? ';' : ':';

final String packageManager;

const NodePackageManager(this.packageManager);

Iterable<String> get executables sync* {
yield packageManager;
yield '$packageManager.cmd';
yield '$packageManager.ps1';
yield '$packageManager.bat';
yield '$packageManager.exe';
}

Iterable<String> get paths sync* {
yield Directory.current.path;
yield* Platform.environment['PATH']?.split(separator) ?? [];
}

/// Find the executable for the package manager full path.
String toExecutable() {
// If the package manager is already an executable, return it.
if (File(packageManager).existsSync()) {
return packageManager;
}

for (final path in paths) {
for (final executable in executables) {
final full = join(path.trim(), executable);
if (File(full).existsSync()) {
return full;
}
}
}

throw StateError('Unable to find $packageManager executable.');
).stdout;

final entries = result
.split('\n')
.where((element) => element.trim().isNotEmpty)
.map((e) => e.split(':'))
.where((element) => element.length == 2)
.map((e) => MapEntry(e[0], e[1]))
.map((e) => MapEntry(e.key.trim().toLowerCase(), e.value.trim()));
final json = Map.fromEntries(entries);

return PrismaInfo._(json['prisma']!, json['current platform']!);
}

bool get isNpm => basename(toExecutable()).toLowerCase().startsWith('npm');
}
19 changes: 3 additions & 16 deletions bin/orm.dart
Original file line number Diff line number Diff line change
Expand Up @@ -12,20 +12,10 @@ import 'generator/prisma_info.dart';
void main(Iterable<String> args) async {
final parase = ArgParser();

// Register the `package-manager` option.
parase.addOption(
'package-manager',
abbr: 'p',
help: 'The NodeJS package manager to use.',
allowed: ['npm', 'yarn', 'pnpm'],
'npm',
defaultsTo: 'npm',
);

// Register the `pm-path` option.
parase.addOption(
'package-manager-executable',
abbr: 'e',
help: 'The path to the NodeJS package manager executable.',
help: 'The npm executable.',
);

// Register the `help` option.
Expand All @@ -42,10 +32,7 @@ void main(Iterable<String> args) async {
// Print the help information if requested.
if (results['help'] as bool) return stdout.writeln(parase.usage);

final executable = results['package-manager-executable'] as String?;
final pm = executable ?? results['package-manager'] as String;
final info = PrismaInfo.lookup(pm);

final info = PrismaInfo.lookup(results['npm']);
final completer = Completer<Generator>();
Library((LibraryBuilder updates) {
updates.name = 'prisma.client';
Expand Down
66 changes: 14 additions & 52 deletions docs/docs/configuration.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,80 +2,42 @@

Prisma client Dart requires no other configuration, just configure the data model in the `schema.prisma` file, but there are always some options!

## Choose Node's package management tool
## NPM executable

Prisma client Dart depends on Node's package management tool, because it needs to interact with Prisma CLI, so you need to choose a Node package management tool, currently supported package management tools are:
By default, you are not required to specify the path to the npm executable. We will search in the global `$PATH`.

- npm
- yarn
- pnpm
If your NPM is not installed globally (after installing NodeJS globally, NPM is available globally), then you need to tell the generator the path of NPM:

You can configure it in the `schema.prisma` file:

::: code-group

```prisma [npm]
generator client {
provider = "dart run orm --package-manager npm"
}
```

```prisma [yarn]
```prisma
generator client {
provider = "dart run orm --package-manager yarn"
provider = "dart run orm --npm=<path>"
}
```

```prisma [pnpm]
generator client {
provider = "dart run orm --package-manager pnpm"
}
```
## Prisma client output

:::
By default, the generated Prisma client Dart code will be placed in the `lib/prisma_client.dart` file.

`--package-manager` is optional, the default value is `npm`, It also has an alias `-p`:
You can change the output path by configuring the `output` option:

::: code-group

```prisma [npm]
generator client {
provider = "dart run orm -p npm"
}
```

```prisma [yarn]
```prisma [Generage into directory]
generator client {
provider = "dart run orm -p yarn"
provider = "dart run orm"
output = "../lib/generated" // Output path is `lib/generated/prisma_client.dart`
}
```

```prisma [pnpm]
```prisma [Generage into file]
generator client {
provider = "dart run orm -p pnpm"
provider = "dart run orm"
output = "../lib/custom_generated.dart" // Output path is `lib/custom_generated.dart
}
```

:::

### Node package manager executable

If you have multiple versions of the Node package manager installed, or if you don't have the Node package manager installed globally. Please use the `-e` parameter to tell the generator:

```prisma
generator client {
provider = "dart run orm -e <executable>"
}
```

Example:

```prisma
generator client {
provider = "dart run orm -e node_modules/.bin/pnpm"
}
```

## About database connection

After the Prisma CLI initializes the project, there will be a `.env` file, which contains the configuration of the database connection. You can configure the database connection in this file:
Expand Down
59 changes: 4 additions & 55 deletions docs/docs/getting-started.md
Original file line number Diff line number Diff line change
Expand Up @@ -21,22 +21,10 @@ dart create hello

Prisma CLI is a command-line tool officially provided by Prisma for managing Prisma projects. It helps you generate Prisma clients, generate the models required by Prisma clients, and perform database migrations.

::: code-group

```bash [npm]
```bash
npm i prisma
```

```bash [yarn]
yarn add prisma
```

```bash [pnpm]
pnpm add prisma
```

:::

> Since Prisma CLI is a Node.js application, you need to have Node.js installed. If you are not familiar with Node.js, you can refer to [Node.js official documentation](https://nodejs.org/).
>
> Also, you need to add the `node_modules` directory to your `.gitignore` file to avoid committing the `node_modules` directory to the Git repository.
Expand All @@ -45,22 +33,10 @@ pnpm add prisma

A Prisma project is a project that contains the Prisma client and the models required by the Prisma client. You can initialize a Prisma project using the Prisma CLI.

::: code-group

```bash [npm]
```bash
npx prisma init
```

```bash [yarn]
yarn prisma init
```

```bash [pnpm]
pnpx prisma init
```

:::

You will get a `.env` file that contains configuration information for the Prisma project.

```bash
Expand Down Expand Up @@ -112,22 +88,10 @@ model User {

## 7. Push the model to the database

::: code-group

```bash [npm]
```bash
npx prisma db push
```

```bash [yarn]
yarn prisma db push
```

```bash [pnpm]
pnpx prisma db push
```

:::

## 8. Install `build_runner` and `json_serializable`

```bash
Expand All @@ -138,23 +102,8 @@ dart pub add json_annotation # optional, but recommended

## 9. Generate Prisma client

::: code-group

```bash [npm]
npx prisma generate
```

```bash [yarn]
yarn prisma generate
```

```bash [pnpm]
pnpx prisma generate
```

:::

```bash
npx prisma generate
dart run build_runner build
```

Expand Down
Loading

1 comment on commit 98cdfe9

@vercel
Copy link

@vercel vercel bot commented on 98cdfe9 Mar 7, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please sign in to comment.