summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--.gitignore2
-rw-r--r--0001-Use-system-uv-zlib.patch25
-rw-r--r--0002-Revert-deps-V8-tagged.patch141
-rw-r--r--btest402.js151
-rw-r--r--nodejs.spec484
-rw-r--r--nodejs_native.attr2
-rw-r--r--npmrc3
-rw-r--r--sources2
8 files changed, 810 insertions, 0 deletions
diff --git a/.gitignore b/.gitignore
index e69de29..104864b 100644
--- a/.gitignore
+++ b/.gitignore
@@ -0,0 +1,2 @@
+/icu4c-73_2-data-bin-l.zip
+/node-v20.11.1.tar.xz
diff --git a/0001-Use-system-uv-zlib.patch b/0001-Use-system-uv-zlib.patch
new file mode 100644
index 0000000..4872187
--- /dev/null
+++ b/0001-Use-system-uv-zlib.patch
@@ -0,0 +1,25 @@
+From 114ef2bb6ce7712a9b6be0593d38e6a8874f8b67 Mon Sep 17 00:00:00 2001
+From: misaka00251 <liuxin@iscas.ac.cn>
+Date: Mon, 22 May 2023 21:44:32 +0800
+Subject: [PATCH] Use system uv & zlib
+
+---
+ Makefile | 2 +-
+ 1 file changed, 1 insertion(+), 1 deletion(-)
+
+diff --git a/Makefile b/Makefile
+index 0be0659d..3c442014 100644
+--- a/Makefile
++++ b/Makefile
+@@ -169,7 +169,7 @@ with-code-cache test-code-cache:
+ $(warning '$@' target is a noop)
+
+ out/Makefile: config.gypi common.gypi node.gyp \
+- deps/uv/uv.gyp deps/llhttp/llhttp.gyp deps/zlib/zlib.gyp \
++ deps/llhttp/llhttp.gyp \
+ deps/simdutf/simdutf.gyp deps/ada/ada.gyp \
+ tools/v8_gypfiles/toolchain.gypi tools/v8_gypfiles/features.gypi \
+ tools/v8_gypfiles/inspector.gypi tools/v8_gypfiles/v8.gyp
+--
+2.39.2 (Apple Git-143)
+
diff --git a/0002-Revert-deps-V8-tagged.patch b/0002-Revert-deps-V8-tagged.patch
new file mode 100644
index 0000000..2dfded5
--- /dev/null
+++ b/0002-Revert-deps-V8-tagged.patch
@@ -0,0 +1,141 @@
+From 1dd706c8c473b31bcef269d6ecde6fc72200ce89 Mon Sep 17 00:00:00 2001
+From: Eustace <eusteuc@outlook.com>
+Date: Mon, 18 Mar 2024 09:31:46 +0800
+Subject: [PATCH] Revert "deps: V8: cherry-pick 13192d6e10fa"
+
+"Tagged" is at an unfinished state here yet.
+This reverts commit bc2ebb972b34f54e042de9636e7451d2526436a9.
+---
+ deps/v8/src/builtins/riscv/builtins-riscv.cc | 2 +-
+ deps/v8/src/codegen/riscv/assembler-riscv-inl.h | 16 ++++++++--------
+ deps/v8/src/codegen/riscv/assembler-riscv.h | 2 +-
+ deps/v8/src/execution/riscv/simulator-riscv.cc | 8 ++++----
+ .../regexp/riscv/regexp-macro-assembler-riscv.cc | 2 +-
+ 5 files changed, 15 insertions(+), 15 deletions(-)
+
+diff --git a/deps/v8/src/builtins/riscv/builtins-riscv.cc b/deps/v8/src/builtins/riscv/builtins-riscv.cc
+index d6091434b9..3404562785 100644
+--- a/deps/v8/src/builtins/riscv/builtins-riscv.cc
++++ b/deps/v8/src/builtins/riscv/builtins-riscv.cc
+@@ -1512,7 +1512,7 @@ static void Generate_InterpreterEnterBytecode(MacroAssembler* masm) {
+ // Set the return address to the correct point in the interpreter entry
+ // trampoline.
+ Label builtin_trampoline, trampoline_loaded;
+- Tagged<Smi> interpreter_entry_return_pc_offset(
++ Smi interpreter_entry_return_pc_offset(
+ masm->isolate()->heap()->interpreter_entry_return_pc_offset());
+ DCHECK_NE(interpreter_entry_return_pc_offset, Smi::zero());
+
+diff --git a/deps/v8/src/codegen/riscv/assembler-riscv-inl.h b/deps/v8/src/codegen/riscv/assembler-riscv-inl.h
+index ca6d641e2c..55f191e6af 100644
+--- a/deps/v8/src/codegen/riscv/assembler-riscv-inl.h
++++ b/deps/v8/src/codegen/riscv/assembler-riscv-inl.h
+@@ -128,9 +128,9 @@ Handle<HeapObject> Assembler::compressed_embedded_object_handle_at(
+ }
+
+ void Assembler::deserialization_set_special_target_at(
+- Address instruction_payload, Tagged<Code> code, Address target) {
++ Address instruction_payload, Code code, Address target) {
+ set_target_address_at(instruction_payload,
+- !code.is_null() ? code->constant_pool() : kNullAddress,
++ !code.is_null() ? code.constant_pool() : kNullAddress,
+ target);
+ }
+
+@@ -159,13 +159,12 @@ void Assembler::deserialization_set_target_internal_reference_at(
+ }
+ }
+
+-Tagged<HeapObject> RelocInfo::target_object(PtrComprCageBase cage_base) {
++HeapObject RelocInfo::target_object(PtrComprCageBase cage_base) {
+ DCHECK(IsCodeTarget(rmode_) || IsEmbeddedObjectMode(rmode_));
+ if (IsCompressedEmbeddedObject(rmode_)) {
+- return HeapObject::cast(
+- Tagged<Object>(V8HeapCompressionScheme::DecompressTagged(
+- cage_base,
+- Assembler::target_compressed_address_at(pc_, constant_pool_))));
++ return HeapObject::cast(Object(V8HeapCompressionScheme::DecompressTagged(
++ cage_base,
++ Assembler::target_compressed_address_at(pc_, constant_pool_))));
+ } else {
+ return HeapObject::cast(
+ Object(Assembler::target_address_at(pc_, constant_pool_)));
+@@ -187,7 +186,8 @@ Handle<HeapObject> RelocInfo::target_object_handle(Assembler* origin) {
+ }
+ }
+
+-void RelocInfo::set_target_object(Tagged<HeapObject> target,
++void RelocInfo::set_target_object(Heap* heap, HeapObject target,
++ WriteBarrierMode write_barrier_mode,
+ ICacheFlushMode icache_flush_mode) {
+ DCHECK(IsCodeTarget(rmode_) || IsEmbeddedObjectMode(rmode_));
+ if (IsCompressedEmbeddedObject(rmode_)) {
+diff --git a/deps/v8/src/codegen/riscv/assembler-riscv.h b/deps/v8/src/codegen/riscv/assembler-riscv.h
+index bcd5a62d32..ed222b52d6 100644
+--- a/deps/v8/src/codegen/riscv/assembler-riscv.h
++++ b/deps/v8/src/codegen/riscv/assembler-riscv.h
+@@ -286,7 +286,7 @@ class V8_EXPORT_PRIVATE Assembler : public AssemblerBase,
+ // This is for calls and branches within generated code. The serializer
+ // has already deserialized the lui/ori instructions etc.
+ inline static void deserialization_set_special_target_at(Address location,
+- Tagged<Code> code,
++ Code code,
+ Address target);
+
+ // Get the size of the special target encoded at 'instruction_payload'.
+diff --git a/deps/v8/src/execution/riscv/simulator-riscv.cc b/deps/v8/src/execution/riscv/simulator-riscv.cc
+index 052a2d67dd..9582db4896 100644
+--- a/deps/v8/src/execution/riscv/simulator-riscv.cc
++++ b/deps/v8/src/execution/riscv/simulator-riscv.cc
+@@ -1781,7 +1781,7 @@ void RiscvDebugger::Debug() {
+ sreg_t value;
+ StdoutStream os;
+ if (GetValue(arg1, &value)) {
+- Tagged<Object> obj(value);
++ Object obj(value);
+ os << arg1 << ": \n";
+ #ifdef DEBUG
+ obj.Print(os);
+@@ -1830,7 +1830,7 @@ void RiscvDebugger::Debug() {
+ PrintF(" 0x%012" PRIxPTR " : 0x%016" REGIx_FORMAT
+ " %14" REGId_FORMAT " ",
+ reinterpret_cast<intptr_t>(cur), *cur, *cur);
+- Tagged<Object> obj(*cur);
++ Object obj(*cur);
+ Heap* current_heap = sim_->isolate_->heap();
+ if (obj.IsSmi() ||
+ IsValidHeapObject(current_heap, HeapObject::cast(obj))) {
+@@ -4692,7 +4692,7 @@ bool Simulator::DecodeRvvVS() {
+ Builtin Simulator::LookUp(Address pc) {
+ for (Builtin builtin = Builtins::kFirst; builtin <= Builtins::kLast;
+ ++builtin) {
+- if (builtins_.code(builtin)->contains(isolate_, pc)) return builtin;
++ if (builtins_.code(builtin).contains(isolate_, pc)) return builtin;
+ }
+ return Builtin::kNoBuiltinId;
+ }
+@@ -4709,7 +4709,7 @@ void Simulator::DecodeRVIType() {
+ if (builtin != Builtin::kNoBuiltinId) {
+ auto code = builtins_.code(builtin);
+ if ((rs1_reg() != ra || imm12() != 0)) {
+- if ((Address)get_pc() == code->instruction_start()) {
++ if ((Address)get_pc() == code.InstructionStart()) {
+ sreg_t arg0 = get_register(a0);
+ sreg_t arg1 = get_register(a1);
+ sreg_t arg2 = get_register(a2);
+diff --git a/deps/v8/src/regexp/riscv/regexp-macro-assembler-riscv.cc b/deps/v8/src/regexp/riscv/regexp-macro-assembler-riscv.cc
+index 72f89767eb..4063b4b3d2 100644
+--- a/deps/v8/src/regexp/riscv/regexp-macro-assembler-riscv.cc
++++ b/deps/v8/src/regexp/riscv/regexp-macro-assembler-riscv.cc
+@@ -1211,7 +1211,7 @@ static T* frame_entry_address(Address re_frame, int frame_offset) {
+ int64_t RegExpMacroAssemblerRISCV::CheckStackGuardState(Address* return_address,
+ Address raw_code,
+ Address re_frame) {
+- Tagged<InstructionStream> re_code = InstructionStream::cast(Object(raw_code));
++ InstructionStream re_code = InstructionStream::cast(Object(raw_code));
+ return NativeRegExpMacroAssembler::CheckStackGuardState(
+ frame_entry<Isolate*>(re_frame, kIsolateOffset),
+ static_cast<int>(frame_entry<int64_t>(re_frame, kStartIndexOffset)),
+--
+2.41.0
+
diff --git a/btest402.js b/btest402.js
new file mode 100644
index 0000000..277319c
--- /dev/null
+++ b/btest402.js
@@ -0,0 +1,151 @@
+// Copyright (C) 2014 IBM Corporation and Others. All Rights Reserved.
+// This file is part of the Node.JS ICU enablement work
+// https://github.com/joyent/node/pull/7719
+// and is under the same license.
+//
+// This is a very, very, very basic test of es402
+//
+// URL: https://github.com/srl295/btest402
+// Author: Steven R. Loomis <srl@icu-project.org>
+//
+// for a complete test, see http://test262.ecmascript.org
+//
+// Usage: node btest402.js
+
+try {
+ console.log("You have console.log.");
+} catch(e) {
+ // this works on d8
+ console = { log: print };
+ console.log("Now you have console.log.");
+}
+
+function runbtest() {
+ var summary = {};
+
+ try {
+ var i = Intl;
+ summary.haveIntl = true;
+ console.log("+ Congrats, you have the Intl object.");
+ } catch(e) {
+ console.log("You don't have the Intl object: " + e);
+ }
+
+ if(summary.haveIntl) {
+ var locs = [ "en", "mt", "ja","tlh"];
+ var d = new Date(196400000);
+ for ( var n=0; n<locs.length; n++ ) {
+ var loc = locs[n];
+ var lsummary = summary[loc] = {};
+
+ console.log(loc+":");
+ var sl=null;
+ try {
+ sl = Intl.DateTimeFormat.supportedLocalesOf([loc]);
+ if( sl.length > 0 ) {
+ lsummary.haveSlo = true;
+ }
+ } catch (e) {
+ console.log("SLO err: " + e);
+ }
+ var dstr = "ERR";
+ try {
+ lsummary.dstr = d.toLocaleString(loc,{month: "long",day:"numeric",weekday:"long",year:"numeric"});
+ console.log(" date: (supported:"+sl+") " + lsummary.dstr);
+ } catch (e) {
+ console.log(" Date Format err: " + e);
+ }
+ try {
+ new Intl.v8BreakIterator();
+ console.log(" Intl.v8BreakIterator:" +
+ Intl.v8BreakIterator.supportedLocalesOf(loc) + " Supported, first()==" +
+ new Intl.v8BreakIterator(loc).first() );
+ lsummary.brkOk = true;
+ } catch ( e) {
+ console.log(" Intl.v8BreakIterator error (NOT part of EcmaScript402): " + e);
+ }
+ console.log();
+ }
+ }
+
+ // print summary
+ console.log();
+ console.log("--------- Analysis ---------");
+ stxt = "";
+ if( summary.haveIntl ) {
+ console.log("* You have the 'Intl' object. Congratulations! You have the possibility of being EcmaScript 402 compliant.");
+ stxt += "Have Intl, ";
+
+ if ( !summary.en.haveSlo ) {
+ stxt += "Date:no EN, ";
+ console.log("* English isn't a supported language by the date formatter. Perhaps the data isn't installed properly?");
+ }
+ if ( !summary.tlh.haveSlo ) {
+ stxt += "Date:no 'tlh', ";
+ console.log("* Klingon isn't a supported language by the date formatter. It is without honor!");
+ }
+ // now, what is it actually saying
+ if( summary.en.dstr.indexOf("1970") == -1) {
+ stxt += "Date:bad 'en', ";
+ console.log("* the English date format text looks bad to me. Doesn't even have the year.");
+ } else {
+ if( summary.en.dstr.indexOf("Jan") == -1) {
+ stxt += "Date:bad 'en', ";
+ console.log("* The English date format text looks bad to me. Doesn't have the right month.");
+ }
+ }
+
+ if( summary.mt.dstr == summary.en.dstr ) {
+ stxt += "Date:'mt'=='en', ";
+ console.log("* The English and Maltese look the same to me. Probably a 'small' build.");
+ } else if( summary.mt.dstr.indexOf("1970") == -1) {
+ stxt += "Date:bad 'mt', ";
+ console.log("* the Maltese date format text looks bad to me. Doesn't even have the year. (This data is missing from the Chromium ICU build)");
+ } else {
+ if( summary.mt.dstr.indexOf("Jann") == -1) {
+ stxt += "Date:bad 'mt', ";
+ console.log("* The Maltese date format text looks bad to me. Doesn't have the right month. (This data is missing from the Chromium ICU build)");
+ }
+ }
+
+ if ( !summary.ja.haveSlo ) {
+ stxt += "Date:no 'ja', ";
+ console.log("* Japanese isn't a supported language by the date formatter. Could be a 'small' build.");
+ } else {
+ if( summary.ja.dstr.indexOf("1970") == -1) {
+ stxt += "Date:bad 'ja', ";
+ console.log("* the Japanese date format text looks bad to me. Doesn't even have the year.");
+ } else {
+ if( summary.ja.dstr.indexOf("日") == -1) {
+ stxt += "Date:bad 'ja', ";
+ console.log("* The Japanese date format text looks bad to me.");
+ }
+ }
+ }
+ if ( summary.en.brkOk ) {
+ stxt += "FYI: v8Brk:have 'en', ";
+ console.log("* You have Intl.v8BreakIterator support. (Note: not part of ES402.)");
+ }
+ } else {
+ console.log("* You don't have the 'Intl' object. You aren't EcmaScript 402 compliant.");
+ stxt += " NO Intl. ";
+ }
+
+ // 1-liner.
+ console.log();
+ console.log("----------------");
+ console.log( "SUMMARY:" + stxt );
+}
+
+var dorun = true;
+
+try {
+ if(btest402_noautorun) {
+ dorun = false;
+ }
+} catch(e) {}
+
+if(dorun) {
+ console.log("Running btest..");
+ runbtest();
+}
diff --git a/nodejs.spec b/nodejs.spec
new file mode 100644
index 0000000..cfeb244
--- /dev/null
+++ b/nodejs.spec
@@ -0,0 +1,484 @@
+%global baserelease 2
+%{?!_pkgdocdir:%global _pkgdocdir %{_docdir}/%{name}-%{version}}
+%global nodejs_epoch 1
+%global nodejs_major 20
+%global nodejs_minor 11
+%global nodejs_patch 1
+%global nodejs_abi %{nodejs_major}.%{nodejs_minor}
+%global nodejs_soversion 115
+%global nodejs_version %{nodejs_major}.%{nodejs_minor}.%{nodejs_patch}
+%global nodejs_release %{baserelease}
+%global nodejs_datadir %{_datarootdir}/nodejs
+%global v8_epoch 3
+%global v8_major 11
+%global v8_minor 3
+%global v8_build 244
+%global v8_patch 8
+%global v8_abi %{v8_major}.%{v8_minor}
+%global v8_version %{v8_major}.%{v8_minor}.%{v8_build}.%{v8_patch}
+%global v8_release %{nodejs_epoch}.%{nodejs_major}.%{nodejs_minor}.%{nodejs_patch}.%{nodejs_release}
+%global c_ares_major 1
+%global c_ares_minor 20
+%global c_ares_patch 1
+%global c_ares_version %{c_ares_major}.%{c_ares_minor}.%{c_ares_patch}
+%global llhttp_major 8
+%global llhttp_minor 1
+%global llhttp_patch 1
+%global llhttp_version %{llhttp_major}.%{llhttp_minor}.%{llhttp_patch}
+%global libuv_major 1
+%global libuv_minor 46
+%global libuv_patch 0
+%global libuv_version %{libuv_major}.%{libuv_minor}.%{libuv_patch}
+%global nghttp2_major 1
+%global nghttp2_minor 58
+%global nghttp2_patch 0
+%global nghttp2_version %{nghttp2_major}.%{nghttp2_minor}.%{nghttp2_patch}
+%global icu_major 73
+%global icu_minor 2
+%global icu_version %{icu_major}.%{icu_minor}
+%global icudatadir %{nodejs_datadir}/icudata
+%{!?little_endian: %global little_endian %(%{__python3} -c "import sys;print (0 if sys.byteorder=='big' else 1)")}
+# " this line just fixes syntax highlighting for vim that is confused by the above and continues literal
+%global openssl_minimum 1:1.1.1
+# OpenSSL3
+%global openssl3_minimum 1:3.0.2
+%global punycode_major 2
+%global punycode_minor 1
+%global punycode_patch 0
+%global punycode_version %{punycode_major}.%{punycode_minor}.%{punycode_patch}
+%global npm_epoch 1
+%global npm_major 10
+%global npm_minor 2
+%global npm_patch 4
+%global npm_version %{npm_major}.%{npm_minor}.%{npm_patch}
+%global uvwasi_major 0
+%global uvwasi_minor 0
+%global uvwasi_patch 19
+%global uvwasi_version %{uvwasi_major}.%{uvwasi_minor}.%{uvwasi_patch}
+%global histogram_major 0
+%global histogram_minor 9
+%global histogram_patch 7
+%global histogram_version %{histogram_major}.%{histogram_minor}.%{histogram_patch}
+%global npm_release %{nodejs_epoch}.%{nodejs_major}.%{nodejs_minor}.%{nodejs_patch}.%{nodejs_release}
+
+# dtrace is not supported on Node.js 19+
+%global dtrace_configure %{nil}
+
+Name: nodejs
+Epoch: %{nodejs_epoch}
+Version: %{nodejs_version}
+Release: %{nodejs_release}
+Summary: JavaScript runtime
+License: MIT and Apache-2.0 and ISC and BSD and AFL-2.1
+Group: Development/Languages
+URL: http://nodejs.org/
+
+Source0: https://nodejs.org/dist/v%{version}/node-v%{version}.tar.xz
+Source1: npmrc
+Source2: btest402.js
+Source3: https://github.com/unicode-org/icu/releases/download/release-%{icu_major}-%{icu_minor}/icu4c-%{icu_major}_%{icu_minor}-data-bin-l.zip
+Source4: nodejs_native.attr
+
+Patch0: 0001-Use-system-uv-zlib.patch
+Patch1: 0002-Revert-deps-V8-tagged.patch
+
+BuildRequires: python3-devel python3-setuptools make
+BuildRequires: zlib-devel python3-jinja2
+BuildRequires: brotli-devel python3-unversioned-command
+BuildRequires: gcc >= 8.3.0 jq
+BuildRequires: gcc-c++ >= 8.3.0 unzip
+BuildRequires: nodejs-packaging
+BuildRequires: chrpath
+BuildRequires: ninja-build
+BuildRequires: libatomic
+BuildRequires: systemtap-sdt-devel
+BuildRequires: libuv-devel >= 1:%{libuv_version}
+Requires: libuv >= 1:%{libuv_version}
+
+Provides: bundled(nghttp2) = %{nghttp2_version}
+Provides: bundled(llhttp) = %{llhttp_version}
+
+BuildRequires: openssl-devel >= %{openssl_minimum}
+Requires: openssl >= %{openssl_minimum}
+Requires: ca-certificates
+Requires: nodejs-libs%{?_isa} = %{nodejs_epoch}:%{version}-%{release}
+Recommends: nodejs-full-i18n%{?_isa} = %{nodejs_epoch}:%{version}-%{release}
+Provides: nodejs(abi) = %{nodejs_abi}
+Provides: nodejs(abi%{nodejs_major}) = %{nodejs_abi}
+Provides: nodejs(v8-abi) = %{v8_abi}
+Provides: nodejs(v8-abi%{v8_major}) = %{v8_abi}
+Provides: nodejs(engine) = %{nodejs_version}
+Conflicts: node <= 0.3.2-12
+Provides: nodejs-punycode = %{punycode_version}
+Provides: npm(punycode) = %{punycode_version}
+Provides: bundled(c-ares) = %{c_ares_version}
+Provides: bundled(v8) = %{v8_version}
+Provides: bundled(icu) = %{icu_version}
+Provides: bundled(uvwasi) = %{uvwasi_version}
+Provides: bundled(histogram) = %{histogram_version}
+Provides: bundled(ada) = 2.7.4
+Requires: (nodejs-packaging if rpm-build)
+Recommends: npm >= %{npm_epoch}:%{npm_version}-%{npm_release}%{?dist}
+
+%description
+Node.js is a platform built on Chrome's JavaScript runtime
+for easily building fast, scalable network applications.
+Node.js uses an event-driven, non-blocking I/O model that
+makes it lightweight and efficient, perfect for data-intensive
+real-time applications that run across distributed devices.
+
+%package devel
+Summary: JavaScript runtime - development headers
+Group: Development/Languages
+Requires: %{name}%{?_isa} = %{epoch}:%{nodejs_version}-%{nodejs_release}%{?dist}
+Requires: openssl-devel%{?_isa}
+Requires: zlib-devel%{?_isa}
+Requires: brotli-devel%{?_isa}
+Requires: nodejs-packaging
+Requires: libuv-devel%{?_isa}
+
+%description devel
+Development headers for the Node.js JavaScript runtime.
+
+%package libs
+Summary: Node.js and v8 libraries
+
+%if 0%{?__isa_bits} == 64
+Provides: libv8.so.%{v8_major}()(64bit)
+Provides: libv8_libbase.so.%{v8_major}()(64bit)
+Provides: libv8_libplatform.so.%{v8_major}()(64bit)
+%else
+Provides: libv8.so.%{v8_major}
+Provides: libv8_libbase.so.%{v8_major}
+Provides: libv8_libplatform.so.%{v8_major}
+%endif
+
+Provides: v8 = %{v8_epoch}:%{v8_version}-%{nodejs_release}%{?dist}
+Provides: v8%{?_isa} = %{v8_epoch}:%{v8_version}-%{nodejs_release}%{?dist}
+Obsoletes: v8 < 1:6.7.17-10
+
+%description libs
+Libraries to support Node.js and provide stable v8 interfaces.
+
+%package full-i18n
+Summary: Non-English locale data for Node.js
+Requires: %{name}%{?_isa} = %{nodejs_epoch}:%{nodejs_version}-%{nodejs_release}%{?dist}
+
+%description full-i18n
+Optional data files to provide full-icu support for Node.js. Remove this
+package to save space if non-English locales are not needed.
+
+%package -n v8-devel
+Summary: v8 - development headers
+Epoch: %{v8_epoch}
+Version: %{v8_version}
+Release: %{v8_release}
+Requires: %{name}-devel%{?_isa} = %{nodejs_epoch}:%{nodejs_version}-%{nodejs_release}%{?dist}
+
+%description -n v8-devel
+Development headers for the v8 runtime.
+
+%package -n npm
+Summary: Node.js Package Manager
+Epoch: %{npm_epoch}
+Version: %{npm_version}
+Release: %{npm_release}
+
+Obsoletes: npm < 1:9
+Provides: npm = %{npm_epoch}:%{npm_version}
+Requires: nodejs = %{nodejs_epoch}:%{nodejs_version}-%{nodejs_release}%{?dist}
+Recommends: nodejs-docs = %{nodejs_epoch}:%{nodejs_version}-%{nodejs_release}%{?dist}
+Provides: npm(npm) = %{npm_version}
+
+%description -n npm
+npm is a package manager for node.js. You can use it to install and publish
+your node programs. It manages dependencies and does other cool stuff.
+
+%package docs
+Summary: Node.js API documentation
+Group: Documentation
+BuildArch: noarch
+
+Conflicts: %{name} > %{nodejs_epoch}:%{nodejs_version}-%{nodejs_release}%{?dist}
+Conflicts: %{name} < %{nodejs_epoch}:%{nodejs_version}-%{nodejs_release}%{?dist}
+
+%description docs
+The API documentation for the Node.js JavaScript runtime.
+
+%prep
+%autosetup -p1 -n node-v%{nodejs_version}
+rm -rf deps/zlib
+rm -rf deps/brotli
+rm -rf deps/v8/third_party/jinja2
+rm -rf tools/inspector_protocol/jinja2
+pathfix.py -i %{__python3} -pn $(find -type f ! -name "*.js")
+find . -type f -exec sed -i "s~/usr\/bin\/env python~/usr/bin/python3~" {} \;
+find . -type f -exec sed -i "s~/usr\/bin\/python\W~/usr/bin/python3~" {} \;
+sed -i "s~usr\/bin\/python2~usr\/bin\/python3~" ./deps/v8/tools/gen-inlining-tests.py
+sed -i "s~usr\/bin\/python.*$~usr\/bin\/python3~" ./deps/v8/tools/mb/mb_test.py
+find . -type f -exec sed -i "s~python -c~python3 -c~" {} \;
+
+%build
+%global optflags %(echo %{optflags} | sed 's/-g /-g1 /')
+
+export CC='%{__cc}'
+export CXX='%{__cxx}'
+export CFLAGS='%{optflags} \
+ -D_LARGEFILE_SOURCE \
+ -D_FILE_OFFSET_BITS=64 \
+ -DZLIB_CONST \
+ -fno-delete-null-pointer-checks \
+ -O3 \
+ -fno-ipa-icf'
+export CXXFLAGS='%{optflags} \
+ -D_LARGEFILE_SOURCE \
+ -D_FILE_OFFSET_BITS=64 \
+ -DZLIB_CONST \
+ -fno-delete-null-pointer-checks \
+ -O3 \
+ -fno-ipa-icf'
+
+export CFLAGS="$(echo ${CFLAGS} | tr '\n\\' ' ')"
+export CXXFLAGS="$(echo ${CXXFLAGS} | tr '\n\\' ' ')"
+export LDFLAGS="%{build_ldflags}"
+
+%{__python3} configure.py --prefix=%{_prefix} \
+ --verbose \
+ --ninja \
+ --shared \
+ --libdir=%{_lib} \
+ --shared-openssl \
+ --shared-zlib \
+ --shared-brotli \
+ --shared-libuv \
+ --enable-lto \
+ %{dtrace_configure} \
+ --with-intl=small-icu \
+ --with-icu-default-data-dir=%{icudatadir} \
+ --without-corepack \
+ --openssl-use-def-ca-store
+
+%ninja_build -C out/Release
+
+%install
+# The ninja build does not put the shared library in the expected location, so
+# we will move it.
+mv out/Release/lib/libnode.so.%{nodejs_soversion} out/Release/
+./tools/install.py install %{buildroot} %{_prefix}
+
+chmod 0755 %{buildroot}/%{_bindir}/node
+chrpath --delete %{buildroot}%{_bindir}/node
+ln -s libnode.so.%{nodejs_soversion} %{buildroot}%{_libdir}/libnode.so
+
+for header in %{buildroot}%{_includedir}/node/libplatform %{buildroot}%{_includedir}/node/v8*.h; do
+ header=$(basename ${header})
+ ln -s ./node/${header} %{buildroot}%{_includedir}/${header}
+done
+ln -s ./node/cppgc %{buildroot}%{_includedir}/cppgc
+for soname in libv8 libv8_libbase libv8_libplatform; do
+ ln -s libnode.so.%{nodejs_soversion} %{buildroot}%{_libdir}/${soname}.so
+ ln -s libnode.so.%{nodejs_soversion} %{buildroot}%{_libdir}/${soname}.so.%{v8_major}
+done
+
+mkdir -p %{buildroot}%{_prefix}/lib/node_modules
+
+install -Dpm0644 %{SOURCE4} %{buildroot}%{_rpmconfigdir}/fileattrs/nodejs_native.attr
+cat << EOF > %{buildroot}%{_rpmconfigdir}/nodejs_native.req
+#!/bin/sh
+echo 'nodejs(abi%{nodejs_major}) >= %nodejs_abi'
+echo 'nodejs(v8-abi%{v8_major}) >= %v8_abi'
+EOF
+chmod 0755 %{buildroot}%{_rpmconfigdir}/nodejs_native.req
+
+mkdir -p %{buildroot}%{_pkgdocdir}/html
+cp -pr doc/* %{buildroot}%{_pkgdocdir}/html
+rm -f %{buildroot}%{_pkgdocdir}/html/nodejs.1
+
+mkdir -p %{buildroot}%{_datadir}/node
+cp -p common.gypi %{buildroot}%{_datadir}/node
+
+mv %{buildroot}/%{_datadir}/doc/node/gdbinit %{buildroot}/%{_pkgdocdir}/gdbinit
+
+mkdir -p %{buildroot}%{_mandir} \
+ %{buildroot}%{_pkgdocdir}/npm
+
+cp -pr deps/npm/man/* %{buildroot}%{_mandir}/
+rm -rf %{buildroot}%{_prefix}/lib/node_modules/npm/man
+ln -sf %{_mandir} %{buildroot}%{_prefix}/lib/node_modules/npm/man
+
+cp -pr deps/npm/docs %{buildroot}%{_pkgdocdir}/npm/
+rm -rf %{buildroot}%{_prefix}/lib/node_modules/npm/docs
+
+ln -sf %{_pkgdocdir}/npm %{buildroot}%{_prefix}/lib/node_modules/npm/docs
+
+rm -f %{buildroot}/%{_defaultdocdir}/node/lldb_commands.py \
+ %{buildroot}/%{_defaultdocdir}/node/lldbinit
+
+find %{buildroot}%{_prefix}/lib/node_modules/npm \
+ -not -path "%{buildroot}%{_prefix}/lib/node_modules/npm/bin/*" \
+ -executable -type f \
+ -exec chmod -x {} \;
+
+chmod 0755 %{buildroot}%{_prefix}/lib/node_modules/npm/node_modules/@npmcli/run-script/lib/node-gyp-bin/node-gyp
+chmod 0755 %{buildroot}%{_prefix}/lib/node_modules/npm/node_modules/node-gyp/bin/node-gyp.js
+
+mkdir -p %{buildroot}%{_sysconfdir}
+cp %{SOURCE1} %{buildroot}%{_sysconfdir}/npmrc
+
+mkdir -p %{buildroot}%{_prefix}/etc
+ln -s %{_sysconfdir}/npmrc %{buildroot}%{_prefix}/etc/npmrc
+
+mkdir -p %{buildroot}%{icudatadir}
+unzip -d %{buildroot}%{icudatadir} %{SOURCE3} icudt%{icu_major}l.dat
+
+%check
+LD_LIBRARY_PATH=%{buildroot}%{_libdir} %{buildroot}/%{_bindir}/node -e "require('assert').equal(process.versions.node, '%{nodejs_version}')"
+LD_LIBRARY_PATH=%{buildroot}%{_libdir} %{buildroot}/%{_bindir}/node -e "require('assert').equal(process.versions.v8.replace(/-node\.\d+$/, ''), '%{v8_version}')"
+LD_LIBRARY_PATH=%{buildroot}%{_libdir} %{buildroot}/%{_bindir}/node -e "require('assert').equal(process.versions.ares.replace(/-DEV$/, ''), '%{c_ares_version}')"
+
+LD_LIBRARY_PATH=%{buildroot}%{_libdir} %{buildroot}/%{_bindir}/node -e "require(\"assert\").equal(require(\"punycode\").version, '%{punycode_version}')"
+
+LD_LIBRARY_PATH=%{buildroot}%{_libdir} %{buildroot}%{_bindir}/node %{buildroot}%{_bindir}/npm version --json |jq -e '.npm == "%{npm_version}"'
+
+NODE_PATH=%{buildroot}%{_prefix}/lib/node_modules:%{buildroot}%{_prefix}/lib/node_modules/npm/node_modules LD_LIBRARY_PATH=%{buildroot}%{_libdir} %{buildroot}/%{_bindir}/node --icu-data-dir=%{buildroot}%{icudatadir} %{SOURCE2}
+
+
+%files
+%{_bindir}/node
+%dir %{_prefix}/lib/node_modules
+%dir %{_datadir}/node
+
+
+%{_rpmconfigdir}/fileattrs/nodejs_native.attr
+%{_rpmconfigdir}/nodejs_native.req
+%doc CHANGELOG.md onboarding.md GOVERNANCE.md README.md
+%doc %{_mandir}/man1/node.1*
+
+%files devel
+%{_includedir}/node
+%{_libdir}/libnode.so
+%{_datadir}/node/common.gypi
+%{_pkgdocdir}/gdbinit
+
+%files full-i18n
+%dir %{icudatadir}
+%{icudatadir}/icudt%{icu_major}*.dat
+
+%files libs
+%license LICENSE
+%{_libdir}/libnode.so.%{nodejs_soversion}
+%{_libdir}/libv8.so.%{v8_major}
+%{_libdir}/libv8_libbase.so.%{v8_major}
+%{_libdir}/libv8_libplatform.so.%{v8_major}
+%dir %{nodejs_datadir}/
+
+%files -n v8-devel
+%{_includedir}/libplatform
+%{_includedir}/v8*.h
+%{_libdir}/libv8.so
+%{_includedir}/cppgc
+%{_libdir}/libv8_libbase.so
+%{_libdir}/libv8_libplatform.so
+
+%files -n npm
+%{_bindir}/npm
+%{_bindir}/npx
+%{_prefix}/lib/node_modules/npm
+%config(noreplace) %{_sysconfdir}/npmrc
+%{_prefix}/etc/npmrc
+%ghost %{_sysconfdir}/npmignore
+%doc %{_mandir}/man*/
+%exclude %doc %{_mandir}/man1/node.1*
+
+
+%files docs
+%dir %{_pkgdocdir}
+%doc doc
+%{_pkgdocdir}/html
+%{_pkgdocdir}/npm/docs
+
+%changelog
+* Mon Mar 18 2024 Eustace <eusteuc@outlook.com> - 1:20.11.1-2
+- Revert some v8 roll
+
+* Mon Feb 19 2024 wangkai <13474090681@163.com> - 1:20.11.1-1
+- Update to 20.11.1
+- Fix CVE-2023-46809 CVE-2024-21896 CVE-2024-22019 CVE-2024-21892 CVE-2024-24758 CVE-2024-22025
+
+* Mon Nov 27 2023 Jingwiw <wangjingwei@iscas.ac.cn> - 1:20.10.0-1
+- Update to the new LTS version 20.10.0
+- Add CFLAGS "-O3 -fno-ipa-icf"
+- Enable lto
+
+* Thu Aug 31 2023 Funda Wang <fundawang@yeah.net> - 1:18.17.1-1
+- Update to 18.17.1
+- Use huaweicloud.com mirror as default registry
+
+* Thu May 18 2023 misaka00251 <liuxin@iscas.ac.cn> - 1:18.16.0-1
+- Update to 18.16.0
+
+* Sat Mar 11 2023 Tom_zc <tom_toworld@163.com> - 1:16.15.0-3
+- support openssl v3.0.8
+
+* Thu Feb 23 2023 yaoxin <yaoxin30@h-partners.com> - 1:16.15.0-2
+- Fix CVE-2023-0286,CVE-2023-0215,CVE-2022-4304 and CVE-2022-4450
+
+* Fri Feb 10 2023 liyanan <liyanan32@h-partners.com> - 1:16.15.0-1
+- Update to 16.15.0
+
+* Fri Feb 03 2023 yaoxin <yaoxin30@h-partners.com> - 1:12.22.11-5
+- Fix build failed due to openssl update to 3.0
+
+* Wed Nov 16 2022 liyuxiang <liyuxiang@ncti-gba.cn> - 1:12.22.11-4
+- fix CVE-2022-43548
+
+* Wed Aug 03 2022 wangkai <wangkai385@h-partners.com> - 1:12.22.11-3
+- Rollback remove dist
+
+* Tue Aug 02 2022 wulei <wulei80@huawei.com> - 1:12.22.11-2
+- Dist is controlled by the version project, dist is deleted
+
+* Mon Jun 06 2022 wangkai <wangkai385@h-partners.com> - 1:12.22.11-1
+- Update to 12.22.11
+
+* Fri Apr 15 2022 liyanan <liyanan32@h-partners.com> - 1:12.18.4-8
+- The third party software jinja2-support python 3.10.0
+
+* Thu Oct 21 2021 yaoxin <yaoxin30@huawei.com> - 1:12.18.4-7
+- fix CVE-2021-22930
+
+* Mon Aug 16 2021 zhouwenpei <zhouwenpei1@huawei.com> - 1:12.18.4-6
+- use getauxval to fix build failure in node_main.cc
+
+* Tue Jul 20 2021 zhouwenpei <zhouwenpei1@huawei.com> - 1:12.18.4-5
+- fix CVE-2021-22918
+
+* Mon Mar 15 2021 xinghe <xinghe1@huawei.com> - 1:12.18.4-4
+- fix CVE-2021-22883 CVE-2021-22884
+
+* Tue Feb 09 2021 xinghe <xinghe1@huawei.com> - 1:12.18.4-3
+- fix CVE-2020-8265 CVE-2020-8287
+
+* Sat Jan 04 2020 huanghaitao <huanghaitao8@huawei.com> - 1:12.18.4-2
+- Make AARCH64 compile on 64KB physical pages to fix build error
+
+* Wed Nov 18 2020 lingsheng <lingsheng@huawei.com> - 1:12.18.4-1
+- Update to 12.18.4
+
+* Tue Nov 17 2020 lingsheng <lingsheng@huawei.com> - 1:10.21.0-3
+- Fix nodejs release version
+
+* Wed Nov 04 2020 gaozhekang <gaozhekang@huawei.com> - 1:10.21.0-2
+- avoid OOB read in URL parser
+
+* Mon Aug 24 2020 lingsheng <lingsheng@huawei.com> - 1:10.21.0-1
+- Update to 10.21.0
+
+* Thu Aug 20 2020 wutao <wutao61@huawei.com> - 1:10.11.0-3
+- fix dist miss problem
+
+* Fri Mar 20 2020 shijian <shijian16@huawei.com> - 1:10.11.0-2
+- Fix CVE-2018-12122 CVE-2019-5737
+
+* Fri Mar 6 2020 openEuler Buildteam <buildteam@openeuler.org> - 1:10.11.0-1
+- Package init
diff --git a/nodejs_native.attr b/nodejs_native.attr
new file mode 100644
index 0000000..0527af6
--- /dev/null
+++ b/nodejs_native.attr
@@ -0,0 +1,2 @@
+%__nodejs_native_requires %{_rpmconfigdir}/nodejs_native.req
+%__nodejs_native_path ^/usr/lib.*/node_modules/.*\\.node$
diff --git a/npmrc b/npmrc
new file mode 100644
index 0000000..ae9341c
--- /dev/null
+++ b/npmrc
@@ -0,0 +1,3 @@
+prefix=/usr/local
+python=/usr/bin/python3
+registry=https://repo.huaweicloud.com/repository/npm
diff --git a/sources b/sources
new file mode 100644
index 0000000..8ab8ea1
--- /dev/null
+++ b/sources
@@ -0,0 +1,2 @@
+4611f981903a24f59287525ba1a4127c icu4c-73_2-data-bin-l.zip
+3b2fe4d7a4c8dadb58563d822de6c4f4 node-v20.11.1.tar.xz