The namegen5 website.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

126 lines
3.8 KiB

3 years ago
  1. #[macro_use]
  2. extern crate serde;
  3. use namegen::{Name, SampleSet, NamePart, FormattingRule};
  4. use std::error::Error;
  5. use wasm_bindgen::prelude::*;
  6. #[cfg(feature = "wee_alloc")]
  7. #[global_allocator]
  8. static ALLOC: wee_alloc::WeeAlloc = wee_alloc::WeeAlloc::INIT;
  9. #[wasm_bindgen]
  10. pub struct WasmNameGenerator {
  11. name: Name,
  12. }
  13. #[derive(Deserialize)]
  14. #[serde(rename_all = "camelCase")]
  15. pub struct AddPartOptions {
  16. name: String,
  17. kind: String,
  18. format_rules: Vec<FormattingRule>,
  19. #[serde(default)]
  20. initial_tokens: Vec<String>,
  21. #[serde(default)]
  22. initial_subtokens: Vec<String>,
  23. #[serde(default)]
  24. rlf: bool,
  25. #[serde(default)]
  26. ral: bool,
  27. #[serde(default)]
  28. lrs: bool,
  29. #[serde(default)]
  30. lrm: bool,
  31. #[serde(default)]
  32. lre: bool,
  33. }
  34. #[wasm_bindgen]
  35. impl WasmNameGenerator {
  36. pub fn generate_one(&self, format_name: &str, seed: Option<u64>) -> Option<String> {
  37. match seed {
  38. Some(seed) => self.name.generate_seeded(seed, format_name)?.next(),
  39. None => self.name.generate(format_name)?.next(),
  40. }
  41. }
  42. pub fn generate_many(&self, format_name: &str, amount: usize, seed: Option<u64>) -> JsValue {
  43. let gen = match seed {
  44. Some(seed) => self.name.generate_seeded(seed, format_name),
  45. None => self.name.generate(format_name),
  46. };
  47. match gen {
  48. Some(gen) => {
  49. let strings: Vec<String> = gen.take(amount).collect();
  50. JsValue::from_serde(&strings).unwrap()
  51. }
  52. None => JsValue::null(),
  53. }
  54. }
  55. pub fn add_part(&mut self, options: JsValue) -> Result<(), JsValue> {
  56. let mut options: AddPartOptions = match options.into_serde() {
  57. Ok(options) => options,
  58. Err(err) => {
  59. return Err(JsValue::from(format!("{}", err)));
  60. }
  61. };
  62. match options.kind.as_str() {
  63. "markov" => {
  64. self.name.add_part(NamePart::new_markov(
  65. &options.name, &options.format_rules,
  66. options.initial_tokens.as_slice(),
  67. options.lrs, options.lrm, options.lre, options.rlf,
  68. ));
  69. },
  70. "cfgrammar" => {
  71. self.name.add_part(NamePart::new_cfgrammar(
  72. &options.name, &options.format_rules,
  73. options.initial_subtokens.as_slice(),
  74. options.rlf, options.ral,
  75. ));
  76. },
  77. "wordlist" => {
  78. self.name.add_part(NamePart::new_wordlist(&options.name, &options.format_rules));
  79. },
  80. _ => {
  81. return Err(JsValue::from("AddPartError: Invalid value for options.kind (outdated wasm?)"));
  82. }
  83. }
  84. Ok(())
  85. }
  86. pub fn add_format(&mut self, format_name: &str, format_str: &str) {
  87. self.name.add_format(format_name, format_str)
  88. }
  89. pub fn learn(&mut self, part_name: &str, sample_set: JsValue) -> Result<(), JsValue> {
  90. console_error_panic_hook::set_once();
  91. let sample_set: SampleSet = match sample_set.into_serde() {
  92. Ok(ss) => ss,
  93. Err(e) => { return Err(JsValue::from(format!("{}", e))); },
  94. };
  95. self.name.learn(part_name, &sample_set).map_err(|e| JsValue::from(format!("{}", e)))
  96. }
  97. pub fn data(&self) -> JsValue {
  98. JsValue::from_serde(&self.name).unwrap()
  99. }
  100. pub fn new() -> WasmNameGenerator {
  101. WasmNameGenerator { name: Name::new() }
  102. }
  103. pub fn load(data: &str) -> Result<WasmNameGenerator, JsValue> {
  104. let name = serde_json::from_slice(data.as_bytes());
  105. return name
  106. .map(|n| WasmNameGenerator { name: n })
  107. .map_err(|e| JsValue::from(format!("{:?}", e)));
  108. }
  109. }