-
-
Notifications
You must be signed in to change notification settings - Fork 681
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Notes on/for v3 / v4 #830
Comments
Maybe do a mix in v3 and in v4 drop callback |
New It handles:
/**
* @typedef {import('filenamify').Options} FilenamifyOptions
* @typedef {import('node:path').ParsedPath} ParsedPath
*
* @typedef FormidableFileOptions
* @property {string} [type] - mime type from header
* @property {number} [lastModified] - file mtimeMs, default `Date.now()`
* @property {number} [size] - file size in bytes
* @property {boolean} [randomName] - generate random filename, default `true` using `cuid`
* @property {FilenamifyOptions} [filenamify] - options passed to `filenamify`
* @property {RenamerFn} [rename] - filename renaming function
* @property {{[key: string]: unknown}} [data] - any additional data or way of storing things
*
* @typedef {(filepath: string, contents: unknown, options: FormidableFileOptions) => string} RenamerFn
* must be synchronous function
*/
import path from 'node:path';
import cuid from 'cuid';
import { filenamifyPath } from 'filenamify';
/**
* @typedef {{type?: string, name: string, lastModified: number}} File
* standard Web API File
*/
// @ts-ignore
import { File } from 'fetch-blob/file.js';
export class FormidableFile extends File {
/**
*
* @param {string} filepath - filename from header
* @param {unknown} contents - any file content value
* @param {FormidableFileOptions} [options]
*/
constructor(filepath, contents, options) {
const { file, opts, fp } = normalize(filepath, contents, options);
super(contents, file.base, {
// @ts-ignore
type: opts.mimetype || opts.mime || opts.type,
lastModified: opts.lastModified || Date.now(),
});
/**
* @type {string} - full filepath
*/
// we assign this here, because to not mess with ParsedPath `file`
this.path = fp;
// eslint-disable-next-line no-underscore-dangle
this._updateProps(file, contents, opts);
}
/**
*
* @param {ParsedPath} file - custom file
* @param {unknown} contents - any file content value
* @param {FormidableFileOptions} [options]
*/
// eslint-disable-next-line no-underscore-dangle
_updateProps(file, contents, options) {
// it's already normalized
// const opts = normalizeOptions(options);
this.basename = file.base;
this.dirname = file.dir;
this.extname = file.ext;
this.stem = file.name; // this.stem == this.name == file.name == basename without extname
this.mime = options.type;
if (options.size) {
// @ts-ignore
this.size = options.size;
}
Reflect.defineProperty(this, 'contents', { value: contents });
Reflect.defineProperty(this, 'options', { value: options });
Reflect.defineProperty(this, 'base', { value: file.base });
Reflect.defineProperty(this, 'dir', { value: file.dir });
Reflect.defineProperty(this, 'ext', { value: file.ext });
Reflect.defineProperty(this, 'data', { value: options.data });
}
}
/**
*
* @param {string} filepath - filename from header
* @param {unknown} contents - any file content value
* @param {FormidableFileOptions} [options]
*
*/
function normalize(filepath, contents, options) {
const opts = normalizeOptions(options);
/**
* @type {string} fp
*/
const fp = filenamifyPath(
// @ts-ignore
opts.rename(filepath, contents, opts),
opts.filenamify,
);
/**
*
* @type {ParsedPath} file
*/
const file = path.parse(fp);
return { file, opts, fp };
}
/**
* @param {{[key: string]: unknown} & FormidableFileOptions} [options]
* @returns {FormidableFileOptions}
*/
function normalizeOptions(options = {}) {
const opts = {
filenamify: { replacement: '-', ...options.filenamify },
randomName: true,
data: { ...options.data },
...options,
};
/**
* @property {RenamerFn}
*/
opts.rename = (f, c, o) => {
if (typeof opts.rename === 'function') {
return opts.rename(f, c, o);
}
return opts.randomName ? cuid() : f;
};
return opts;
}
/**
* examples
*/
const virtualFile = new FormidableFile(
'./foo/bar/qu_sas<mail@at.com>zab.gz<script onchange="alert(1)"></script>sax.png',
await fs.readFile('./examples/10k.json'),
{
type: 'application/json',
randomName: true,
data: { sasa: 1 },
},
);
console.log(virtualFile); Ideally it will be called from Something like that: part.on('data', async () => {
let file = new FormidableFile(nameFromHeader, dataFromParser, {
type: contentTypeFromParser,
rename: formidableOptions.rename,
});
const blob = await createTemporaryBlob(
file.contents,
file.type,
file.options.signal,
);
file.blob = blob;
this.emit('file', file, blob);
if (formidableOptions.save) {
await fs.writeFile(file.path, file.contents);
}
}); We are using The reason of having our own File implementation on top, is that it's for useful properties that we need to pass to the users. Plus I always seeing it that way since I got in here in v1. The properties of it is based on enough battle-tested Currently The whole thing can be introduced in v3 (and require node 14) and backport to v2 (without the temp file creation thing and without The rest of the interesting stuff like iterators and new APIs goes to v4. |
Okay, so the @jimmywarting's cleaned Formidable parser, which is on the Based on the |
wow, didn't know that mine was 2x faster. did you come to any conclusion of why that is? |
Not yet. I'm not in parsers and that parser looks like magic to me LOL. Btw, to make a note: it's faster when use just the class, but not the I'll commit updated benchmarks soon. |
fetch-blob@3.2.0 is publised, you can now use |
btw, formdata is built into NodeJS core now... maybe you can do a conditional import like use some formdata package like formdata-polyfill (used by node-fetch) or formdata-node (both are spec compatible) if it isn't avalible on global namespace so something like
|
Was a super long time ago since i even used vinyl should also impl text, arrayBuffer and stream() 👍 |
Well, yea. But they predated everything i think, that's why. I built some ah.. what a times.. I'm open for ideas and feedback. Not sure how to use these |
So my plan is to add a few tests to #855 then merge and publish. This will probably be the last version 3. And then probably mark 3 as latest. So we can avoid touching v2 like #908 and void having issues that were already fixed in version 3 Then finish #899 which transitions to typescript, so this will be version 4. |
@GrosSacASac Agree. Seems like people are always just using Note: v3 also need to have CJS. We can use |
it could be possible to also just create a simple but small cjs wrapper around ESM since the nature of the parser is async anyways... // index.cjs
module.exports = function (opts) {
return {
// import the ESM module lazily when needed, and pass the arguments along
parse: (...args) => import('./index.js').then(mod => mod.dafault(opts).parse(...args))
}
} Then the code could still be written as ESM and you wouldn't have to duplicate all of your code as a dual cjs/esm package hazard that increases the pkg size? |
@jimmywarting uh, yea. I actually do it similarly here and there. Good point. |
Why not merge the TypeScript PR into version 3? |
Don't think i have much saying in this... but i don't like TS syntax, it dose not run in browsers and never will. you can have equally good type support with plain vanilla javascript. most of the times you don't even need to annotate either javascript or typescript. function getDruation() {
return new Promise<number>((rs) => {
const audio = new Audio()
audio.preload = 'metadata'
audio.src = url
audio.load()
audio.onloadedmetadata = () => {
rs(audio.duration)
}
})
}
async function getDruation() {
const audio = new Audio()
audio.preload = 'metadata'
audio.src = url
audio.load()
await new Promise(rs => audio.onloadedmetadata = rs)
return audio.duration
} then you suddenly don't need to annotate anything. |
@jimmywarting agree. Not fan of TypeScript too. Love the explosion and the attention, and how the ecosystem looks, but gosh I can't stand it to write a whole codebase in TS. I think if it's so needed we can expose just a separate d.ts, and when the codebase is drastically simplified it won't be that needed too. As I said at #899: "I'll review it at a later point, but I don't think a review is much needed. Maybe the only thing that I could ask is to try with separate file approach - e.g. just a typings file that we will distribute with the package. Just to see how it looks like." |
I can understand a few of the frustrations with TypeScript, but maybe it would help to view it as a necessary evil. As it is, I already found a few coding mistakes in the process of converting the codebase to TS. It sounds like the drawback here is having a compilation step and being forced to write code a little differently. IMO, compilation isn’t a big deal. Taking away the ability to write code more freely is what we need consensus on. The alternative to the PR is probably going to be the status quo, type-less development in JS and a .d.ts file that may fall behind on any given commit. |
@tommyhtran agree. But I'm coming from that that the parser could be hard to be typed, and really there's not much need for that - it's an internal. The thing is that we obviously (based on downloads stats of older versions) should focus on legacy and backporting things.. which drives out the motivation to work at all. In 2023 I'll figure out how to make users to migrate or upgrade, and why not we can get paid for helping and consulting. I'm very excited about v3, v4 and formidable-mini, but.. |
Ok lets have a vote (react with one) Continue with current situation (types in a separate project) ❤️ Add type defintion side by side to original source files (.js files + .d.ts files) 🚀 Convert project to typescript and have type definition in the source files, use a compiler to have js files on publish 😄 |
@GrosSacASac i'll move Currently it is on https://github.com/tunnckoCore/opensource/tree/master/modules/formidable-mini. We can start moving tests there, and maybe tag it as v4? It's already API compatible kind of, without the file size options and stuff, but could be used for migrating users. And we can type it too. |
workspaces/
misc
(turborepo /nx.dev + lerna ^5.1 + yarn workspaces)- Hela v6latest
as masterFile
The text was updated successfully, but these errors were encountered: