Vigenère cipher in Ruby Announcing the arrival of Valued Associate #679: Cesar Manara Unicorn Meta Zoo #1: Why another podcast?Encrypter - Double Vigenere Cipher in PythonVigenère cipher in CVigenère cipher 2Vigenère Cipher in HaskellSimple Caesar cipher cryptanalysis in RubyVigenere cipher with random saltVigenère cipher in PythonBrute-force Vigenere Cipher using multiple threadsVigenere Cipher in CCracking Vigenere and Caesar Ciphered Text in Python

What is it called when you ride around on your front wheel?

Does Feeblemind produce an ongoing magical effect that can be dispelled?

c++ diamond problem - How to call base method only once

Could Neutrino technically as side-effect, incentivize centralization of the bitcoin network?

How to avoid introduction cliches

Retract an already submitted recommendation letter (written for an undergrad student)

Is there any hidden 'W' sound after 'comment' in : Comment est-elle?

Expansion//Explosion and Siren Stormtamer

Mistake in years of experience in resume?

What is the best way to deal with NPC-NPC combat?

As an international instructor, should I openly talk about my accent?

How to not starve gigantic beasts

A faster way to compute the largest prime factor

A strange hotel

Israeli soda type drink

How to open locks without disable device?

What is the term for a person whose job is to place products on shelves in stores?

Map material from china not allowed to leave the country

"My boss was furious with me and I have been fired" vs. "My boss was furious with me and I was fired"

How to use @AuraEnabled base class method in Lightning Component?

Are all CP/M-80 implementations binary compatible?

Married in secret, can marital status in passport be changed at a later date?

Does Mathematica have an implementation of the Poisson binomial distribution?

How would I use different systems of magic when they are capable of the same effects?



Vigenère cipher in Ruby



Announcing the arrival of Valued Associate #679: Cesar Manara
Unicorn Meta Zoo #1: Why another podcast?Encrypter - Double Vigenere Cipher in PythonVigenère cipher in CVigenère cipher 2Vigenère Cipher in HaskellSimple Caesar cipher cryptanalysis in RubyVigenere cipher with random saltVigenère cipher in PythonBrute-force Vigenere Cipher using multiple threadsVigenere Cipher in CCracking Vigenere and Caesar Ciphered Text in Python



.everyoneloves__top-leaderboard:empty,.everyoneloves__mid-leaderboard:empty,.everyoneloves__bot-mid-leaderboard:empty margin-bottom:0;








3












$begingroup$


What I'm trying to do: implement the Vigenère cipher in Ruby. I already have a working version, but I want to make sure it is efficient and well-designed.



module Crypto
# Vigenère cipher encryption and decryption abstraction
module Vigenere
LETTERS = ('a'..'z').to_a.freeze
private_constant :LETTERS

module_function

# Encrypts a string
#
# @param string [String] the string that will be encrypted
# @param key [String] the key that will be used to encrypt the string
#
# @return [String] the encrypted string
def encrypt(string:, key:)
key = make_key(length: string.length, key: key)

string.length.times.map i.join
end

# Decrypts an encrypted string
#
# @param string [String] the encrypted string that will be decrypted
# @param key [String] the key that will be used to decrypt the string
#
# @return [String] the decrypted string
def decrypt(string:, key:)
key = make_key(length: string.length, key: key)

string.length.times.map .join
end

# Repeats a word until it matches a certain length
#
# @param length [Integer] the length of the word being encrypted/decrypted
# @param key [String] the word that will be repeated
#
# @return [String] the word in its new form
def make_key(length:, key:)
i = 0
length.times do
i = 0 if i == key.length
break if key.length == length

key << key[i]
i += 1
end

key
end

private_class_method :make_key
end
end


I do have some specific questions:



1. private_class_method



This is the best way I found to define a private method in a module, but it feels weird to me. Isn't there a better way to do that? My first implementation was this:



module Crypto
class Vigenere
class << self
def encrypt # ...
def decrypt # ...

private

def make_key # ...
end
end
end


which was fine for me. But then I read this rule on the Ruby Style Guide repository. So I switched to using module, but it doesn't feel right to use private methods in this structure. Am I wrong?



2. reseting a counter (index)



Take a look at this snippet of code:



def make_key(length:, key:)
i = 0
length.times do
i = 0 if i == key.length
break if key.length == length

key << key[i]
i += 1
end

key
end


Defining a counter (i) and manually incrementing it... looks awkward, doesn't it (at least in the Ruby world)? Is there a better way to do this?



3. valid multi-line block with curly braces?



Now take a look at this snippet:



string.length.times.map .join


I know most Ruby developers tend to use curly braces only for one-line blocks and do-end for multi-line blocks, but this time it seems okay using curly braces with a multi-line block, since I'm chaining #join right after. What would you do:



1. use do-end, store it in a variable and invoke #join after that



new_letters = string.length.times.map do |i|
p = LETTERS.find_index(string[i])
k = LETTERS.find_index(key[i])

LETTERS[(p + k) % 26]
end

new_letters.join


2. what I did (use curly braces even with multi-line block and chain #join)



And of course, if you have any other observations, please share.










share|improve this question







New contributor




sPaGhEtTiCaSe is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.







$endgroup$







  • 1




    $begingroup$
    If you take the time to spell Vigenère correctly then that's enough for me to tell that the code is going to be all right :)
    $endgroup$
    – Maarten Bodewes
    4 hours ago


















3












$begingroup$


What I'm trying to do: implement the Vigenère cipher in Ruby. I already have a working version, but I want to make sure it is efficient and well-designed.



module Crypto
# Vigenère cipher encryption and decryption abstraction
module Vigenere
LETTERS = ('a'..'z').to_a.freeze
private_constant :LETTERS

module_function

# Encrypts a string
#
# @param string [String] the string that will be encrypted
# @param key [String] the key that will be used to encrypt the string
#
# @return [String] the encrypted string
def encrypt(string:, key:)
key = make_key(length: string.length, key: key)

string.length.times.map i.join
end

# Decrypts an encrypted string
#
# @param string [String] the encrypted string that will be decrypted
# @param key [String] the key that will be used to decrypt the string
#
# @return [String] the decrypted string
def decrypt(string:, key:)
key = make_key(length: string.length, key: key)

string.length.times.map .join
end

# Repeats a word until it matches a certain length
#
# @param length [Integer] the length of the word being encrypted/decrypted
# @param key [String] the word that will be repeated
#
# @return [String] the word in its new form
def make_key(length:, key:)
i = 0
length.times do
i = 0 if i == key.length
break if key.length == length

key << key[i]
i += 1
end

key
end

private_class_method :make_key
end
end


I do have some specific questions:



1. private_class_method



This is the best way I found to define a private method in a module, but it feels weird to me. Isn't there a better way to do that? My first implementation was this:



module Crypto
class Vigenere
class << self
def encrypt # ...
def decrypt # ...

private

def make_key # ...
end
end
end


which was fine for me. But then I read this rule on the Ruby Style Guide repository. So I switched to using module, but it doesn't feel right to use private methods in this structure. Am I wrong?



2. reseting a counter (index)



Take a look at this snippet of code:



def make_key(length:, key:)
i = 0
length.times do
i = 0 if i == key.length
break if key.length == length

key << key[i]
i += 1
end

key
end


Defining a counter (i) and manually incrementing it... looks awkward, doesn't it (at least in the Ruby world)? Is there a better way to do this?



3. valid multi-line block with curly braces?



Now take a look at this snippet:



string.length.times.map .join


I know most Ruby developers tend to use curly braces only for one-line blocks and do-end for multi-line blocks, but this time it seems okay using curly braces with a multi-line block, since I'm chaining #join right after. What would you do:



1. use do-end, store it in a variable and invoke #join after that



new_letters = string.length.times.map do |i|
p = LETTERS.find_index(string[i])
k = LETTERS.find_index(key[i])

LETTERS[(p + k) % 26]
end

new_letters.join


2. what I did (use curly braces even with multi-line block and chain #join)



And of course, if you have any other observations, please share.










share|improve this question







New contributor




sPaGhEtTiCaSe is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.







$endgroup$







  • 1




    $begingroup$
    If you take the time to spell Vigenère correctly then that's enough for me to tell that the code is going to be all right :)
    $endgroup$
    – Maarten Bodewes
    4 hours ago














3












3








3





$begingroup$


What I'm trying to do: implement the Vigenère cipher in Ruby. I already have a working version, but I want to make sure it is efficient and well-designed.



module Crypto
# Vigenère cipher encryption and decryption abstraction
module Vigenere
LETTERS = ('a'..'z').to_a.freeze
private_constant :LETTERS

module_function

# Encrypts a string
#
# @param string [String] the string that will be encrypted
# @param key [String] the key that will be used to encrypt the string
#
# @return [String] the encrypted string
def encrypt(string:, key:)
key = make_key(length: string.length, key: key)

string.length.times.map i.join
end

# Decrypts an encrypted string
#
# @param string [String] the encrypted string that will be decrypted
# @param key [String] the key that will be used to decrypt the string
#
# @return [String] the decrypted string
def decrypt(string:, key:)
key = make_key(length: string.length, key: key)

string.length.times.map .join
end

# Repeats a word until it matches a certain length
#
# @param length [Integer] the length of the word being encrypted/decrypted
# @param key [String] the word that will be repeated
#
# @return [String] the word in its new form
def make_key(length:, key:)
i = 0
length.times do
i = 0 if i == key.length
break if key.length == length

key << key[i]
i += 1
end

key
end

private_class_method :make_key
end
end


I do have some specific questions:



1. private_class_method



This is the best way I found to define a private method in a module, but it feels weird to me. Isn't there a better way to do that? My first implementation was this:



module Crypto
class Vigenere
class << self
def encrypt # ...
def decrypt # ...

private

def make_key # ...
end
end
end


which was fine for me. But then I read this rule on the Ruby Style Guide repository. So I switched to using module, but it doesn't feel right to use private methods in this structure. Am I wrong?



2. reseting a counter (index)



Take a look at this snippet of code:



def make_key(length:, key:)
i = 0
length.times do
i = 0 if i == key.length
break if key.length == length

key << key[i]
i += 1
end

key
end


Defining a counter (i) and manually incrementing it... looks awkward, doesn't it (at least in the Ruby world)? Is there a better way to do this?



3. valid multi-line block with curly braces?



Now take a look at this snippet:



string.length.times.map .join


I know most Ruby developers tend to use curly braces only for one-line blocks and do-end for multi-line blocks, but this time it seems okay using curly braces with a multi-line block, since I'm chaining #join right after. What would you do:



1. use do-end, store it in a variable and invoke #join after that



new_letters = string.length.times.map do |i|
p = LETTERS.find_index(string[i])
k = LETTERS.find_index(key[i])

LETTERS[(p + k) % 26]
end

new_letters.join


2. what I did (use curly braces even with multi-line block and chain #join)



And of course, if you have any other observations, please share.










share|improve this question







New contributor




sPaGhEtTiCaSe is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.







$endgroup$




What I'm trying to do: implement the Vigenère cipher in Ruby. I already have a working version, but I want to make sure it is efficient and well-designed.



module Crypto
# Vigenère cipher encryption and decryption abstraction
module Vigenere
LETTERS = ('a'..'z').to_a.freeze
private_constant :LETTERS

module_function

# Encrypts a string
#
# @param string [String] the string that will be encrypted
# @param key [String] the key that will be used to encrypt the string
#
# @return [String] the encrypted string
def encrypt(string:, key:)
key = make_key(length: string.length, key: key)

string.length.times.map i.join
end

# Decrypts an encrypted string
#
# @param string [String] the encrypted string that will be decrypted
# @param key [String] the key that will be used to decrypt the string
#
# @return [String] the decrypted string
def decrypt(string:, key:)
key = make_key(length: string.length, key: key)

string.length.times.map .join
end

# Repeats a word until it matches a certain length
#
# @param length [Integer] the length of the word being encrypted/decrypted
# @param key [String] the word that will be repeated
#
# @return [String] the word in its new form
def make_key(length:, key:)
i = 0
length.times do
i = 0 if i == key.length
break if key.length == length

key << key[i]
i += 1
end

key
end

private_class_method :make_key
end
end


I do have some specific questions:



1. private_class_method



This is the best way I found to define a private method in a module, but it feels weird to me. Isn't there a better way to do that? My first implementation was this:



module Crypto
class Vigenere
class << self
def encrypt # ...
def decrypt # ...

private

def make_key # ...
end
end
end


which was fine for me. But then I read this rule on the Ruby Style Guide repository. So I switched to using module, but it doesn't feel right to use private methods in this structure. Am I wrong?



2. reseting a counter (index)



Take a look at this snippet of code:



def make_key(length:, key:)
i = 0
length.times do
i = 0 if i == key.length
break if key.length == length

key << key[i]
i += 1
end

key
end


Defining a counter (i) and manually incrementing it... looks awkward, doesn't it (at least in the Ruby world)? Is there a better way to do this?



3. valid multi-line block with curly braces?



Now take a look at this snippet:



string.length.times.map .join


I know most Ruby developers tend to use curly braces only for one-line blocks and do-end for multi-line blocks, but this time it seems okay using curly braces with a multi-line block, since I'm chaining #join right after. What would you do:



1. use do-end, store it in a variable and invoke #join after that



new_letters = string.length.times.map do |i|
p = LETTERS.find_index(string[i])
k = LETTERS.find_index(key[i])

LETTERS[(p + k) % 26]
end

new_letters.join


2. what I did (use curly braces even with multi-line block and chain #join)



And of course, if you have any other observations, please share.







ruby vigenere-cipher






share|improve this question







New contributor




sPaGhEtTiCaSe is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.











share|improve this question







New contributor




sPaGhEtTiCaSe is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.









share|improve this question




share|improve this question






New contributor




sPaGhEtTiCaSe is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.









asked 5 hours ago









sPaGhEtTiCaSesPaGhEtTiCaSe

183




183




New contributor




sPaGhEtTiCaSe is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.





New contributor





sPaGhEtTiCaSe is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.






sPaGhEtTiCaSe is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.







  • 1




    $begingroup$
    If you take the time to spell Vigenère correctly then that's enough for me to tell that the code is going to be all right :)
    $endgroup$
    – Maarten Bodewes
    4 hours ago













  • 1




    $begingroup$
    If you take the time to spell Vigenère correctly then that's enough for me to tell that the code is going to be all right :)
    $endgroup$
    – Maarten Bodewes
    4 hours ago








1




1




$begingroup$
If you take the time to spell Vigenère correctly then that's enough for me to tell that the code is going to be all right :)
$endgroup$
– Maarten Bodewes
4 hours ago





$begingroup$
If you take the time to spell Vigenère correctly then that's enough for me to tell that the code is going to be all right :)
$endgroup$
– Maarten Bodewes
4 hours ago











2 Answers
2






active

oldest

votes


















1












$begingroup$

The code is rather clear, so consider the following to be nitpicks.




I'd say that you are generating a new key stream from the key. I'd certainly not reuse the key variable.




The first i = 0 before the loop seems spurious.



Using i as a counter is well understood, and I'd not worry overly much on the style of it. You are probably the only one who cares if it is really Ruby-esk; developers down the line will understand it.



What I wonder though is that you run your loop length times, but there is a break that seems to trigger before that. That's not all too clear to me.



I wonder what happens if you supply it an "empty" key string. Some guard statements may be in order.




Same for the curly braces. It's clear as it is, choose whatever you want. Personally I slightly favor the braces.




You could consider creating a mod function, however since % is already the modulus, which will never return a negative value if the right value is positive, it seems to me that removing the + 26 is probably the only thing you need to change (during decryption).



Instead of using 26 as unexplained magic value, you should get the size of the LETTERS range instead. That way you can also expand your ciphertext later.




I've got no opinion on the private_class_method as I'm not a Ruby developer (I'm specialized in knowing many languages / constructs and of course applied crypto).






share|improve this answer











$endgroup$












  • $begingroup$
    Thanks, that's really helpful! Sadly, I don't have enough reputation to upvote your answer. :(
    $endgroup$
    – sPaGhEtTiCaSe
    2 hours ago


















1












$begingroup$

Looping using string.length.times.map … and length.times do … is OK, but slightly on the awkward side. I recommend adhering to the convention of writing same-line blocks using and line-spanning blocks using do … end.



To extend the key, you can use the string multiplication operator. (Note that extending the key longer than necessary doesn't do much harm.)



You can also factor out more of the commonality between the encrypt and decrypt functions.



Instead of searching the LETTERS array, I recommend performing arithmetic on ASCII codes.



module Crypto
module Vigenere
module_function
def encrypt(plaintext, key)
vigenere(plaintext, key) p, k
end

def decrypt(ciphertext, key)
vigenere(ciphertext, key)
end

# Implementation of Vigenere cipher. The combiner block accepts
# one character from the text and the corresponding character from
# the key (encoded as a=0, b=1, ..., z=25), and returns the
# result using the same numerical scheme.
def vigenere(text, key, &combiner)
a = 'a'.ord
ext_key = key * (text.length / key.length + 1)
text.chars.zip(ext_key.chars).collect do |t, k|
(a + combiner.call(t.ord - a, k.ord - a)).chr
end.join
end
private_class_method :vigenere
end
end





share|improve this answer











$endgroup$













    Your Answer






    StackExchange.ifUsing("editor", function ()
    StackExchange.using("externalEditor", function ()
    StackExchange.using("snippets", function ()
    StackExchange.snippets.init();
    );
    );
    , "code-snippets");

    StackExchange.ready(function()
    var channelOptions =
    tags: "".split(" "),
    id: "196"
    ;
    initTagRenderer("".split(" "), "".split(" "), channelOptions);

    StackExchange.using("externalEditor", function()
    // Have to fire editor after snippets, if snippets enabled
    if (StackExchange.settings.snippets.snippetsEnabled)
    StackExchange.using("snippets", function()
    createEditor();
    );

    else
    createEditor();

    );

    function createEditor()
    StackExchange.prepareEditor(
    heartbeatType: 'answer',
    autoActivateHeartbeat: false,
    convertImagesToLinks: false,
    noModals: true,
    showLowRepImageUploadWarning: true,
    reputationToPostImages: null,
    bindNavPrevention: true,
    postfix: "",
    imageUploader:
    brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
    contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
    allowUrls: true
    ,
    onDemand: true,
    discardSelector: ".discard-answer"
    ,immediatelyShowMarkdownHelp:true
    );



    );






    sPaGhEtTiCaSe is a new contributor. Be nice, and check out our Code of Conduct.









    draft saved

    draft discarded


















    StackExchange.ready(
    function ()
    StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f219053%2fvigen%25c3%25a8re-cipher-in-ruby%23new-answer', 'question_page');

    );

    Post as a guest















    Required, but never shown

























    2 Answers
    2






    active

    oldest

    votes








    2 Answers
    2






    active

    oldest

    votes









    active

    oldest

    votes






    active

    oldest

    votes









    1












    $begingroup$

    The code is rather clear, so consider the following to be nitpicks.




    I'd say that you are generating a new key stream from the key. I'd certainly not reuse the key variable.




    The first i = 0 before the loop seems spurious.



    Using i as a counter is well understood, and I'd not worry overly much on the style of it. You are probably the only one who cares if it is really Ruby-esk; developers down the line will understand it.



    What I wonder though is that you run your loop length times, but there is a break that seems to trigger before that. That's not all too clear to me.



    I wonder what happens if you supply it an "empty" key string. Some guard statements may be in order.




    Same for the curly braces. It's clear as it is, choose whatever you want. Personally I slightly favor the braces.




    You could consider creating a mod function, however since % is already the modulus, which will never return a negative value if the right value is positive, it seems to me that removing the + 26 is probably the only thing you need to change (during decryption).



    Instead of using 26 as unexplained magic value, you should get the size of the LETTERS range instead. That way you can also expand your ciphertext later.




    I've got no opinion on the private_class_method as I'm not a Ruby developer (I'm specialized in knowing many languages / constructs and of course applied crypto).






    share|improve this answer











    $endgroup$












    • $begingroup$
      Thanks, that's really helpful! Sadly, I don't have enough reputation to upvote your answer. :(
      $endgroup$
      – sPaGhEtTiCaSe
      2 hours ago















    1












    $begingroup$

    The code is rather clear, so consider the following to be nitpicks.




    I'd say that you are generating a new key stream from the key. I'd certainly not reuse the key variable.




    The first i = 0 before the loop seems spurious.



    Using i as a counter is well understood, and I'd not worry overly much on the style of it. You are probably the only one who cares if it is really Ruby-esk; developers down the line will understand it.



    What I wonder though is that you run your loop length times, but there is a break that seems to trigger before that. That's not all too clear to me.



    I wonder what happens if you supply it an "empty" key string. Some guard statements may be in order.




    Same for the curly braces. It's clear as it is, choose whatever you want. Personally I slightly favor the braces.




    You could consider creating a mod function, however since % is already the modulus, which will never return a negative value if the right value is positive, it seems to me that removing the + 26 is probably the only thing you need to change (during decryption).



    Instead of using 26 as unexplained magic value, you should get the size of the LETTERS range instead. That way you can also expand your ciphertext later.




    I've got no opinion on the private_class_method as I'm not a Ruby developer (I'm specialized in knowing many languages / constructs and of course applied crypto).






    share|improve this answer











    $endgroup$












    • $begingroup$
      Thanks, that's really helpful! Sadly, I don't have enough reputation to upvote your answer. :(
      $endgroup$
      – sPaGhEtTiCaSe
      2 hours ago













    1












    1








    1





    $begingroup$

    The code is rather clear, so consider the following to be nitpicks.




    I'd say that you are generating a new key stream from the key. I'd certainly not reuse the key variable.




    The first i = 0 before the loop seems spurious.



    Using i as a counter is well understood, and I'd not worry overly much on the style of it. You are probably the only one who cares if it is really Ruby-esk; developers down the line will understand it.



    What I wonder though is that you run your loop length times, but there is a break that seems to trigger before that. That's not all too clear to me.



    I wonder what happens if you supply it an "empty" key string. Some guard statements may be in order.




    Same for the curly braces. It's clear as it is, choose whatever you want. Personally I slightly favor the braces.




    You could consider creating a mod function, however since % is already the modulus, which will never return a negative value if the right value is positive, it seems to me that removing the + 26 is probably the only thing you need to change (during decryption).



    Instead of using 26 as unexplained magic value, you should get the size of the LETTERS range instead. That way you can also expand your ciphertext later.




    I've got no opinion on the private_class_method as I'm not a Ruby developer (I'm specialized in knowing many languages / constructs and of course applied crypto).






    share|improve this answer











    $endgroup$



    The code is rather clear, so consider the following to be nitpicks.




    I'd say that you are generating a new key stream from the key. I'd certainly not reuse the key variable.




    The first i = 0 before the loop seems spurious.



    Using i as a counter is well understood, and I'd not worry overly much on the style of it. You are probably the only one who cares if it is really Ruby-esk; developers down the line will understand it.



    What I wonder though is that you run your loop length times, but there is a break that seems to trigger before that. That's not all too clear to me.



    I wonder what happens if you supply it an "empty" key string. Some guard statements may be in order.




    Same for the curly braces. It's clear as it is, choose whatever you want. Personally I slightly favor the braces.




    You could consider creating a mod function, however since % is already the modulus, which will never return a negative value if the right value is positive, it seems to me that removing the + 26 is probably the only thing you need to change (during decryption).



    Instead of using 26 as unexplained magic value, you should get the size of the LETTERS range instead. That way you can also expand your ciphertext later.




    I've got no opinion on the private_class_method as I'm not a Ruby developer (I'm specialized in knowing many languages / constructs and of course applied crypto).







    share|improve this answer














    share|improve this answer



    share|improve this answer








    edited 4 hours ago

























    answered 4 hours ago









    Maarten BodewesMaarten Bodewes

    592212




    592212











    • $begingroup$
      Thanks, that's really helpful! Sadly, I don't have enough reputation to upvote your answer. :(
      $endgroup$
      – sPaGhEtTiCaSe
      2 hours ago
















    • $begingroup$
      Thanks, that's really helpful! Sadly, I don't have enough reputation to upvote your answer. :(
      $endgroup$
      – sPaGhEtTiCaSe
      2 hours ago















    $begingroup$
    Thanks, that's really helpful! Sadly, I don't have enough reputation to upvote your answer. :(
    $endgroup$
    – sPaGhEtTiCaSe
    2 hours ago




    $begingroup$
    Thanks, that's really helpful! Sadly, I don't have enough reputation to upvote your answer. :(
    $endgroup$
    – sPaGhEtTiCaSe
    2 hours ago













    1












    $begingroup$

    Looping using string.length.times.map … and length.times do … is OK, but slightly on the awkward side. I recommend adhering to the convention of writing same-line blocks using and line-spanning blocks using do … end.



    To extend the key, you can use the string multiplication operator. (Note that extending the key longer than necessary doesn't do much harm.)



    You can also factor out more of the commonality between the encrypt and decrypt functions.



    Instead of searching the LETTERS array, I recommend performing arithmetic on ASCII codes.



    module Crypto
    module Vigenere
    module_function
    def encrypt(plaintext, key)
    vigenere(plaintext, key) p, k
    end

    def decrypt(ciphertext, key)
    vigenere(ciphertext, key)
    end

    # Implementation of Vigenere cipher. The combiner block accepts
    # one character from the text and the corresponding character from
    # the key (encoded as a=0, b=1, ..., z=25), and returns the
    # result using the same numerical scheme.
    def vigenere(text, key, &combiner)
    a = 'a'.ord
    ext_key = key * (text.length / key.length + 1)
    text.chars.zip(ext_key.chars).collect do |t, k|
    (a + combiner.call(t.ord - a, k.ord - a)).chr
    end.join
    end
    private_class_method :vigenere
    end
    end





    share|improve this answer











    $endgroup$

















      1












      $begingroup$

      Looping using string.length.times.map … and length.times do … is OK, but slightly on the awkward side. I recommend adhering to the convention of writing same-line blocks using and line-spanning blocks using do … end.



      To extend the key, you can use the string multiplication operator. (Note that extending the key longer than necessary doesn't do much harm.)



      You can also factor out more of the commonality between the encrypt and decrypt functions.



      Instead of searching the LETTERS array, I recommend performing arithmetic on ASCII codes.



      module Crypto
      module Vigenere
      module_function
      def encrypt(plaintext, key)
      vigenere(plaintext, key) p, k
      end

      def decrypt(ciphertext, key)
      vigenere(ciphertext, key)
      end

      # Implementation of Vigenere cipher. The combiner block accepts
      # one character from the text and the corresponding character from
      # the key (encoded as a=0, b=1, ..., z=25), and returns the
      # result using the same numerical scheme.
      def vigenere(text, key, &combiner)
      a = 'a'.ord
      ext_key = key * (text.length / key.length + 1)
      text.chars.zip(ext_key.chars).collect do |t, k|
      (a + combiner.call(t.ord - a, k.ord - a)).chr
      end.join
      end
      private_class_method :vigenere
      end
      end





      share|improve this answer











      $endgroup$















        1












        1








        1





        $begingroup$

        Looping using string.length.times.map … and length.times do … is OK, but slightly on the awkward side. I recommend adhering to the convention of writing same-line blocks using and line-spanning blocks using do … end.



        To extend the key, you can use the string multiplication operator. (Note that extending the key longer than necessary doesn't do much harm.)



        You can also factor out more of the commonality between the encrypt and decrypt functions.



        Instead of searching the LETTERS array, I recommend performing arithmetic on ASCII codes.



        module Crypto
        module Vigenere
        module_function
        def encrypt(plaintext, key)
        vigenere(plaintext, key) p, k
        end

        def decrypt(ciphertext, key)
        vigenere(ciphertext, key)
        end

        # Implementation of Vigenere cipher. The combiner block accepts
        # one character from the text and the corresponding character from
        # the key (encoded as a=0, b=1, ..., z=25), and returns the
        # result using the same numerical scheme.
        def vigenere(text, key, &combiner)
        a = 'a'.ord
        ext_key = key * (text.length / key.length + 1)
        text.chars.zip(ext_key.chars).collect do |t, k|
        (a + combiner.call(t.ord - a, k.ord - a)).chr
        end.join
        end
        private_class_method :vigenere
        end
        end





        share|improve this answer











        $endgroup$



        Looping using string.length.times.map … and length.times do … is OK, but slightly on the awkward side. I recommend adhering to the convention of writing same-line blocks using and line-spanning blocks using do … end.



        To extend the key, you can use the string multiplication operator. (Note that extending the key longer than necessary doesn't do much harm.)



        You can also factor out more of the commonality between the encrypt and decrypt functions.



        Instead of searching the LETTERS array, I recommend performing arithmetic on ASCII codes.



        module Crypto
        module Vigenere
        module_function
        def encrypt(plaintext, key)
        vigenere(plaintext, key) p, k
        end

        def decrypt(ciphertext, key)
        vigenere(ciphertext, key)
        end

        # Implementation of Vigenere cipher. The combiner block accepts
        # one character from the text and the corresponding character from
        # the key (encoded as a=0, b=1, ..., z=25), and returns the
        # result using the same numerical scheme.
        def vigenere(text, key, &combiner)
        a = 'a'.ord
        ext_key = key * (text.length / key.length + 1)
        text.chars.zip(ext_key.chars).collect do |t, k|
        (a + combiner.call(t.ord - a, k.ord - a)).chr
        end.join
        end
        private_class_method :vigenere
        end
        end






        share|improve this answer














        share|improve this answer



        share|improve this answer








        edited 32 mins ago

























        answered 1 hour ago









        200_success200_success

        131k17157422




        131k17157422




















            sPaGhEtTiCaSe is a new contributor. Be nice, and check out our Code of Conduct.









            draft saved

            draft discarded


















            sPaGhEtTiCaSe is a new contributor. Be nice, and check out our Code of Conduct.












            sPaGhEtTiCaSe is a new contributor. Be nice, and check out our Code of Conduct.











            sPaGhEtTiCaSe is a new contributor. Be nice, and check out our Code of Conduct.














            Thanks for contributing an answer to Code Review Stack Exchange!


            • Please be sure to answer the question. Provide details and share your research!

            But avoid


            • Asking for help, clarification, or responding to other answers.

            • Making statements based on opinion; back them up with references or personal experience.

            Use MathJax to format equations. MathJax reference.


            To learn more, see our tips on writing great answers.




            draft saved


            draft discarded














            StackExchange.ready(
            function ()
            StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f219053%2fvigen%25c3%25a8re-cipher-in-ruby%23new-answer', 'question_page');

            );

            Post as a guest















            Required, but never shown





















































            Required, but never shown














            Required, but never shown












            Required, but never shown







            Required, but never shown

































            Required, but never shown














            Required, but never shown












            Required, but never shown







            Required, but never shown







            Popular posts from this blog

            На ростанях Змест Гісторыя напісання | Месца дзеяння | Час дзеяння | Назва | Праблематыка трылогіі | Аўтабіяграфічнасць | Трылогія ў тэатры і кіно | Пераклады | У культуры | Зноскі Літаратура | Спасылкі | НавігацыяДагледжаная версіяправерана1 зменаДагледжаная версіяправерана1 зменаАкадэмік МІЦКЕВІЧ Канстанцін Міхайлавіч (Якуб Колас) Прадмова М. І. Мушынскага, доктара філалагічных навук, члена-карэспандэнта Нацыянальнай акадэміі навук Рэспублікі Беларусь, прафесараНашаніўцы ў трылогіі Якуба Коласа «На ростанях»: вобразы і прататыпы125 лет Янке МавруКнижно-документальная выставка к 125-летию со дня рождения Якуба Коласа (1882—1956)Колас Якуб. Новая зямля (паэма), На ростанях (трылогія). Сулкоўскі Уладзімір. Радзіма Якуба Коласа (серыял жывапісных палотнаў)Вокладка кнігіІлюстрацыя М. С. БасалыгіНа ростаняхАўдыёверсія трылогііВ. Жолтак У Люсiнскай школе 1959

            Францішак Багушэвіч Змест Сям'я | Біяграфія | Творчасць | Мова Багушэвіча | Ацэнкі дзейнасці | Цікавыя факты | Спадчына | Выбраная бібліяграфія | Ушанаванне памяці | У філатэліі | Зноскі | Літаратура | Спасылкі | НавігацыяЛяхоўскі У. Рупіўся дзеля Бога і людзей: Жыццёвы шлях Лявона Вітан-Дубейкаўскага // Вольскі і Памідораў з песняй пра немца Адвакат, паэт, народны заступнік Ашмянскі веснікВ Минске появится площадь Богушевича и улица Сырокомли, Белорусская деловая газета, 19 июля 2001 г.Айцец беларускай нацыянальнай ідэі паўстаў у бронзе Сяргей Аляксандравіч Адашкевіч (1918, Мінск). 80-я гады. Бюст «Францішак Багушэвіч».Яўген Мікалаевіч Ціхановіч. «Партрэт Францішка Багушэвіча»Мікола Мікалаевіч Купава. «Партрэт зачынальніка новай беларускай літаратуры Францішка Багушэвіча»Уладзімір Іванавіч Мелехаў. На помніку «Змагарам за родную мову» Барэльеф «Францішак Багушэвіч»Памяць пра Багушэвіча на Віленшчыне Страчаная сталіца. Беларускія шыльды на вуліцах Вільні«Krynica». Ideologia i przywódcy białoruskiego katolicyzmuФранцішак БагушэвічТворы на knihi.comТворы Францішка Багушэвіча на bellib.byСодаль Уладзімір. Францішак Багушэвіч на Лідчыне;Луцкевіч Антон. Жыцьцё і творчасьць Фр. Багушэвіча ў успамінах ягоных сучасьнікаў // Запісы Беларускага Навуковага таварыства. Вільня, 1938. Сшытак 1. С. 16-34.Большая российская1188761710000 0000 5537 633Xn9209310021619551927869394п

            Беларусь Змест Назва Гісторыя Геаграфія Сімволіка Дзяржаўны лад Палітычныя партыі Міжнароднае становішча і знешняя палітыка Адміністрацыйны падзел Насельніцтва Эканоміка Культура і грамадства Сацыяльная сфера Узброеныя сілы Заўвагі Літаратура Спасылкі НавігацыяHGЯOiТоп-2011 г. (па версіі ej.by)Топ-2013 г. (па версіі ej.by)Топ-2016 г. (па версіі ej.by)Топ-2017 г. (па версіі ej.by)Нацыянальны статыстычны камітэт Рэспублікі БеларусьШчыльнасць насельніцтва па краінахhttp://naviny.by/rubrics/society/2011/09/16/ic_articles_116_175144/А. Калечыц, У. Ксяндзоў. Спробы засялення краю неандэртальскім чалавекам.І ў Менску былі мамантыА. Калечыц, У. Ксяндзоў. Старажытны каменны век (палеаліт). Першапачатковае засяленне тэрыторыіГ. Штыхаў. Балты і славяне ў VI—VIII стст.М. Клімаў. Полацкае княства ў IX—XI стст.Г. Штыхаў, В. Ляўко. Палітычная гісторыя Полацкай зямліГ. Штыхаў. Дзяржаўны лад у землях-княствахГ. Штыхаў. Дзяржаўны лад у землях-княствахБеларускія землі ў складзе Вялікага Княства ЛітоўскагаЛюблінская унія 1569 г."The Early Stages of Independence"Zapomniane prawdy25 гадоў таму было аб'яўлена, што Язэп Пілсудскі — беларус (фота)Наша вадаДакументы ЧАЭС: Забруджванне тэрыторыі Беларусі « ЧАЭС Зона адчужэнняСведения о политических партиях, зарегистрированных в Республике Беларусь // Министерство юстиции Республики БеларусьСтатыстычны бюлетэнь „Полаўзроставая структура насельніцтва Рэспублікі Беларусь на 1 студзеня 2012 года і сярэднегадовая колькасць насельніцтва за 2011 год“Индекс человеческого развития Беларуси — не было бы нижеБеларусь занимает первое место в СНГ по индексу развития с учетом гендерного факцёраНацыянальны статыстычны камітэт Рэспублікі БеларусьКанстытуцыя РБ. Артыкул 17Трансфармацыйныя задачы БеларусіВыйсце з крызісу — далейшае рэфармаванне Беларускі рубель — сусветны лідар па дэвальвацыяхПра змену коштаў у кастрычніку 2011 г.Бядней за беларусаў у СНД толькі таджыкіСярэдні заробак у верасні дасягнуў 2,26 мільёна рублёўЭканомікаГаласуем за ТОП-100 беларускай прозыСучасныя беларускія мастакіАрхитектура Беларуси BELARUS.BYА. Каханоўскі. Культура Беларусі ўсярэдзіне XVII—XVIII ст.Анталогія беларускай народнай песні, гуказапісы спеваўБеларускія Музычныя IнструментыБеларускі рок, які мы страцілі. Топ-10 гуртоў«Мясцовы час» — нязгаслая легенда беларускай рок-музыкіСЯРГЕЙ БУДКІН. МЫ НЯ ЗНАЕМ СВАЁЙ МУЗЫКІМ. А. Каладзінскі. НАРОДНЫ ТЭАТРМагнацкія культурныя цэнтрыПублічная дыскусія «Беларуская новая пьеса: без беларускай мовы ці беларуская?»Беларускія драматургі па-ранейшаму лепш ставяцца за мяжой, чым на радзіме«Працэс незалежнага кіно пайшоў, і дзяржаву турбуе яго непадкантрольнасць»Беларускія філосафы ў пошуках прасторыВсе идём в библиотекуАрхіваванаАб Нацыянальнай праграме даследавання і выкарыстання касмічнай прасторы ў мірных мэтах на 2008—2012 гадыУ космас — разам.У суседнім з Барысаўскім раёне пабудуюць Камандна-вымяральны пунктСвяты і абрады беларусаў«Мірныя бульбашы з малой краіны» — 5 непраўдзівых стэрэатыпаў пра БеларусьМ. Раманюк. Беларускае народнае адзеннеУ Беларусі скарачаецца колькасць злачынстваўЛукашэнка незадаволены мінскімі ўладамі Крадзяжы складаюць у Мінску каля 70% злачынстваў Узровень злачыннасці ў Мінскай вобласці — адзін з самых высокіх у краіне Генпракуратура аналізуе стан са злачыннасцю ў Беларусі па каэфіцыенце злачыннасці У Беларусі стабілізавалася крымінагеннае становішча, лічыць генпракурорЗамежнікі сталі здзяйсняць у Беларусі больш злачынстваўМУС Беларусі турбуе рост рэцыдыўнай злачыннасціЯ з ЖЭСа. Дазволіце вас абкрасці! Рэйтынг усіх службаў і падраздзяленняў ГУУС Мінгарвыканкама вырасАб КДБ РБГісторыя Аператыўна-аналітычнага цэнтра РБГісторыя ДКФРТаможняagentura.ruБеларусьBelarus.by — Афіцыйны сайт Рэспублікі БеларусьСайт урада БеларусіRadzima.org — Збор архітэктурных помнікаў, гісторыя Беларусі«Глобус Беларуси»Гербы и флаги БеларусиАсаблівасці каменнага веку на БеларусіА. Калечыц, У. Ксяндзоў. Старажытны каменны век (палеаліт). Першапачатковае засяленне тэрыторыіУ. Ксяндзоў. Сярэдні каменны век (мезаліт). Засяленне краю плямёнамі паляўнічых, рыбакоў і збіральнікаўА. Калечыц, М. Чарняўскі. Плямёны на тэрыторыі Беларусі ў новым каменным веку (неаліце)А. Калечыц, У. Ксяндзоў, М. Чарняўскі. Гаспадарчыя заняткі ў каменным векуЭ. Зайкоўскі. Духоўная культура ў каменным векуАсаблівасці бронзавага веку на БеларусіФарміраванне супольнасцей ранняга перыяду бронзавага векуФотографии БеларусиРоля беларускіх зямель ва ўтварэнні і ўмацаванні ВКЛВ. Фадзеева. З гісторыі развіцця беларускай народнай вышыўкіDMOZGran catalanaБольшая российскаяBritannica (анлайн)Швейцарскі гістарычны15325917611952699xDA123282154079143-90000 0001 2171 2080n9112870100577502ge128882171858027501086026362074122714179пппппп