Edge transformers is a Rust implementation of Huggingface's pipelines based on ONNX Runtime backend.
| Model export feature/task | Class name |
|---------------------------|-----------------------------------------------------|
| causal-lm | ConditionalGenerationPipeline |
| causal-lm-with-past | ConditionalGenerationPipelineWithPKVs |
| default | EmbeddingPipeline |
| seq2seq-lm | Seq2SeqGenerationPipeline or OptimumSeq2SeqPipeline |
| seq2seq-lm-with-past | OptimumSeq2SeqPipelineWithPKVs |
| sequence-classification | SequenceClassificationPipeline |
| token-classification | TokenClassificationPipeline |
Your linker must be able to find onnxruntime.dll and edge-transformers.dll (or *.so on Linux).
You can find C and C# wrappers in c
and csharp
folders respectively.
Documentation is WIP, refer to Rust documentation for now.
```csharp using EdgeTransformers;
... var env = EnvContainer.New();
var conditionalGen = ConditionalGenerationPipelineFFI.FromPretrained(
env.Context, "optimum/gpt2", DeviceFFI.CPU, GraphOptimizationLevelFFI.Level3
);
var outp = conditionalGen.GenerateTopkSampling("Hello", 2, 50, 0.9f);
Assert.IsNotNull(outp);
... ```
Batch processing is supported, but is a bit unintuitive and requires StringBatch class.
```csharp using EdgeTransformers; ... var env = EnvContainer.New(); var condPipelinePkv = ConditionalGenerationPipelineFFI.FromPretrained( env.Context, "optimum/gpt2", DeviceFFI.DML, GraphOptimizationLevelFFI.All); var stringbatch = StringBatch.New(); stringbatch.Add("Hello world"); string_batch.Add("Hello world");
var o_batch_pkv = condPipelinePkv.GenerateRandomSamplingBatch(string_batch.Context, 10, 0.5f);
Debug.LogFormat("Cond generation output 0: {0} 1: {1}", o_batch_pkv[0].ascii_string, o_batch_pkv[1].ascii_string);
... ```
TODO
```csharp use std::fs; use onnxruntime::environment::Environment; use onnxruntime::{GraphOptimizationLevel, LoggingLevel}; use edge_transformers::{ConditionalGenerationPipelineWithPKVs, TopKSampler, Device};
let environment = Environment::builder() .withname("test") .withlog_level(LoggingLevel::Verbose) .build() .unwrap();
let sampler = TopKSampler::new(50, 0.9); let pipeline = ConditionalGenerationPipelineWithPKVs::frompretrained( &environment, "optimum/gpt2".tostring(), Device::CPU, GraphOptimizationLevel::All, ).unwrap();
let input = "This is a test";
println!("{}", pipeline.generate(input, 10, &sampler).unwrap()); ```
Please refer to ONNX Runtime bindings docs on detailed how to.
Tests require to be running on a single thread at least for the first time. The reason is that they use *::from_pretrained function that downloads data from Huggingface Hub and some tests rely on the same files being downloaded. Second time they can run in parallel because they use cached files.
e.g. First time command:
bash
cargo test -- --test-threads=1
e.g. Second time:
bash
cargo test