{"id":482276,"date":"2026-06-03T15:18:27","date_gmt":"2026-06-03T15:18:27","guid":{"rendered":"https:\/\/savepearlharbor.com\/?p=482276"},"modified":"-0001-11-30T00:00:00","modified_gmt":"-0001-11-29T21:00:00","slug":"","status":"publish","type":"post","link":"https:\/\/savepearlharbor.com\/?p=482276","title":{"rendered":"Rust \u0432\u043d\u0443\u0442\u0440\u0438 .NET: \u043a\u0430\u043a \u0443\u043f\u0430\u043a\u043e\u0432\u0430\u0442\u044c native-\u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0443 \u0432 \u043e\u0434\u0438\u043d NuGet-\u043f\u0430\u043a\u0435\u0442"},"content":{"rendered":"<div xmlns=\"http:\/\/www.w3.org\/1999\/xhtml\">\n<p><strong>FFI, P\/Invoke, EmbeddedResource, DllImportResolver \u0438 \u043a\u0440\u043e\u0441\u0441\u043f\u043b\u0430\u0442\u0444\u043e\u0440\u043c\u0435\u043d\u043d\u0430\u044f \u0434\u043e\u0441\u0442\u0430\u0432\u043a\u0430 \u0431\u0435\u0437 \u0440\u0443\u0447\u043d\u043e\u0433\u043e \u043a\u043e\u043f\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f\u00a0<\/strong><code><strong>.dll<\/strong><\/code><strong>,\u00a0<\/strong><code><strong>.so<\/strong><\/code><strong>\u00a0\u0438\u00a0<\/strong><code><strong>.dylib<\/strong><\/code><strong>.<\/strong><\/p>\n<blockquote>\n<p>\u0412 \u043f\u0440\u0438\u043c\u0435\u0440\u0430\u0445 \u043d\u0438\u0436\u0435 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442\u0441\u044f \u0443\u0441\u043b\u043e\u0432\u043d\u0430\u044f \u0444\u0443\u043d\u043a\u0446\u0438\u044f \u0448\u0438\u0444\u0440\u043e\u0432\u0430\u043d\u0438\u044f, \u043d\u043e \u0441\u0442\u0430\u0442\u044c\u044f \u043d\u0435 \u043f\u0440\u043e \u043a\u0440\u0438\u043f\u0442\u043e\u0433\u0440\u0430\u0444\u0438\u044e. \u041e\u0441\u043d\u043e\u0432\u043d\u0430\u044f \u0442\u0435\u043c\u0430 &#8212; FFI, \u0432\u043b\u0430\u0434\u0435\u043d\u0438\u0435 \u043f\u0430\u043c\u044f\u0442\u044c\u044e \u0438 \u0434\u043e\u0441\u0442\u0430\u0432\u043a\u0430 native-\u0431\u0438\u043d\u0430\u0440\u0435\u0439 \u0432 .NET. \u0414\u043b\u044f production-\u043a\u0440\u0438\u043f\u0442\u043e\u0433\u0440\u0430\u0444\u0438\u0438 \u043b\u0443\u0447\u0448\u0435 \u0431\u0440\u0430\u0442\u044c \u043f\u0440\u043e\u0432\u0435\u0440\u0435\u043d\u043d\u044b\u0435 \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0438 \u0438 \u0440\u0435\u0436\u0438\u043c\u044b, \u0430 \u043d\u0435 \u043f\u0438\u0441\u0430\u0442\u044c \u0441\u043e\u0431\u0441\u0442\u0432\u0435\u043d\u043d\u044b\u0439 \u0430\u043b\u0433\u043e\u0440\u0438\u0442\u043c.<\/p>\n<\/blockquote>\n<h3>\u0417\u0430\u0447\u0435\u043c \u044d\u0442\u043e \u043f\u043e\u043d\u0430\u0434\u043e\u0431\u0438\u043b\u043e\u0441\u044c<\/h3>\n<p>\u041a\u043e\u0433\u0434\u0430 .NET-\u043a\u043e\u0434\u0443 \u043d\u0443\u0436\u043d\u043e \u0432\u044b\u0437\u0432\u0430\u0442\u044c Rust-\u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0443, \u043f\u0435\u0440\u0432\u044b\u0439 \u043f\u0440\u043e\u0442\u043e\u0442\u0438\u043f \u043e\u0431\u044b\u0447\u043d\u043e \u0437\u0430\u0432\u043e\u0434\u0438\u0442\u0441\u044f \u0431\u044b\u0441\u0442\u0440\u043e:<\/p>\n<ul>\n<li>\n<p>Rust \u0441\u043e\u0431\u0438\u0440\u0430\u0435\u0442\u0441\u044f \u043a\u0430\u043a\u00a0<code>cdylib<\/code>;<\/p>\n<\/li>\n<li>\n<p>\u0444\u0443\u043d\u043a\u0446\u0438\u0438 \u044d\u043a\u0441\u043f\u043e\u0440\u0442\u0438\u0440\u0443\u044e\u0442\u0441\u044f \u0447\u0435\u0440\u0435\u0437\u00a0<code>extern \"C\"<\/code>;<\/p>\n<\/li>\n<li>\n<p>C# \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u0442 \u0438\u0445 \u0447\u0435\u0440\u0435\u0437\u00a0<code>DllImport<\/code>;<\/p>\n<\/li>\n<li>\n<p>\u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442 \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u0442\u0441\u044f \u0447\u0435\u0440\u0435\u0437 \u0443\u043a\u0430\u0437\u0430\u0442\u0435\u043b\u044c.<\/p>\n<\/li>\n<\/ul>\n<p>\u041f\u0440\u043e\u0431\u043b\u0435\u043c\u044b \u043d\u0430\u0447\u0438\u043d\u0430\u044e\u0442\u0441\u044f \u043f\u043e\u0437\u0436\u0435, \u043a\u043e\u0433\u0434\u0430 \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0443 \u043d\u0443\u0436\u043d\u043e \u043e\u0442\u0434\u0430\u0442\u044c \u0434\u0440\u0443\u0433\u0438\u043c \u043a\u043e\u043c\u0430\u043d\u0434\u0430\u043c \u0438\u043b\u0438 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u0432 \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u0438\u0445 \u0441\u0435\u0440\u0432\u0438\u0441\u0430\u0445.<\/p>\n<p>\u041f\u043e\u0434 Windows \u043d\u0443\u0436\u0435\u043d\u00a0<code>.dll<\/code>, \u043f\u043e\u0434 Linux &#8212;\u00a0<code>.so<\/code>, \u043f\u043e\u0434 macOS &#8212;\u00a0<code>.dylib<\/code>. \u041a\u0442\u043e-\u0442\u043e \u0437\u0430\u0431\u044b\u0432\u0430\u0435\u0442 \u043f\u043e\u043b\u043e\u0436\u0438\u0442\u044c \u0444\u0430\u0439\u043b \u0440\u044f\u0434\u043e\u043c \u0441 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435\u043c, CI \u0441\u043e\u0431\u0438\u0440\u0430\u0435\u0442 \u043d\u0435 \u0442\u043e\u0442 target, \u043f\u0443\u0442\u044c \u0434\u043e native-\u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0438 \u043e\u0442\u043b\u0438\u0447\u0430\u0435\u0442\u0441\u044f \u043d\u0430 \u0440\u0430\u0437\u043d\u044b\u0445 \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u044f\u0445, \u0430 \u043e\u0448\u0438\u0431\u043a\u0430 \u0432\u0441\u043f\u043b\u044b\u0432\u0430\u0435\u0442 \u0442\u043e\u043b\u044c\u043a\u043e \u0432 runtime.<\/p>\n<p>\u0425\u043e\u0447\u0435\u0442\u0441\u044f \u0434\u0440\u0443\u0433\u043e\u0433\u043e \u0441\u0446\u0435\u043d\u0430\u0440\u0438\u044f:<\/p>\n<pre><code class=\"bash\">dotnet add package Ted.Encryption<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:87px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0418 \u0447\u0442\u043e\u0431\u044b \u043f\u043e\u0441\u043b\u0435 \u044d\u0442\u043e\u0433\u043e \u0432\u0441\u0435 \u0440\u0430\u0431\u043e\u0442\u0430\u043b\u043e \u0431\u0435\u0437 \u0440\u0443\u0447\u043d\u043e\u0433\u043e \u043a\u043e\u043f\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f native-\u0444\u0430\u0439\u043b\u043e\u0432.<\/p>\n<p>\u0412 \u044d\u0442\u043e\u0439 \u0441\u0442\u0430\u0442\u044c\u0435 \u043f\u043e\u043a\u0430\u0436\u0443 \u0441\u0445\u0435\u043c\u0443, \u043f\u0440\u0438 \u043a\u043e\u0442\u043e\u0440\u043e\u0439 \u0432\u0441\u0435 native-\u0431\u0438\u043d\u0430\u0440\u043d\u0438\u043a\u0438 \u0443\u043f\u0430\u043a\u043e\u0432\u0430\u043d\u044b \u0432 \u043e\u0434\u0438\u043d NuGet-\u043f\u0430\u043a\u0435\u0442, \u0430 .NET \u0441\u0430\u043c \u0432\u044b\u0431\u0438\u0440\u0430\u0435\u0442 \u0438 \u0437\u0430\u0433\u0440\u0443\u0436\u0430\u0435\u0442 \u043d\u0443\u0436\u043d\u044b\u0439 \u0444\u0430\u0439\u043b \u0447\u0435\u0440\u0435\u0437\u00a0<code>DllImportResolver<\/code>.<\/p>\n<h3>\u0427\u0442\u043e \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u0441\u044f \u0432 \u0438\u0442\u043e\u0433\u0435<\/h3>\n<p>\u041d\u0430 \u0443\u0440\u043e\u0432\u043d\u0435 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f \u043f\u0430\u043a\u0435\u0442 \u0432\u044b\u0433\u043b\u044f\u0434\u0438\u0442 \u043a\u0430\u043a \u043e\u0431\u044b\u0447\u043d\u0430\u044f .NET-\u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0430:<\/p>\n<pre><code>string encrypted = Encryptor.Encrypt(\"hello\", \"key-123\");string decrypted = Encryptor.Decrypt(encrypted, \"key-123\");<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0410 \u0432\u043d\u0443\u0442\u0440\u0438 \u043f\u0440\u043e\u0438\u0441\u0445\u043e\u0434\u0438\u0442 \u0432\u043e\u0442 \u044d\u0442\u043e:<\/p>\n<pre><code>+------------------+        dotnet add package        +-----------------------+| Consumer .NET app | -------------------------------&gt; | NuGet package         |+------------------+                                  +-----------------------+        |                                                        |        | DllImport(\"ted_encryption\")                            |        v                                                        v+------------------+                                  +-----------------------+| Managed wrapper  |                                  | Embedded native files || C# \/ .NET 8      |                                  | .dll \/ .so \/ .dylib   |+------------------+                                  +-----------------------+        |                                                        |        | P\/Invoke                                               |        v                                                        v+--------------------------------------------------------------------------+| Rust cdylib: extern \"C\" functions, C-compatible ABI, manual memory owner |+--------------------------------------------------------------------------+<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u041a\u043b\u044e\u0447\u0435\u0432\u0430\u044f \u043c\u044b\u0441\u043b\u044c:\u00a0<code>P\/Invoke<\/code>\u00a0\u0440\u0435\u0448\u0430\u0435\u0442 \u0432\u044b\u0437\u043e\u0432 \u0444\u0443\u043d\u043a\u0446\u0438\u0438, \u043d\u043e \u043d\u0435 \u0440\u0435\u0448\u0430\u0435\u0442 \u0434\u043e\u0441\u0442\u0430\u0432\u043a\u0443 native-\u0431\u0438\u043d\u0430\u0440\u0435\u0439. \u0414\u043e\u0441\u0442\u0430\u0432\u043a\u0443 \u0440\u0435\u0448\u0430\u0435\u0442 \u0441\u0432\u044f\u0437\u043a\u0430\u00a0<code>EmbeddedResource<\/code>\u00a0+\u00a0<code>DllImportResolver<\/code>.<\/p>\n<h3>\u041e\u0431\u0449\u0430\u044f \u0430\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u0430<\/h3>\n<p>\u0420\u0435\u0448\u0435\u043d\u0438\u0435 \u0441\u043e\u0441\u0442\u043e\u0438\u0442 \u0438\u0437 \u0447\u0435\u0442\u044b\u0440\u0435\u0445 \u0447\u0430\u0441\u0442\u0435\u0439:<\/p>\n<pre><code>+----------------------+      +--------------------------+| Rust crate           |      | .NET wrapper             || crate-type = cdylib  | ---&gt; | DllImport + safe facade  |+----------------------+      +--------------------------+            |                              |            v                              v+----------------------+      +--------------------------+| Native binaries      | ---&gt; | EmbeddedResource         || win\/linux\/macos      |      | inside .NET assembly     |+----------------------+      +--------------------------+                                           |                                           v                              +--------------------------+                              | DllImportResolver        |                              | extract + NativeLibrary  |                              +--------------------------+<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u041d\u0430 runtime-\u043f\u0443\u0442\u0438 \u044d\u0442\u043e \u0432\u044b\u0433\u043b\u044f\u0434\u0438\u0442 \u0442\u0430\u043a:<\/p>\n<pre><code>DllImport(\"ted_encryption\")        |        vNativeLibrary.SetDllImportResolver        |        vDetect OS and architecture        |        vExtract embedded native binary        |        vNativeLibrary.Load(path)        |        vCall Rust function<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<h3>1. Rust: C-compatible ABI<\/h3>\n<p>C# \u043d\u0435 \u043c\u043e\u0436\u0435\u0442 \u043d\u0430\u043f\u0440\u044f\u043c\u0443\u044e \u0432\u044b\u0437\u0432\u0430\u0442\u044c Rust-\u0444\u0443\u043d\u043a\u0446\u0438\u044e \u0441\u00a0<code>String<\/code>,\u00a0<code>Result<\/code>\u00a0\u0438\u043b\u0438 \u0432\u043b\u0430\u0434\u0435\u043d\u0438\u0435\u043c \u0432 \u0441\u0442\u0438\u043b\u0435 Rust. \u041d\u0430 \u0433\u0440\u0430\u043d\u0438\u0446\u0435 \u043d\u0443\u0436\u0435\u043d C-\u0441\u043e\u0432\u043c\u0435\u0441\u0442\u0438\u043c\u044b\u0439 ABI: \u043f\u0440\u0438\u043c\u0438\u0442\u0438\u0432\u044b, \u0441\u044b\u0440\u044b\u0435 \u0443\u043a\u0430\u0437\u0430\u0442\u0435\u043b\u0438 \u0438 \u044f\u0432\u043d\u043e\u0435 \u043f\u0440\u0430\u0432\u0438\u043b\u043e, \u043a\u0442\u043e \u0432\u044b\u0434\u0435\u043b\u044f\u0435\u0442 \u0438 \u043a\u0442\u043e \u043e\u0441\u0432\u043e\u0431\u043e\u0436\u0434\u0430\u0435\u0442 \u043f\u0430\u043c\u044f\u0442\u044c.<\/p>\n<p>\u041f\u0440\u0438\u043c\u0435\u0440:<\/p>\n<pre><code class=\"rust\">use std::ffi::{CStr, CString};use std::os::raw::c_char;#[no_mangle]pub extern \"C\" fn encrypt(input: *const c_char, key: *const c_char) -&gt; *mut c_char {    let input = unsafe { CStr::from_ptr(input) }.to_string_lossy().into_owned();    let key = unsafe { CStr::from_ptr(key) }.to_string_lossy().into_owned();    let result = match do_encrypt(&amp;input, &amp;key) {        Ok(value) =&gt; value,        Err(_) =&gt; String::new(),    };    CString::new(result).unwrap().into_raw()}#[no_mangle]pub extern \"C\" fn free_string(ptr: *mut c_char) {    if ptr.is_null() {        return;    }    unsafe {        let _ = CString::from_raw(ptr);    }}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0417\u0434\u0435\u0441\u044c \u0432\u0430\u0436\u043d\u044b \u0434\u0432\u0430 \u043f\u0440\u0430\u0432\u0438\u043b\u0430.<\/p>\n<p>\u041f\u0435\u0440\u0432\u043e\u0435: \u0443 \u044d\u043a\u0441\u043f\u043e\u0440\u0442\u0438\u0440\u0443\u0435\u043c\u044b\u0445 \u0444\u0443\u043d\u043a\u0446\u0438\u0439 \u0434\u043e\u043b\u0436\u043d\u044b \u0431\u044b\u0442\u044c\u00a0<code>#[no_mangle]<\/code>\u00a0\u0438\u00a0<code>extern \"C\"<\/code>. \u0411\u0435\u0437 \u044d\u0442\u043e\u0433\u043e \u0438\u043c\u044f \u0441\u0438\u043c\u0432\u043e\u043b\u0430 \u0438\u0437\u043c\u0435\u043d\u0438\u0442\u0441\u044f, \u0438\u00a0<code>DllImport<\/code>\u00a0\u0435\u0433\u043e \u043d\u0435 \u043d\u0430\u0439\u0434\u0435\u0442.<\/p>\n<p>\u0412\u0442\u043e\u0440\u043e\u0435: \u043a\u0442\u043e \u0432\u044b\u0434\u0435\u043b\u0438\u043b \u043f\u0430\u043c\u044f\u0442\u044c, \u0442\u043e\u0442 \u0435\u0435 \u0438 \u043e\u0441\u0432\u043e\u0431\u043e\u0436\u0434\u0430\u0435\u0442. \u0415\u0441\u043b\u0438 Rust \u043e\u0442\u0434\u0430\u043b \u0441\u0442\u0440\u043e\u043a\u0443 \u0447\u0435\u0440\u0435\u0437\u00a0<code>CString::into_raw<\/code>, \u043e\u0441\u0432\u043e\u0431\u043e\u0436\u0434\u0430\u0442\u044c \u0435\u0435 \u0434\u043e\u043b\u0436\u0435\u043d Rust \u0447\u0435\u0440\u0435\u0437 \u043f\u0430\u0440\u043d\u0443\u044e \u0444\u0443\u043d\u043a\u0446\u0438\u044e \u0432\u0440\u043e\u0434\u0435\u00a0<code>free_string<\/code>. \u041e\u0441\u0432\u043e\u0431\u043e\u0436\u0434\u0430\u0442\u044c \u0442\u0430\u043a\u043e\u0439 \u0443\u043a\u0430\u0437\u0430\u0442\u0435\u043b\u044c \u0447\u0435\u0440\u0435\u0437\u00a0<code>Marshal.FreeHGlobal<\/code>\u00a0\u043d\u0435\u043b\u044c\u0437\u044f.<\/p>\n<p><code>Cargo.toml<\/code>:<\/p>\n<pre><code>[lib]crate-type = [\"cdylib\"][dependencies]aes-gcm = \"0.10\"base64 = \"0.22\"<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<h3>2. C#: P\/Invoke-\u043e\u0431\u0435\u0440\u0442\u043a\u0430<\/h3>\n<p>\u041d\u0430 C#-\u0441\u0442\u043e\u0440\u043e\u043d\u0435 \u0434\u0435\u043b\u0430\u0435\u043c \u0442\u043e\u043d\u043a\u0438\u0439 native layer \u0438 \u043f\u0443\u0431\u043b\u0438\u0447\u043d\u044b\u0439 \u0431\u0435\u0437\u043e\u043f\u0430\u0441\u043d\u044b\u0439 facade:<\/p>\n<pre><code class=\"cs\">using System.Runtime.InteropServices;internal static class Native{    private const string Lib = \"ted_encryption\";    [DllImport(Lib, EntryPoint = \"encrypt\", CallingConvention = CallingConvention.Cdecl)]    public static extern IntPtr Encrypt(string input, string key);    [DllImport(Lib, EntryPoint = \"free_string\", CallingConvention = CallingConvention.Cdecl)]    public static extern void FreeString(IntPtr ptr);}public static class Encryptor{    public static string Encrypt(string input, string key)    {        IntPtr ptr = Native.Encrypt(input, key);        try        {            return Marshal.PtrToStringAnsi(ptr) ?? string.Empty;        }        finally        {            Native.FreeString(ptr);        }    }}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p><code>CallingConvention.Cdecl<\/code>\u00a0\u043b\u0443\u0447\u0448\u0435 \u0443\u043a\u0430\u0437\u044b\u0432\u0430\u0442\u044c \u044f\u0432\u043d\u043e. Rust\u00a0<code>extern \"C\"<\/code>\u00a0\u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442 C calling convention, \u0430 \u043d\u0435\u044f\u0432\u043d\u044b\u0435 \u0437\u043d\u0430\u0447\u0435\u043d\u0438\u044f \u0432 interop-\u043a\u043e\u0434\u0435 &#8212; \u0445\u043e\u0440\u043e\u0448\u0438\u0439 \u0441\u043f\u043e\u0441\u043e\u0431 \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u044c \u0441\u0442\u0440\u0430\u043d\u043d\u043e\u0435 \u043f\u043e\u0432\u0435\u0434\u0435\u043d\u0438\u0435 \u043d\u0430 \u043e\u0434\u043d\u043e\u0439 \u041e\u0421 \u0438 &#171;\u043f\u043e\u0447\u0435\u043c\u0443-\u0442\u043e \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442&#187; \u043d\u0430 \u0434\u0440\u0443\u0433\u043e\u0439.<\/p>\n<p>\u0415\u0449\u0435 \u043e\u0434\u0438\u043d \u043d\u044e\u0430\u043d\u0441:\u00a0<code>Marshal.PtrToStringAnsi<\/code>\u00a0\u043d\u0435 \u0440\u0430\u0432\u0435\u043d \u0443\u043d\u0438\u0432\u0435\u0440\u0441\u0430\u043b\u044c\u043d\u043e\u043c\u0443 UTF-8-\u0440\u0435\u0448\u0435\u043d\u0438\u044e. \u0415\u0441\u043b\u0438 \u0447\u0435\u0440\u0435\u0437 \u0433\u0440\u0430\u043d\u0438\u0446\u0443 \u0434\u043e\u043b\u0436\u043d\u044b \u0441\u0442\u0430\u0431\u0438\u043b\u044c\u043d\u043e \u0445\u043e\u0434\u0438\u0442\u044c Unicode-\u0441\u0442\u0440\u043e\u043a\u0438, \u043b\u0443\u0447\u0448\u0435 \u044f\u0432\u043d\u043e \u0434\u043e\u0433\u043e\u0432\u043e\u0440\u0438\u0442\u044c\u0441\u044f \u043e UTF-8 \u0438 \u043f\u0435\u0440\u0435\u0434\u0430\u0432\u0430\u0442\u044c \u0431\u0430\u0439\u0442\u044b \u043b\u0438\u0431\u043e \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u0441\u043e\u043e\u0442\u0432\u0435\u0442\u0441\u0442\u0432\u0443\u044e\u0449\u0438\u0439 marshaling. \u0412 \u043b\u044e\u0431\u043e\u043c \u0441\u043b\u0443\u0447\u0430\u0435 Unicode \u0434\u043e\u043b\u0436\u0435\u043d \u0431\u044b\u0442\u044c \u0432 \u0442\u0435\u0441\u0442\u0430\u0445.<\/p>\n<h3>3. \u0423\u043f\u0430\u043a\u043e\u0432\u043a\u0430 native-\u0431\u0438\u043d\u0430\u0440\u0435\u0439 \u0432 assembly<\/h3>\n<p>\u0422\u0435\u043f\u0435\u0440\u044c \u043e\u0441\u043d\u043e\u0432\u043d\u0430\u044f \u0447\u0430\u0441\u0442\u044c: \u0434\u043e\u0441\u0442\u0430\u0432\u043a\u0430.<\/p>\n<p>\u0412\u043c\u0435\u0441\u0442\u043e \u0442\u043e\u0433\u043e \u0447\u0442\u043e\u0431\u044b \u043f\u0440\u043e\u0441\u0438\u0442\u044c \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f \u043f\u0430\u043a\u0435\u0442\u0430 \u0432\u0440\u0443\u0447\u043d\u0443\u044e \u0440\u0430\u0441\u043a\u043b\u0430\u0434\u044b\u0432\u0430\u0442\u044c\u00a0<code>.dll<\/code>,\u00a0<code>.so<\/code>\u00a0\u0438\u00a0<code>.dylib<\/code>, \u0434\u043e\u0431\u0430\u0432\u0438\u043c \u0438\u0445 \u0432 .NET-\u0441\u0431\u043e\u0440\u043a\u0443 \u043a\u0430\u043a\u00a0<code>EmbeddedResource<\/code>.<\/p>\n<p>\u041f\u0440\u0438\u043c\u0435\u0440\u00a0<code>.csproj<\/code>:<\/p>\n<pre><code class=\"xml\">&lt;ItemGroup&gt;  &lt;EmbeddedResource Include=\"native\/win-x64\/ted_encryption.dll\"                    LogicalName=\"ted_encryption.dll\" \/&gt;  &lt;EmbeddedResource Include=\"native\/linux-x64\/libted_encryption.so\"                    LogicalName=\"libted_encryption.so\" \/&gt;  &lt;EmbeddedResource Include=\"native\/osx-x64\/libted_encryption.dylib\"                    LogicalName=\"libted_encryption.dylib\" \/&gt;&lt;\/ItemGroup&gt;<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u041f\u043e\u043b\u0443\u0447\u0430\u0435\u0442\u0441\u044f \u0442\u0430\u043a\u0430\u044f \u0443\u043f\u0430\u043a\u043e\u0432\u043a\u0430:<\/p>\n<pre><code>Ted.Encryption.dll|+-- Managed C# wrapper|+-- Embedded resources    |    +-- ted_encryption.dll    +-- libted_encryption.so    +-- libted_encryption.dylib<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u041f\u043e\u0442\u0440\u0435\u0431\u0438\u0442\u0435\u043b\u044c \u0432\u0438\u0434\u0438\u0442 \u043e\u0431\u044b\u0447\u043d\u044b\u0439 NuGet-\u043f\u0430\u043a\u0435\u0442. Native-\u0444\u0430\u0439\u043b\u044b \u043b\u0435\u0436\u0430\u0442 \u0432\u043d\u0443\u0442\u0440\u0438 \u0441\u0431\u043e\u0440\u043a\u0438 \u0438 \u0434\u043e\u0441\u0442\u0430\u044e\u0442\u0441\u044f \u0442\u043e\u043b\u044c\u043a\u043e \u0432 \u043c\u043e\u043c\u0435\u043d\u0442 \u0437\u0430\u0433\u0440\u0443\u0437\u043a\u0438 \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0438.<\/p>\n<h3>4. DllImportResolver: \u0433\u043b\u0430\u0432\u043d\u044b\u0439 \u0442\u0440\u044e\u043a<\/h3>\n<p>\u0412 .NET Core 3.0+ \u0435\u0441\u0442\u044c \u043c\u0435\u0445\u0430\u043d\u0438\u0437\u043c\u00a0<code>NativeLibrary.SetDllImportResolver<\/code>. \u041e\u043d \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u0435\u0442 \u043f\u0435\u0440\u0435\u0445\u0432\u0430\u0442\u0438\u0442\u044c \u043f\u043e\u043f\u044b\u0442\u043a\u0443 \u0437\u0430\u0433\u0440\u0443\u0437\u0438\u0442\u044c native-\u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0443 \u0438 \u0441\u0430\u043c\u043e\u043c\u0443 \u0440\u0435\u0448\u0438\u0442\u044c, \u043e\u0442\u043a\u0443\u0434\u0430 \u0435\u0435 \u0431\u0440\u0430\u0442\u044c.<\/p>\n<p>\u0420\u0435\u0433\u0438\u0441\u0442\u0440\u0438\u0440\u0443\u0435\u043c resolver \u043e\u0434\u0438\u043d \u0440\u0430\u0437:<\/p>\n<pre><code class=\"cs\">using System.Reflection;using System.Runtime.InteropServices;internal static class Native{    private const string Lib = \"ted_encryption\";    static Native()    {        NativeLibrary.SetDllImportResolver(typeof(Native).Assembly, Resolve);    }    private static IntPtr Resolve(        string libraryName,        Assembly assembly,        DllImportSearchPath? searchPath)    {        if (libraryName != Lib)        {            return IntPtr.Zero;        }        string path = ExtractNativeLibrary(assembly);        return NativeLibrary.Load(path);    }}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0422\u0435\u043f\u0435\u0440\u044c \u0432\u044b\u0431\u0438\u0440\u0430\u0435\u043c \u0440\u0435\u0441\u0443\u0440\u0441 \u043f\u043e \u043f\u043b\u0430\u0442\u0444\u043e\u0440\u043c\u0435 \u0438 \u0440\u0430\u0441\u043f\u0430\u043a\u043e\u0432\u044b\u0432\u0430\u0435\u043c \u0435\u0433\u043e \u0432\u043e \u0432\u0440\u0435\u043c\u0435\u043d\u043d\u0443\u044e \u0434\u0438\u0440\u0435\u043a\u0442\u043e\u0440\u0438\u044e:<\/p>\n<pre><code class=\"cs\">private static string ExtractNativeLibrary(Assembly assembly){    string resourceName =        RuntimeInformation.IsOSPlatform(OSPlatform.Windows) ? \"ted_encryption.dll\" :        RuntimeInformation.IsOSPlatform(OSPlatform.Linux) ? \"libted_encryption.so\" :        RuntimeInformation.IsOSPlatform(OSPlatform.OSX) ? \"libted_encryption.dylib\" :        throw new PlatformNotSupportedException();    string directory = Path.Combine(Path.GetTempPath(), \"ted_encryption\");    Directory.CreateDirectory(directory);    string targetPath = Path.Combine(directory, resourceName);    if (!File.Exists(targetPath))    {        using Stream source = assembly.GetManifestResourceStream(resourceName)            ?? throw new InvalidOperationException($\"Resource {resourceName} not found\");        using FileStream target = File.Create(targetPath);        source.CopyTo(target);    }    return targetPath;}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u041f\u043e\u0441\u043b\u0435 \u044d\u0442\u043e\u0433\u043e\u00a0<code>DllImport(\"ted_encryption\")<\/code>\u00a0\u043d\u0435 \u043f\u044b\u0442\u0430\u0435\u0442\u0441\u044f \u0438\u0441\u043a\u0430\u0442\u044c \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0443 \u0440\u044f\u0434\u043e\u043c \u0441 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435\u043c. \u0412\u043c\u0435\u0441\u0442\u043e \u044d\u0442\u043e\u0433\u043e:<\/p>\n<ol>\n<li>\n<p>.NET \u0432\u0438\u0434\u0438\u0442 DllImport(\u201cted_encryption\u201d)<\/p>\n<\/li>\n<li>\n<p>\u0412\u044b\u0437\u044b\u0432\u0430\u0435\u0442 \u0437\u0430\u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u044b\u0439 resolver<\/p>\n<\/li>\n<li>\n<p>Resolver \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u044f\u0435\u0442 \u041e\u0421<\/p>\n<\/li>\n<li>\n<p>\u0414\u043e\u0441\u0442\u0430\u0435\u0442 \u043d\u0443\u0436\u043d\u044b\u0439 EmbeddedResource<\/p>\n<\/li>\n<li>\n<p>\u0421\u043e\u0445\u0440\u0430\u043d\u044f\u0435\u0442 \u0435\u0433\u043e \u0432\u043e \u0432\u0440\u0435\u043c\u0435\u043d\u043d\u0443\u044e \u043f\u0430\u043f\u043a\u0443<\/p>\n<\/li>\n<li>\n<p>\u0417\u0430\u0433\u0440\u0443\u0436\u0430\u0435\u0442 \u0447\u0435\u0440\u0435\u0437 NativeLibrary.Load<\/p>\n<\/li>\n<li>\n<p>P\/Invoke \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u0442 Rust-\u0444\u0443\u043d\u043a\u0446\u0438\u044e<\/p>\n<\/li>\n<\/ol>\n<p>\u0414\u043b\u044f \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f \u043f\u0430\u043a\u0435\u0442\u0430 \u0432\u0441\u0435 \u044d\u0442\u043e \u043f\u0440\u043e\u0437\u0440\u0430\u0447\u043d\u043e.<\/p>\n<h3>5. \u0410\u0432\u0442\u043e\u043c\u0430\u0442\u0438\u0437\u0430\u0446\u0438\u044f \u0441\u0431\u043e\u0440\u043a\u0438 Rust-\u0431\u0438\u043d\u0430\u0440\u0435\u0439<\/h3>\n<p>\u0427\u0442\u043e\u0431\u044b \u0432 NuGet \u043d\u0435 \u043f\u043e\u043f\u0430\u0434\u0430\u043b\u0438 \u0443\u0441\u0442\u0430\u0440\u0435\u0432\u0448\u0438\u0435 native-\u0444\u0430\u0439\u043b\u044b, \u0441\u0431\u043e\u0440\u043a\u0443 Rust \u043c\u043e\u0436\u043d\u043e \u043f\u0440\u0438\u0432\u044f\u0437\u0430\u0442\u044c \u043a\u00a0<code>dotnet pack<\/code>\u00a0\u0438\u043b\u0438\u00a0<code>Release<\/code>-\u0441\u0431\u043e\u0440\u043a\u0435.<\/p>\n<p>\u041d\u0430\u043f\u0440\u0438\u043c\u0435\u0440, \u0447\u0435\u0440\u0435\u0437 MSBuild target:<\/p>\n<pre><code class=\"xml\">&lt;Target Name=\"BuildRust\" BeforeTargets=\"Build\" Condition=\"'$(Configuration)' == 'Release'\"&gt;  &lt;Exec Command=\"python build.py\" \/&gt;&lt;\/Target&gt;<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0410 \u0432\u00a0<a href=\"http:\/\/build.py\" rel=\"noopener noreferrer nofollow\"><code>build.py<\/code><\/a>\u00a0\u0441\u043e\u0431\u0440\u0430\u0442\u044c \u043d\u0443\u0436\u043d\u044b\u0435 target&#8217;\u044b:<\/p>\n<pre><code class=\"python\">import subprocessTARGETS = {    \"win-x64\": \"x86_64-pc-windows-gnu\",    \"linux-x64\": \"x86_64-unknown-linux-gnu\",    \"osx-x64\": \"x86_64-apple-darwin\",}for output_dir, triple in TARGETS.items():    subprocess.run(        [\"cargo\", \"build\", \"--release\", \"--target\", triple],        check=True,    )    # \u0414\u0430\u043b\u0435\u0435: \u0441\u043a\u043e\u043f\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442 \u0432 native\/&lt;output_dir&gt;\/<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>Python \u0437\u0434\u0435\u0441\u044c \u043d\u0435 \u043e\u0431\u044f\u0437\u0430\u0442\u0435\u043b\u0435\u043d, \u043d\u043e \u0443\u0434\u043e\u0431\u0435\u043d: \u043c\u043e\u0436\u043d\u043e \u043e\u0434\u0438\u043d\u0430\u043a\u043e\u0432\u043e \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0442\u044c \u0441\u0431\u043e\u0440\u043a\u0443 \u043b\u043e\u043a\u0430\u043b\u044c\u043d\u043e \u0438 \u043d\u0430 CI, \u043a\u043e\u043f\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0430\u0440\u0442\u0435\u0444\u0430\u043a\u0442\u044b, \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0442\u044c \u043d\u0430\u043b\u0438\u0447\u0438\u0435 target&#8217;\u043e\u0432 \u0438 \u043f\u0443\u0431\u043b\u0438\u043a\u043e\u0432\u0430\u0442\u044c \u043f\u0430\u043a\u0435\u0442 \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u044b\u043c \u0448\u0430\u0433\u043e\u043c.<\/p>\n<h3>6. WASM \u0438\u0437 \u0442\u043e\u0433\u043e \u0436\u0435 Rust-\u043a\u043e\u0434\u0430<\/h3>\n<p>\u0415\u0441\u043b\u0438 Rust-\u043a\u0440\u0435\u0439\u0442 \u043d\u0443\u0436\u0435\u043d \u0435\u0449\u0435 \u0438 \u0432 \u0431\u0440\u0430\u0443\u0437\u0435\u0440\u0435, \u0435\u0433\u043e \u043c\u043e\u0436\u043d\u043e \u0441\u043e\u0431\u0440\u0430\u0442\u044c \u043f\u043e\u0434\u00a0<code>wasm32-unknown-unknown<\/code>\u00a0\u0438 \u043e\u0442\u0434\u0430\u0442\u044c \u043d\u0430\u0440\u0443\u0436\u0443 \u0447\u0435\u0440\u0435\u0437\u00a0<code>wasm-bindgen<\/code>.<\/p>\n<p>\u041d\u043e FFI-\u0444\u0443\u043d\u043a\u0446\u0438\u0438 \u0441 \u0441\u044b\u0440\u044b\u043c\u0438 \u0443\u043a\u0430\u0437\u0430\u0442\u0435\u043b\u044f\u043c\u0438 \u0438 WASM API \u043b\u0443\u0447\u0448\u0435 \u0440\u0430\u0437\u0432\u0435\u0441\u0442\u0438 \u0447\u0435\u0440\u0435\u0437\u00a0<code>cfg<\/code>:<\/p>\n<pre><code class=\"rust\">#[cfg(target_arch = \"wasm32\")]#[wasm_bindgen]pub fn encrypt_wasm(input: &amp;str, key: &amp;str) -&gt; String {    do_encrypt(input, key).unwrap_or_default()}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0422\u043e\u0433\u0434\u0430 \u043e\u0434\u0438\u043d \u0438 \u0442\u043e\u0442 \u0436\u0435 core-\u043a\u043e\u0434 \u043c\u043e\u0436\u0435\u0442 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c\u0441\u044f \u0432 \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u0438\u0445 \u0432\u0430\u0440\u0438\u0430\u043d\u0442\u0430\u0445 \u0434\u043e\u0441\u0442\u0430\u0432\u043a\u0438:<\/p>\n<pre><code>             +-- Windows .dll             |Rust core ---+-- Linux .so             |             +-- macOS .dylib             |             +-- WASM module<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0412\u0430\u0436\u043d\u043e: WASM &#8212; \u044d\u0442\u043e \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u044b\u0439 \u0441\u043f\u043e\u0441\u043e\u0431 \u0434\u043e\u0441\u0442\u0430\u0432\u043a\u0438. \u041d\u0435 \u0441\u0442\u043e\u0438\u0442 \u043f\u044b\u0442\u0430\u0442\u044c\u0441\u044f \u0442\u0430\u0449\u0438\u0442\u044c C-style FFI API \u0432 \u0431\u0440\u0430\u0443\u0437\u0435\u0440\u043d\u0443\u044e \u0441\u0431\u043e\u0440\u043a\u0443, \u0435\u0441\u043b\u0438 \u0434\u043b\u044f WASM \u043c\u043e\u0436\u043d\u043e \u0434\u0430\u0442\u044c \u043d\u043e\u0440\u043c\u0430\u043b\u044c\u043d\u0443\u044e \u0444\u0443\u043d\u043a\u0446\u0438\u044e \u0441\u00a0<code>&amp;str<\/code>.<\/p>\n<h3>7. \u0422\u0435\u0441\u0442\u044b interop-\u0433\u0440\u0430\u043d\u0438\u0446\u044b<\/h3>\n<p>Interop \u043b\u043e\u043c\u0430\u0435\u0442\u0441\u044f \u043d\u0435 \u0442\u0430\u043c, \u0433\u0434\u0435 \u043f\u0440\u0438\u044f\u0442\u043d\u043e. \u041f\u043e\u044d\u0442\u043e\u043c\u0443 \u0442\u0435\u0441\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043d\u0443\u0436\u043d\u043e \u043d\u0435 \u0442\u043e\u043b\u044c\u043a\u043e happy path.<\/p>\n<p>\u041c\u0438\u043d\u0438\u043c\u0430\u043b\u044c\u043d\u044b\u0439 \u043d\u0430\u0431\u043e\u0440:<\/p>\n<ul>\n<li>\n<p>\u043e\u0431\u044b\u0447\u043d\u0430\u044f \u0441\u0442\u0440\u043e\u043a\u0430;<\/p>\n<\/li>\n<li>\n<p>\u043f\u0443\u0441\u0442\u0430\u044f \u0441\u0442\u0440\u043e\u043a\u0430;<\/p>\n<\/li>\n<li>\n<p>Unicode;<\/p>\n<\/li>\n<li>\n<p>\u043d\u0435\u043f\u0440\u0430\u0432\u0438\u043b\u044c\u043d\u044b\u0439 \u043a\u043b\u044e\u0447;<\/p>\n<\/li>\n<li>\n<p>\u043f\u043e\u0432\u0440\u0435\u0436\u0434\u0435\u043d\u043d\u044b\u0439 payload;<\/p>\n<\/li>\n<li>\n<p>\u043e\u0442\u0441\u0443\u0442\u0441\u0442\u0432\u0438\u0435 native-\u0440\u0435\u0441\u0443\u0440\u0441\u0430;<\/p>\n<\/li>\n<li>\n<p>\u043f\u043e\u0432\u0442\u043e\u0440\u043d\u044b\u0439 \u0432\u044b\u0437\u043e\u0432 \u043f\u043e\u0441\u043b\u0435 \u043f\u0435\u0440\u0432\u043e\u0439 \u0437\u0430\u0433\u0440\u0443\u0437\u043a\u0438 \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0438.<\/p>\n<\/li>\n<\/ul>\n<p>\u041f\u0440\u0438\u043c\u0435\u0440 xUnit-\u0442\u0435\u0441\u0442\u0430:<\/p>\n<pre><code class=\"cs\">public class EncryptorTests{    [Theory]    [InlineData(\"hello\", \"key-123\")]    [InlineData(\"\", \"key-123\")]    [InlineData(\"\u0434\u043b\u0438\u043d\u043d\u0430\u044f \u0441\u0442\u0440\u043e\u043a\u0430 \u0441 \u044e\u043d\u0438\u043a\u043e\u0434\u043e\u043c\", \"another-key\")]    public void RoundTrip_ReturnsOriginal(string text, string key)    {        string encrypted = Encryptor.Encrypt(text, key);        string decrypted = Encryptor.Decrypt(encrypted, key);        Assert.Equal(text, decrypted);    }    [Fact]    public void CorruptToken_DoesNotDecryptToOriginal()    {        string encrypted = Encryptor.Encrypt(\"secret\", \"key\");        string corrupted = encrypted[..^4];        Assert.NotEqual(\"secret\", Encryptor.Decrypt(corrupted, \"key\"));    }}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0415\u0441\u043b\u0438 \u043f\u0430\u043a\u0435\u0442 \u043a\u0440\u043e\u0441\u0441\u043f\u043b\u0430\u0442\u0444\u043e\u0440\u043c\u0435\u043d\u043d\u044b\u0439, \u043f\u043e\u043b\u0435\u0437\u043d\u043e \u0433\u043e\u043d\u044f\u0442\u044c smoke-\u0442\u0435\u0441\u0442\u044b \u043d\u0430 GitHub Actions \u0438\u043b\u0438 \u0434\u0440\u0443\u0433\u043e\u043c CI \u043c\u0438\u043d\u0438\u043c\u0443\u043c \u043f\u043e\u0434 Windows \u0438 Linux. macOS \u0442\u043e\u0436\u0435 \u0436\u0435\u043b\u0430\u0442\u0435\u043b\u044c\u043d\u043e, \u0435\u0441\u043b\u0438 \u043e\u043d\u0430 \u0437\u0430\u044f\u0432\u043b\u0435\u043d\u0430 \u043a\u0430\u043a \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u0432\u0430\u0435\u043c\u0430\u044f \u043f\u043b\u0430\u0442\u0444\u043e\u0440\u043c\u0430.<\/p>\n<h3>\u0413\u0440\u0430\u0431\u043b\u0438, \u043d\u0430 \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0441\u0442\u043e\u0438\u0442 \u0441\u043c\u043e\u0442\u0440\u0435\u0442\u044c<\/h3>\n<p><strong>Calling convention.<\/strong>\u00a0\u0423\u043a\u0430\u0437\u044b\u0432\u0430\u0439\u0442\u0435\u00a0<code>CallingConvention.Cdecl<\/code>\u00a0\u044f\u0432\u043d\u043e. \u041e\u0448\u0438\u0431\u043a\u0438 calling convention \u043e\u0441\u043e\u0431\u0435\u043d\u043d\u043e \u043d\u0435\u043f\u0440\u0438\u044f\u0442\u043d\u044b \u0442\u0435\u043c, \u0447\u0442\u043e \u043c\u043e\u0433\u0443\u0442 \u043f\u0440\u043e\u044f\u0432\u043b\u044f\u0442\u044c\u0441\u044f \u043f\u043e-\u0440\u0430\u0437\u043d\u043e\u043c\u0443 \u043d\u0430 \u0440\u0430\u0437\u043d\u044b\u0445 \u043f\u043b\u0430\u0442\u0444\u043e\u0440\u043c\u0430\u0445.<\/p>\n<p><strong>\u0412\u043b\u0430\u0434\u0435\u043d\u0438\u0435 \u043f\u0430\u043c\u044f\u0442\u044c\u044e.<\/strong>\u00a0\u0415\u0441\u043b\u0438 \u043f\u0430\u043c\u044f\u0442\u044c \u0432\u044b\u0434\u0435\u043b\u0438\u043b Rust, \u043e\u0441\u0432\u043e\u0431\u043e\u0436\u0434\u0430\u0442\u044c \u0435\u0435 \u0434\u043e\u043b\u0436\u0435\u043d Rust. \u0414\u0435\u043b\u0430\u0439\u0442\u0435 \u043f\u0430\u0440\u043d\u044b\u0435 \u0444\u0443\u043d\u043a\u0446\u0438\u0438 \u0432\u0440\u043e\u0434\u0435\u00a0<code>free_string<\/code>.<\/p>\n<p><strong>\u041a\u043e\u0434\u0438\u0440\u043e\u0432\u043a\u0430.<\/strong>\u00a0\u041d\u0435-ASCII \u0441\u0442\u0440\u043e\u043a\u0438 \u0434\u043e\u043b\u0436\u043d\u044b \u0431\u044b\u0442\u044c \u0432 \u0442\u0435\u0441\u0442\u0430\u0445. \u0415\u0441\u043b\u0438 \u043d\u0443\u0436\u0435\u043d \u043f\u0440\u0435\u0434\u0441\u043a\u0430\u0437\u0443\u0435\u043c\u044b\u0439 UTF-8, \u043b\u0443\u0447\u0448\u0435 \u043f\u0435\u0440\u0435\u0434\u0430\u0432\u0430\u0442\u044c \u0431\u0430\u0439\u0442\u044b \u0438 \u044f\u0432\u043d\u043e \u0444\u0438\u043a\u0441\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043a\u043e\u043d\u0442\u0440\u0430\u043a\u0442.<\/p>\n<p><strong>\u041f\u043e\u0432\u0442\u043e\u0440\u043d\u0430\u044f \u0440\u0430\u0441\u043f\u0430\u043a\u043e\u0432\u043a\u0430.<\/strong>\u00a0\u0412 \u043f\u0440\u0438\u043c\u0435\u0440\u0435 \u0444\u0430\u0439\u043b \u043a\u0435\u0448\u0438\u0440\u0443\u0435\u0442\u0441\u044f \u0432\u043e \u0432\u0440\u0435\u043c\u0435\u043d\u043d\u043e\u0439 \u043f\u0430\u043f\u043a\u0435. \u0412 production \u043c\u043e\u0436\u043d\u043e \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0432\u0435\u0440\u0441\u0438\u043e\u043d\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u0443\u044e \u0434\u0438\u0440\u0435\u043a\u0442\u043e\u0440\u0438\u044e \u0438\u043b\u0438 hash, \u0447\u0442\u043e\u0431\u044b \u043e\u0431\u043d\u043e\u0432\u043b\u0435\u043d\u0438\u044f \u043f\u0430\u043a\u0435\u0442\u0430 \u043d\u0435 \u043a\u043e\u043d\u0444\u043b\u0438\u043a\u0442\u043e\u0432\u0430\u043b\u0438 \u0441\u043e \u0441\u0442\u0430\u0440\u044b\u043c extracted-\u0444\u0430\u0439\u043b\u043e\u043c.<\/p>\n<p><strong>\u041f\u0440\u0430\u0432\u0430 \u043d\u0430 \u0432\u044b\u043f\u043e\u043b\u043d\u0435\u043d\u0438\u0435.<\/strong>\u00a0\u041d\u0430 Linux\/macOS \u0438\u043d\u043e\u0433\u0434\u0430 \u0432\u0430\u0436\u043d\u044b \u043f\u0440\u0430\u0432\u0430 \u0444\u0430\u0439\u043b\u0430 \u043f\u043e\u0441\u043b\u0435 \u0440\u0430\u0441\u043f\u0430\u043a\u043e\u0432\u043a\u0438. \u0415\u0441\u043b\u0438 \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u0435 \u0441\u0442\u0440\u043e\u0433\u043e\u0435, \u043f\u0440\u043e\u0432\u0435\u0440\u044c\u0442\u0435 \u044d\u0442\u043e \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u043e.<\/p>\n<p><strong>\u0410\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u0430 CPU.<\/strong>\u00a0\u0412 \u0441\u0442\u0430\u0442\u044c\u0435 \u043f\u043e\u043a\u0430\u0437\u0430\u043d\u044b x64 target&#8217;\u044b. \u0415\u0441\u043b\u0438 \u043d\u0443\u0436\u043d\u044b arm64 \u0438\u043b\u0438 Alpine\/musl, \u0438\u0445 \u043b\u0443\u0447\u0448\u0435 \u044f\u0432\u043d\u043e \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0432 \u043c\u0430\u0442\u0440\u0438\u0446\u0443 \u0441\u0431\u043e\u0440\u043a\u0438 \u0438 naming convention.<\/p>\n<p><strong>WASM \u0438 FFI.<\/strong>\u00a0\u0420\u0430\u0437\u0432\u043e\u0434\u0438\u0442\u0435 C-style FFI \u0438 wasm-bindgen API \u0447\u0435\u0440\u0435\u0437\u00a0<code>cfg<\/code>, \u0438\u043d\u0430\u0447\u0435 \u043e\u0434\u0438\u043d target \u043d\u0430\u0447\u043d\u0435\u0442 \u043c\u0435\u0448\u0430\u0442\u044c \u0434\u0440\u0443\u0433\u043e\u043c\u0443.<\/p>\n<h3>\u0427\u0442\u043e \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u043e\u0441\u044c<\/h3>\n<p>\u041c\u044b \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u0438 \u0441\u0445\u0435\u043c\u0443, \u0432 \u043a\u043e\u0442\u043e\u0440\u043e\u0439:<\/p>\n<ul>\n<li>\n<p>Rust-\u043a\u043e\u0434 \u0441\u043e\u0431\u0438\u0440\u0430\u0435\u0442\u0441\u044f \u0432 native-\u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0438 \u043f\u043e\u0434 \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u041e\u0421;<\/p>\n<\/li>\n<li>\n<p>C# \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u0442 Rust \u0447\u0435\u0440\u0435\u0437 P\/Invoke;<\/p>\n<\/li>\n<li>\n<p>\u043f\u0430\u043c\u044f\u0442\u044c, \u0432\u044b\u0434\u0435\u043b\u0435\u043d\u043d\u0430\u044f \u0432 Rust, \u043e\u0441\u0432\u043e\u0431\u043e\u0436\u0434\u0430\u0435\u0442\u0441\u044f \u043d\u0430 Rust-\u0441\u0442\u043e\u0440\u043e\u043d\u0435;<\/p>\n<\/li>\n<li>\n<p>native-\u0431\u0438\u043d\u0430\u0440\u043d\u0438\u043a\u0438 \u0443\u043f\u0430\u043a\u043e\u0432\u0430\u043d\u044b \u0432\u043d\u0443\u0442\u0440\u044c .NET assembly \u043a\u0430\u043a\u00a0<code>EmbeddedResource<\/code>;<\/p>\n<\/li>\n<li>\n<p><code>DllImportResolver<\/code>\u00a0\u0441\u0430\u043c \u0432\u044b\u0431\u0438\u0440\u0430\u0435\u0442, \u0438\u0437\u0432\u043b\u0435\u043a\u0430\u0435\u0442 \u0438 \u0437\u0430\u0433\u0440\u0443\u0436\u0430\u0435\u0442 \u043d\u0443\u0436\u043d\u0443\u044e \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0443;<\/p>\n<\/li>\n<li>\n<p>\u043f\u043e\u0442\u0440\u0435\u0431\u0438\u0442\u0435\u043b\u044c \u0441\u0442\u0430\u0432\u0438\u0442 \u043e\u0434\u0438\u043d NuGet-\u043f\u0430\u043a\u0435\u0442 \u0438 \u043d\u0435 \u0440\u0430\u0441\u043a\u043b\u0430\u0434\u044b\u0432\u0430\u0435\u0442\u00a0<code>.dll<\/code>,\u00a0<code>.so<\/code>,\u00a0<code>.dylib<\/code>\u00a0\u0432\u0440\u0443\u0447\u043d\u0443\u044e.<\/p>\n<\/li>\n<\/ul>\n<p>\u042d\u0442\u043e \u043d\u0435 \u0441\u0430\u043c\u0430\u044f \u043e\u0447\u0435\u0432\u0438\u0434\u043d\u0430\u044f \u043d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0430, \u043d\u043e \u043f\u043e\u0441\u043b\u0435 \u043f\u0435\u0440\u0432\u043e\u0439 \u0441\u0431\u043e\u0440\u043a\u0438 \u043e\u043d\u0430 \u0441\u0438\u043b\u044c\u043d\u043e \u0443\u043f\u0440\u043e\u0449\u0430\u0435\u0442 \u0436\u0438\u0437\u043d\u044c: \u043c\u0435\u043d\u044c\u0448\u0435 \u0440\u0443\u0447\u043d\u044b\u0445 \u0438\u043d\u0441\u0442\u0440\u0443\u043a\u0446\u0438\u0439, \u043c\u0435\u043d\u044c\u0448\u0435 runtime-\u0441\u044e\u0440\u043f\u0440\u0438\u0437\u043e\u0432 \u0438 \u043f\u043e\u043d\u044f\u0442\u043d\u044b\u0439 \u043f\u0443\u0442\u044c \u0434\u043b\u044f CI\/CD.<\/p>\n<p>\u0415\u0441\u043b\u0438 \u0442\u0435\u043c\u0430 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u0430, \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u044b\u043c \u043f\u0440\u043e\u0434\u043e\u043b\u0436\u0435\u043d\u0438\u0435\u043c \u043c\u043e\u0436\u043d\u043e \u0440\u0430\u0437\u043e\u0431\u0440\u0430\u0442\u044c \u0441\u0431\u043e\u0440\u043a\u0443 \u0442\u0430\u043a\u043e\u0433\u043e \u0436\u0435 Rust-\u043a\u0440\u0435\u0439\u0442\u0430 \u043f\u043e\u0434 WASM \u0438 \u043f\u043e\u0434\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0435 \u0432 \u0431\u0440\u0430\u0443\u0437\u0435\u0440\u043d\u044b\u0439 frontend.<\/p>\n<\/div>\n<p>\u0441\u0441\u044b\u043b\u043a\u0430 \u043d\u0430 \u043e\u0440\u0438\u0433\u0438\u043d\u0430\u043b \u0441\u0442\u0430\u0442\u044c\u0438 <a href=\"https:\/\/habr.com\/ru\/articles\/1043276\/\">https:\/\/habr.com\/ru\/articles\/1043276\/<\/a><\/p>\n","protected":false},"excerpt":{"rendered":"<p>FFI, P\/Invoke, EmbeddedResource, DllImportResolver \u0438 \u043a\u0440\u043e\u0441\u0441\u043f\u043b\u0430\u0442\u0444\u043e\u0440\u043c\u0435\u043d\u043d\u0430\u044f \u0434\u043e\u0441\u0442\u0430\u0432\u043a\u0430 \u0431\u0435\u0437 \u0440\u0443\u0447\u043d\u043e\u0433\u043e \u043a\u043e\u043f\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f\u00a0.dll,\u00a0.so\u00a0\u0438\u00a0.dylib.\u0412 \u043f\u0440\u0438\u043c\u0435\u0440\u0430\u0445 \u043d\u0438\u0436\u0435 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442\u0441\u044f \u0443\u0441\u043b\u043e\u0432\u043d\u0430\u044f \u0444\u0443\u043d\u043a\u0446\u0438\u044f \u0448\u0438\u0444\u0440\u043e\u0432\u0430\u043d\u0438\u044f, \u043d\u043e \u0441\u0442\u0430\u0442\u044c\u044f \u043d\u0435 \u043f\u0440\u043e \u043a\u0440\u0438\u043f\u0442\u043e\u0433\u0440\u0430\u0444\u0438\u044e. \u041e\u0441\u043d\u043e\u0432\u043d\u0430\u044f \u0442\u0435\u043c\u0430 &#8212; FFI, \u0432\u043b\u0430\u0434\u0435\u043d\u0438\u0435 \u043f\u0430\u043c\u044f\u0442\u044c\u044e \u0438 \u0434\u043e\u0441\u0442\u0430\u0432\u043a\u0430 native-\u0431\u0438\u043d\u0430\u0440\u0435\u0439 \u0432 .NET. \u0414\u043b\u044f production-\u043a\u0440\u0438\u043f\u0442\u043e\u0433\u0440\u0430\u0444\u0438\u0438 \u043b\u0443\u0447\u0448\u0435 \u0431\u0440\u0430\u0442\u044c \u043f\u0440\u043e\u0432\u0435\u0440\u0435\u043d\u043d\u044b\u0435 \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0438 \u0438 \u0440\u0435\u0436\u0438\u043c\u044b, \u0430 \u043d\u0435 \u043f\u0438\u0441\u0430\u0442\u044c \u0441\u043e\u0431\u0441\u0442\u0432\u0435\u043d\u043d\u044b\u0439 \u0430\u043b\u0433\u043e\u0440\u0438\u0442\u043c.\u0417\u0430\u0447\u0435\u043c \u044d\u0442\u043e \u043f\u043e\u043d\u0430\u0434\u043e\u0431\u0438\u043b\u043e\u0441\u044c\u041a\u043e\u0433\u0434\u0430 .NET-\u043a\u043e\u0434\u0443 \u043d\u0443\u0436\u043d\u043e \u0432\u044b\u0437\u0432\u0430\u0442\u044c Rust-\u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0443, \u043f\u0435\u0440\u0432\u044b\u0439 \u043f\u0440\u043e\u0442\u043e\u0442\u0438\u043f \u043e\u0431\u044b\u0447\u043d\u043e \u0437\u0430\u0432\u043e\u0434\u0438\u0442\u0441\u044f \u0431\u044b\u0441\u0442\u0440\u043e:Rust \u0441\u043e\u0431\u0438\u0440\u0430\u0435\u0442\u0441\u044f \u043a\u0430\u043a\u00a0cdylib;\u0444\u0443\u043d\u043a\u0446\u0438\u0438 \u044d\u043a\u0441\u043f\u043e\u0440\u0442\u0438\u0440\u0443\u044e\u0442\u0441\u044f \u0447\u0435\u0440\u0435\u0437\u00a0extern &#171;C&#187;;C# \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u0442 \u0438\u0445 \u0447\u0435\u0440\u0435\u0437\u00a0DllImport;\u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442 \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u0442\u0441\u044f \u0447\u0435\u0440\u0435\u0437 \u0443\u043a\u0430\u0437\u0430\u0442\u0435\u043b\u044c.\u041f\u0440\u043e\u0431\u043b\u0435\u043c\u044b \u043d\u0430\u0447\u0438\u043d\u0430\u044e\u0442\u0441\u044f \u043f\u043e\u0437\u0436\u0435, \u043a\u043e\u0433\u0434\u0430 \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0443 \u043d\u0443\u0436\u043d\u043e \u043e\u0442\u0434\u0430\u0442\u044c \u0434\u0440\u0443\u0433\u0438\u043c \u043a\u043e\u043c\u0430\u043d\u0434\u0430\u043c \u0438\u043b\u0438 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u0432 \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u0438\u0445 \u0441\u0435\u0440\u0432\u0438\u0441\u0430\u0445.\u041f\u043e\u0434 Windows \u043d\u0443\u0436\u0435\u043d\u00a0.dll, \u043f\u043e\u0434 Linux &#8212;\u00a0.so, \u043f\u043e\u0434 macOS &#8212;\u00a0.dylib. \u041a\u0442\u043e-\u0442\u043e \u0437\u0430\u0431\u044b\u0432\u0430\u0435\u0442 \u043f\u043e\u043b\u043e\u0436\u0438\u0442\u044c \u0444\u0430\u0439\u043b \u0440\u044f\u0434\u043e\u043c \u0441 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435\u043c, CI \u0441\u043e\u0431\u0438\u0440\u0430\u0435\u0442 \u043d\u0435 \u0442\u043e\u0442 target, \u043f\u0443\u0442\u044c \u0434\u043e native-\u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0438 \u043e\u0442\u043b\u0438\u0447\u0430\u0435\u0442\u0441\u044f \u043d\u0430 \u0440\u0430\u0437\u043d\u044b\u0445 \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u044f\u0445, \u0430 \u043e\u0448\u0438\u0431\u043a\u0430 \u0432\u0441\u043f\u043b\u044b\u0432\u0430\u0435\u0442 \u0442\u043e\u043b\u044c\u043a\u043e \u0432 runtime.\u0425\u043e\u0447\u0435\u0442\u0441\u044f \u0434\u0440\u0443\u0433\u043e\u0433\u043e \u0441\u0446\u0435\u043d\u0430\u0440\u0438\u044f:dotnet add package Ted.Encryption\u0418 \u0447\u0442\u043e\u0431\u044b \u043f\u043e\u0441\u043b\u0435 \u044d\u0442\u043e\u0433\u043e \u0432\u0441\u0435 \u0440\u0430\u0431\u043e\u0442\u0430\u043b\u043e \u0431\u0435\u0437 \u0440\u0443\u0447\u043d\u043e\u0433\u043e \u043a\u043e\u043f\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f native-\u0444\u0430\u0439\u043b\u043e\u0432.\u0412 \u044d\u0442\u043e\u0439 \u0441\u0442\u0430\u0442\u044c\u0435 \u043f\u043e\u043a\u0430\u0436\u0443 \u0441\u0445\u0435\u043c\u0443, \u043f\u0440\u0438 \u043a\u043e\u0442\u043e\u0440\u043e\u0439 \u0432\u0441\u0435 native-\u0431\u0438\u043d\u0430\u0440\u043d\u0438\u043a\u0438 \u0443\u043f\u0430\u043a\u043e\u0432\u0430\u043d\u044b \u0432 \u043e\u0434\u0438\u043d NuGet-\u043f\u0430\u043a\u0435\u0442, \u0430 .NET \u0441\u0430\u043c \u0432\u044b\u0431\u0438\u0440\u0430\u0435\u0442 \u0438 \u0437\u0430\u0433\u0440\u0443\u0436\u0430\u0435\u0442 \u043d\u0443\u0436\u043d\u044b\u0439 \u0444\u0430\u0439\u043b \u0447\u0435\u0440\u0435\u0437\u00a0DllImportResolver.\u0427\u0442\u043e \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u0441\u044f \u0432 \u0438\u0442\u043e\u0433\u0435\u041d\u0430 \u0443\u0440\u043e\u0432\u043d\u0435 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f \u043f\u0430\u043a\u0435\u0442 \u0432\u044b\u0433\u043b\u044f\u0434\u0438\u0442 \u043a\u0430\u043a \u043e\u0431\u044b\u0447\u043d\u0430\u044f .NET-\u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0430:string encrypted = Encryptor.Encrypt(&#171;hello&#187;, &#171;key-123&#187;);string decrypted = Encryptor.Decrypt(encrypted, &#171;key-123&#187;);\u0410 \u0432\u043d\u0443\u0442\u0440\u0438 \u043f\u0440\u043e\u0438\u0441\u0445\u043e\u0434\u0438\u0442 \u0432\u043e\u0442 \u044d\u0442\u043e:+&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;+        dotnet add package        +&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;+| Consumer .NET app | &#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;-&gt; | NuGet package         |+&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;+                                  +&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;+        |                                                        |        | DllImport(&#171;ted_encryption&#187;)                            |        v                                                        v+&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;+                                  +&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;+| Managed wrapper  |                                  | Embedded native files || C# \/ .NET 8      |                                  | .dll \/ .so \/ .dylib   |+&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;+                                  +&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;+        |                                                        |        | P\/Invoke                                               |        v                                                        v+&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;+| Rust cdylib: extern &#171;C&#187; functions, C-compatible ABI, manual memory owner |+&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;+\u041a\u043b\u044e\u0447\u0435\u0432\u0430\u044f \u043c\u044b\u0441\u043b\u044c:\u00a0P\/Invoke\u00a0\u0440\u0435\u0448\u0430\u0435\u0442 \u0432\u044b\u0437\u043e\u0432 \u0444\u0443\u043d\u043a\u0446\u0438\u0438, \u043d\u043e \u043d\u0435 \u0440\u0435\u0448\u0430\u0435\u0442 \u0434\u043e\u0441\u0442\u0430\u0432\u043a\u0443 native-\u0431\u0438\u043d\u0430\u0440\u0435\u0439. \u0414\u043e\u0441\u0442\u0430\u0432\u043a\u0443 \u0440\u0435\u0448\u0430\u0435\u0442 \u0441\u0432\u044f\u0437\u043a\u0430\u00a0EmbeddedResource\u00a0+\u00a0DllImportResolver.\u041e\u0431\u0449\u0430\u044f \u0430\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u0430\u0420\u0435\u0448\u0435\u043d\u0438\u0435 \u0441\u043e\u0441\u0442\u043e\u0438\u0442 \u0438\u0437 \u0447\u0435\u0442\u044b\u0440\u0435\u0445 \u0447\u0430\u0441\u0442\u0435\u0439:+&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;-+      +&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;+| Rust crate           |      | .NET wrapper             || crate-type = cdylib  | &#8212;&gt; | DllImport + safe facade  |+&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;-+      +&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;+            |                              |            v                              v+&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;-+      +&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;+| Native binaries      | &#8212;&gt; | EmbeddedResource         || win\/linux\/macos      |      | inside .NET assembly     |+&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;-+      +&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;+                                           |                                           v                              +&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;+                              | DllImportResolver        |                              | extract + NativeLibrary  |                              +&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;&#8212;+\u041d\u0430 runtime-\u043f\u0443\u0442\u0438 \u044d\u0442\u043e \u0432\u044b\u0433\u043b\u044f\u0434\u0438\u0442 \u0442\u0430\u043a:DllImport(&#171;ted_encryption&#187;)        |        vNativeLibrary.SetDllImportResolver        |        vDetect OS and architecture        |        vExtract embedded native binary        |        vNativeLibrary.Load(path)        |        vCall Rust function1. Rust: C-compatible ABIC# \u043d\u0435 \u043c\u043e\u0436\u0435\u0442 \u043d\u0430\u043f\u0440\u044f\u043c\u0443\u044e \u0432\u044b\u0437\u0432\u0430\u0442\u044c Rust-\u0444\u0443\u043d\u043a\u0446\u0438\u044e \u0441\u00a0String,\u00a0Result\u00a0\u0438\u043b\u0438 \u0432\u043b\u0430\u0434\u0435\u043d\u0438\u0435\u043c \u0432 \u0441\u0442\u0438\u043b\u0435 Rust. \u041d\u0430 \u0433\u0440\u0430\u043d\u0438\u0446\u0435 \u043d\u0443\u0436\u0435\u043d C-\u0441\u043e\u0432\u043c\u0435\u0441\u0442\u0438\u043c\u044b\u0439 ABI: \u043f\u0440\u0438\u043c\u0438\u0442\u0438\u0432\u044b, \u0441\u044b\u0440\u044b\u0435 \u0443\u043a\u0430\u0437\u0430\u0442\u0435\u043b\u0438 \u0438 \u044f\u0432\u043d\u043e\u0435 \u043f\u0440\u0430\u0432\u0438\u043b\u043e, \u043a\u0442\u043e \u0432\u044b\u0434\u0435\u043b\u044f\u0435\u0442 \u0438 \u043a\u0442\u043e \u043e\u0441\u0432\u043e\u0431\u043e\u0436\u0434\u0430\u0435\u0442 \u043f\u0430\u043c\u044f\u0442\u044c.\u041f\u0440\u0438\u043c\u0435\u0440:use std::ffi::{CStr, CString};use std::os::raw::c_char;#[no_mangle]pub extern &#171;C&#187; fn encrypt(input: *const c_char, key: *const c_char) -&gt; *mut c_char {    let input = unsafe { CStr::from_ptr(input) }.to_string_lossy().into_owned();    let key = unsafe { CStr::from_ptr(key) }.to_string_lossy().into_owned();    let result = match do_encrypt(&amp;input, &amp;key) {        Ok(value) =&gt; value,        Err(_) =&gt; String::new(),    };    CString::new(result).unwrap().into_raw()}#[no_mangle]pub extern &#171;C&#187; fn free_string(ptr: *mut c_char) {    if ptr.is_null() {        return;    }    unsafe {        let _ = CString::from_raw(ptr);    }}\u0417\u0434\u0435\u0441\u044c \u0432\u0430\u0436\u043d\u044b \u0434\u0432\u0430 \u043f\u0440\u0430\u0432\u0438\u043b\u0430.\u041f\u0435\u0440\u0432\u043e\u0435: \u0443 \u044d\u043a\u0441\u043f\u043e\u0440\u0442\u0438\u0440\u0443\u0435\u043c\u044b\u0445 \u0444\u0443\u043d\u043a\u0446\u0438\u0439 \u0434\u043e\u043b\u0436\u043d\u044b \u0431\u044b\u0442\u044c\u00a0#[no_mangle]\u00a0\u0438\u00a0extern &#171;C&#187;. \u0411\u0435\u0437 \u044d\u0442\u043e\u0433\u043e \u0438\u043c\u044f \u0441\u0438\u043c\u0432\u043e\u043b\u0430 \u0438\u0437\u043c\u0435\u043d\u0438\u0442\u0441\u044f, \u0438\u00a0DllImport\u00a0\u0435\u0433\u043e \u043d\u0435 \u043d\u0430\u0439\u0434\u0435\u0442.\u0412\u0442\u043e\u0440\u043e\u0435: \u043a\u0442\u043e \u0432\u044b\u0434\u0435\u043b\u0438\u043b \u043f\u0430\u043c\u044f\u0442\u044c, \u0442\u043e\u0442 \u0435\u0435 \u0438 \u043e\u0441\u0432\u043e\u0431\u043e\u0436\u0434\u0430\u0435\u0442. \u0415\u0441\u043b\u0438 Rust \u043e\u0442\u0434\u0430\u043b \u0441\u0442\u0440\u043e\u043a\u0443 \u0447\u0435\u0440\u0435\u0437\u00a0CString::into_raw, \u043e\u0441\u0432\u043e\u0431\u043e\u0436\u0434\u0430\u0442\u044c \u0435\u0435 \u0434\u043e\u043b\u0436\u0435\u043d Rust \u0447\u0435\u0440\u0435\u0437 \u043f\u0430\u0440\u043d\u0443\u044e \u0444\u0443\u043d\u043a\u0446\u0438\u044e \u0432\u0440\u043e\u0434\u0435\u00a0free_string. \u041e\u0441\u0432\u043e\u0431\u043e\u0436\u0434\u0430\u0442\u044c \u0442\u0430\u043a\u043e\u0439 \u0443\u043a\u0430\u0437\u0430\u0442\u0435\u043b\u044c \u0447\u0435\u0440\u0435\u0437\u00a0Marshal.FreeHGlobal\u00a0\u043d\u0435\u043b\u044c\u0437\u044f.Cargo.toml:[lib]crate-type = [&#171;cdylib&#187;][dependencies]aes-gcm = &#171;0.10&#187;base64 = &#171;0.22&#187;2. C#: P\/Invoke-\u043e\u0431\u0435\u0440\u0442\u043a\u0430\u041d\u0430 C#-\u0441\u0442\u043e\u0440\u043e\u043d\u0435 \u0434\u0435\u043b\u0430\u0435\u043c \u0442\u043e\u043d\u043a\u0438\u0439 native layer \u0438 \u043f\u0443\u0431\u043b\u0438\u0447\u043d\u044b\u0439 \u0431\u0435\u0437\u043e\u043f\u0430\u0441\u043d\u044b\u0439 facade:using System.Runtime.InteropServices;internal static class Native{    private const string Lib = &#171;ted_encryption&#187;;    [DllImport(Lib, EntryPoint = &#171;encrypt&#187;, CallingConvention = CallingConvention.Cdecl)]    public static extern IntPtr Encrypt(string input, string key);    [DllImport(Lib, EntryPoint = &#171;free_string&#187;, CallingConvention = CallingConvention.Cdecl)]    public static extern void FreeString(IntPtr ptr);}public static class Encryptor{    public static string Encrypt(string input, string key)    {        IntPtr ptr = Native.Encrypt(input, key);        try        {            return Marshal.PtrToStringAnsi(ptr) ?? string.Empty;        }        finally        {            Native.FreeString(ptr);        }    }}CallingConvention.Cdecl\u00a0\u043b\u0443\u0447\u0448\u0435 \u0443\u043a\u0430\u0437\u044b\u0432\u0430\u0442\u044c \u044f\u0432\u043d\u043e. Rust\u00a0extern &#171;C&#187;\u00a0\u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442 C calling convention, \u0430 \u043d\u0435\u044f\u0432\u043d\u044b\u0435 \u0437\u043d\u0430\u0447\u0435\u043d\u0438\u044f \u0432 interop-\u043a\u043e\u0434\u0435 &#8212; \u0445\u043e\u0440\u043e\u0448\u0438\u0439 \u0441\u043f\u043e\u0441\u043e\u0431 \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u044c \u0441\u0442\u0440\u0430\u043d\u043d\u043e\u0435 \u043f\u043e\u0432\u0435\u0434\u0435\u043d\u0438\u0435 \u043d\u0430 \u043e\u0434\u043d\u043e\u0439 \u041e\u0421 \u0438 &#171;\u043f\u043e\u0447\u0435\u043c\u0443-\u0442\u043e \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442&#187; \u043d\u0430 \u0434\u0440\u0443\u0433\u043e\u0439.\u0415\u0449\u0435 \u043e\u0434\u0438\u043d \u043d\u044e\u0430\u043d\u0441:\u00a0Marshal.PtrToStringAnsi\u00a0\u043d\u0435 \u0440\u0430\u0432\u0435\u043d \u0443\u043d\u0438\u0432\u0435\u0440\u0441\u0430\u043b\u044c\u043d\u043e\u043c\u0443 UTF-8-\u0440\u0435\u0448\u0435\u043d\u0438\u044e. \u0415\u0441\u043b\u0438 \u0447\u0435\u0440\u0435\u0437 \u0433\u0440\u0430\u043d\u0438\u0446\u0443 \u0434\u043e\u043b\u0436\u043d\u044b \u0441\u0442\u0430\u0431\u0438\u043b\u044c\u043d\u043e \u0445\u043e\u0434\u0438\u0442\u044c Unicode-\u0441\u0442\u0440\u043e\u043a\u0438, \u043b\u0443\u0447\u0448\u0435 \u044f\u0432\u043d\u043e \u0434\u043e\u0433\u043e\u0432\u043e\u0440\u0438\u0442\u044c\u0441\u044f \u043e UTF-8 \u0438 \u043f\u0435\u0440\u0435\u0434\u0430\u0432\u0430\u0442\u044c \u0431\u0430\u0439\u0442\u044b \u043b\u0438\u0431\u043e \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u0441\u043e\u043e\u0442\u0432\u0435\u0442\u0441\u0442\u0432\u0443\u044e\u0449\u0438\u0439 marshaling. \u0412 \u043b\u044e\u0431\u043e\u043c \u0441\u043b\u0443\u0447\u0430\u0435 Unicode \u0434\u043e\u043b\u0436\u0435\u043d \u0431\u044b\u0442\u044c \u0432 \u0442\u0435\u0441\u0442\u0430\u0445.3. \u0423\u043f\u0430\u043a\u043e\u0432\u043a\u0430 native-\u0431\u0438\u043d\u0430\u0440\u0435\u0439 \u0432 assembly\u0422\u0435\u043f\u0435\u0440\u044c \u043e\u0441\u043d\u043e\u0432\u043d\u0430\u044f \u0447\u0430\u0441\u0442\u044c: \u0434\u043e\u0441\u0442\u0430\u0432\u043a\u0430.\u0412\u043c\u0435\u0441\u0442\u043e \u0442\u043e\u0433\u043e \u0447\u0442\u043e\u0431\u044b \u043f\u0440\u043e\u0441\u0438\u0442\u044c \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f \u043f\u0430\u043a\u0435\u0442\u0430 \u0432\u0440\u0443\u0447\u043d\u0443\u044e \u0440\u0430\u0441\u043a\u043b\u0430\u0434\u044b\u0432\u0430\u0442\u044c\u00a0.dll,\u00a0.so\u00a0\u0438\u00a0.dylib, \u0434\u043e\u0431\u0430\u0432\u0438\u043c \u0438\u0445 \u0432 .NET-\u0441\u0431\u043e\u0440\u043a\u0443 \u043a\u0430\u043a\u00a0EmbeddedResource.\u041f\u0440\u0438\u043c\u0435\u0440\u00a0.csproj:&lt;ItemGroup&gt;  &lt;EmbeddedResource Include=&#187;native\/win-x64\/ted_encryption.dll&#187;                    LogicalName=&#187;ted_encryption.dll&#187; \/&gt;  &lt;EmbeddedResource Include=&#187;native\/linux-x64\/libted_encryption.so&#187;                    LogicalName=&#187;libted_encryption.so&#187; \/&gt;  &lt;EmbeddedResource Include=&#187;native\/osx-x64\/libted_encryption.dylib&#187;                    LogicalName=&#187;libted_encryption.dylib&#187; \/&gt;&lt;\/ItemGroup&gt;\u041f\u043e\u043b\u0443\u0447\u0430\u0435\u0442\u0441\u044f \u0442\u0430\u043a\u0430\u044f \u0443\u043f\u0430\u043a\u043e\u0432\u043a\u0430:Ted.Encryption.dll|+&#8212; Managed C# wrapper|+&#8212; Embedded resources    |    +&#8212; ted_encryption.dll    +&#8212; libted_encryption.so    +&#8212; libted_encryption.dylib\u041f\u043e\u0442\u0440\u0435\u0431\u0438\u0442\u0435\u043b\u044c \u0432\u0438\u0434\u0438\u0442 \u043e\u0431\u044b\u0447\u043d\u044b\u0439 NuGet-\u043f\u0430\u043a\u0435\u0442. Native-\u0444\u0430\u0439\u043b\u044b \u043b\u0435\u0436\u0430\u0442 \u0432\u043d\u0443\u0442\u0440\u0438 \u0441\u0431\u043e\u0440\u043a\u0438 \u0438 \u0434\u043e\u0441\u0442\u0430\u044e\u0442\u0441\u044f \u0442\u043e\u043b\u044c\u043a\u043e \u0432 \u043c\u043e\u043c\u0435\u043d\u0442 \u0437\u0430\u0433\u0440\u0443\u0437\u043a\u0438 \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0438.4. DllImportResolver: \u0433\u043b\u0430\u0432\u043d\u044b\u0439 \u0442\u0440\u044e\u043a\u0412 .NET Core 3.0+ \u0435\u0441\u0442\u044c \u043c\u0435\u0445\u0430\u043d\u0438\u0437\u043c\u00a0NativeLibrary.SetDllImportResolver. \u041e\u043d \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u0435\u0442 \u043f\u0435\u0440\u0435\u0445\u0432\u0430\u0442\u0438\u0442\u044c \u043f\u043e\u043f\u044b\u0442\u043a\u0443 \u0437\u0430\u0433\u0440\u0443\u0437\u0438\u0442\u044c native-\u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0443 \u0438 \u0441\u0430\u043c\u043e\u043c\u0443 \u0440\u0435\u0448\u0438\u0442\u044c, \u043e\u0442\u043a\u0443\u0434\u0430 \u0435\u0435 \u0431\u0440\u0430\u0442\u044c.\u0420\u0435\u0433\u0438\u0441\u0442\u0440\u0438\u0440\u0443\u0435\u043c resolver \u043e\u0434\u0438\u043d \u0440\u0430\u0437:using System.Reflection;using System.Runtime.InteropServices;internal static class Native{    private const string Lib = &#171;ted_encryption&#187;;    static Native()    {        NativeLibrary.SetDllImportResolver(typeof(Native).Assembly, Resolve);    }    private static IntPtr Resolve(        string libraryName,        Assembly assembly,        DllImportSearchPath? searchPath)    {        if (libraryName != Lib)        {            return IntPtr.Zero;        }        string path = ExtractNativeLibrary(assembly);        return NativeLibrary.Load(path);    }}\u0422\u0435\u043f\u0435\u0440\u044c \u0432\u044b\u0431\u0438\u0440\u0430\u0435\u043c \u0440\u0435\u0441\u0443\u0440\u0441 \u043f\u043e \u043f\u043b\u0430\u0442\u0444\u043e\u0440\u043c\u0435 \u0438 \u0440\u0430\u0441\u043f\u0430\u043a\u043e\u0432\u044b\u0432\u0430\u0435\u043c \u0435\u0433\u043e \u0432\u043e \u0432\u0440\u0435\u043c\u0435\u043d\u043d\u0443\u044e \u0434\u0438\u0440\u0435\u043a\u0442\u043e\u0440\u0438\u044e:private static string ExtractNativeLibrary(Assembly assembly){    string resourceName =        RuntimeInformation.IsOSPlatform(OSPlatform.Windows) ? &#171;ted_encryption.dll&#187; :        RuntimeInformation.IsOSPlatform(OSPlatform.Linux) ? &#171;libted_encryption.so&#187; :        RuntimeInformation.IsOSPlatform(OSPlatform.OSX) ? &#171;libted_encryption.dylib&#187; :        throw new PlatformNotSupportedException();    string directory = Path.Combine(Path.GetTempPath(), &#171;ted_encryption&#187;);    Directory.CreateDirectory(directory);    string targetPath = Path.Combine(directory, resourceName);    if (!File.Exists(targetPath))    {        using Stream source = assembly.GetManifestResourceStream(resourceName)            ?? throw new InvalidOperationException($&#187;Resource {resourceName} not found&#187;);        using FileStream target = File.Create(targetPath);        source.CopyTo(target);    }    return targetPath;}\u041f\u043e\u0441\u043b\u0435 \u044d\u0442\u043e\u0433\u043e\u00a0DllImport(&#171;ted_encryption&#187;)\u00a0\u043d\u0435 \u043f\u044b\u0442\u0430\u0435\u0442\u0441\u044f \u0438\u0441\u043a\u0430\u0442\u044c \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0443 \u0440\u044f\u0434\u043e\u043c \u0441 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435\u043c. \u0412\u043c\u0435\u0441\u0442\u043e \u044d\u0442\u043e\u0433\u043e:.NET \u0432\u0438\u0434\u0438\u0442 DllImport(\u201cted_encryption\u201d)\u0412\u044b\u0437\u044b\u0432\u0430\u0435\u0442 \u0437\u0430\u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u044b\u0439 resolverResolver \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u044f\u0435\u0442 \u041e\u0421\u0414\u043e\u0441\u0442\u0430\u0435\u0442 \u043d\u0443\u0436\u043d\u044b\u0439 EmbeddedResource\u0421\u043e\u0445\u0440\u0430\u043d\u044f\u0435\u0442 \u0435\u0433\u043e \u0432\u043e \u0432\u0440\u0435\u043c\u0435\u043d\u043d\u0443\u044e \u043f\u0430\u043f\u043a\u0443\u0417\u0430\u0433\u0440\u0443\u0436\u0430\u0435\u0442 \u0447\u0435\u0440\u0435\u0437 NativeLibrary.LoadP\/Invoke \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u0442 Rust-\u0444\u0443\u043d\u043a\u0446\u0438\u044e\u0414\u043b\u044f \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f \u043f\u0430\u043a\u0435\u0442\u0430 \u0432\u0441\u0435 \u044d\u0442\u043e \u043f\u0440\u043e\u0437\u0440\u0430\u0447\u043d\u043e.5. \u0410\u0432\u0442\u043e\u043c\u0430\u0442\u0438\u0437\u0430\u0446\u0438\u044f \u0441\u0431\u043e\u0440\u043a\u0438 Rust-\u0431\u0438\u043d\u0430\u0440\u0435\u0439\u0427\u0442\u043e\u0431\u044b \u0432 NuGet \u043d\u0435 \u043f\u043e\u043f\u0430\u0434\u0430\u043b\u0438 \u0443\u0441\u0442\u0430\u0440\u0435\u0432\u0448\u0438\u0435 native-\u0444\u0430\u0439\u043b\u044b, \u0441\u0431\u043e\u0440\u043a\u0443 Rust \u043c\u043e\u0436\u043d\u043e \u043f\u0440\u0438\u0432\u044f\u0437\u0430\u0442\u044c \u043a\u00a0dotnet pack\u00a0\u0438\u043b\u0438\u00a0Release-\u0441\u0431\u043e\u0440\u043a\u0435.\u041d\u0430\u043f\u0440\u0438\u043c\u0435\u0440, \u0447\u0435\u0440\u0435\u0437 MSBuild target:&lt;Target Name=&#187;BuildRust&#187; BeforeTargets=&#187;Build&#187;&#8230;<\/p>\n","protected":false},"author":1,"featured_media":0,"comment_status":"closed","ping_status":"closed","sticky":false,"template":"","format":"standard","meta":{"footnotes":""},"categories":[],"tags":[],"class_list":["post-482276","post","type-post","status-publish","format-standard","hentry"],"_links":{"self":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/482276","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcomments&post=482276"}],"version-history":[{"count":0,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/482276\/revisions"}],"wp:attachment":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fmedia&parent=482276"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcategories&post=482276"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Ftags&post=482276"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}