Playing an MP3 through Ruby on Rails in chrome using send_file

288 views
Skip to first unread message

Garrett

unread,
Sep 30, 2011, 2:00:49 PM9/30/11
to rubyonra...@googlegroups.com

I'm writing a Ruby on Rails application which allows a user to upload an mp3 file, then play it back. I have it working to the point where a user can do those things, BUT there is an issue when seeking through the song. If a user seeks ahead (or lets it play) to a spot in the song, usually about 2/3 or 3/4 the way through the song, then attempts to seek back to the beginning (for example 0:20), the play timer will go to 0:20, like it should but the actual audio will start over again as if the user seeked to 0:00.

Right now I'm simply attempting to get the song to play in chrome's basic html5 mp3 player that it uses when passed an mp3 file. This is the code I'm using to serve up the file, hopefully with all the correct headers:


I think it has something to do with send_file not adhering to the Content-Range headers.  I'd appreciate any insight into this problem. I feel like I'm so close because it pretty much all works, except for seeking near the beginning causing a new request.

Thanks!

Garrett S.

unread,
Sep 30, 2011, 4:06:21 PM9/30/11
to rubyonra...@googlegroups.com
I'm writing a Ruby on Rails application which allows a user to upload an
mp3 file, then play it back. I have it working to the point where a user
can do those things, BUT there is an issue when seeking through the
song. If a user seeks ahead (or lets it play) to a spot in the song,
usually about 2/3 or 3/4 the way through the song, then attempts to seek
back to the beginning (for example 0:20), the play timer will go to
0:20, like it should but the actual audio will start over again as if
the user seeked to 0:00.

Right now I'm simply attempting to get the song to play in chrome's
basic html5 mp3 player that it uses when passed an mp3 file. This is the
code I'm using to serve up the file, hopefully with all the correct
headers:

http://privatepaste.com/8406f69836

I think it has something to do with send_file not accepting the range
header, maybe it's just the way I'm setting it? I'd appreciate any


insight into this problem. I feel like I'm so close because it pretty
much all works, except for seeking near the beginning causing a new
request.

Thanks!

--
Posted via http://www.ruby-forum.com/.

Walter Lee Davis

unread,
Sep 30, 2011, 4:13:10 PM9/30/11
to rubyonra...@googlegroups.com

Can you just pass the URL to the file, and not pass it through send_file? I've used

def download
if cannot?( :download, Title )
raise CanCan::AccessDenied
else
redirect_to @asset.source.expiring_url(10)
end
end

from Paperclip to handle a link to a protected file on S3. Works like a charm, and it's just a file download from S3, so it's up to the browser to handle that.

Walter

>
> Thanks!
>
> --
> Posted via http://www.ruby-forum.com/.
>

> --
> You received this message because you are subscribed to the Google Groups "Ruby on Rails: Talk" group.
> To post to this group, send email to rubyonra...@googlegroups.com.
> To unsubscribe from this group, send email to rubyonrails-ta...@googlegroups.com.
> For more options, visit this group at http://groups.google.com/group/rubyonrails-talk?hl=en.
>

Garrett S.

unread,
Sep 30, 2011, 4:26:56 PM9/30/11
to rubyonra...@googlegroups.com
I think I like what you're saying, but the only problem I have with it
is controlling access to the paperclip url. This may mean I just don't
understand how the paperclip urls work. But I'm also trying to ensure
that only the user who uploaded it, can access it. I wasn't sure if I
would be able to control this once a url for an audio file was exposed.

Walter Lee Davis

unread,
Sep 30, 2011, 4:38:25 PM9/30/11
to rubyonra...@googlegroups.com

On Sep 30, 2011, at 4:26 PM, Garrett S. wrote:

> I think I like what you're saying, but the only problem I have with it
> is controlling access to the paperclip url. This may mean I just don't
> understand how the paperclip urls work. But I'm also trying to ensure
> that only the user who uploaded it, can access it. I wasn't sure if I
> would be able to control this once a url for an audio file was exposed.

That's the beauty of how this works. When you save the file to S3, you have two basic options. Usually, you want these to be world-readable, so you pass

:s3_permissions => :public

in your has_attached_file call.

But if you pass :private instead, then the file can only be accessed through your S3 credentials. Okay, now only your app can read it. But that's what the expiring_url method is for. That gins up a one-time token that expires in N minutes, and allows one download during that window, after which it simply won't work at all. Your keys are used to create this token, but it's a one-way hash so you don't have any leakage possible of your actual credentials.

Walter

>
> --
> Posted via http://www.ruby-forum.com/.
>

Garrett S.

unread,
Sep 30, 2011, 4:51:28 PM9/30/11
to rubyonra...@googlegroups.com
>
> :s3_permissions => :public
>
> in your has_attached_file call.
>
> But if you pass :private instead, then the file can only be accessed
> through your S3 credentials. Okay, now only your app can read it. But
> that's what the expiring_url method is for. That gins up a one-time
> token that expires in N minutes, and allows one download during that
> window, after which it simply won't work at all. Your keys are used to
> create this token, but it's a one-way hash so you don't have any leakage
> possible of your actual credentials.
>
> Walter

I think the only problem I have left is that I was hoping to not use S3
and be able to store the files on my own server.

blaz boy

unread,
Sep 30, 2011, 4:32:44 PM9/30/11
to rubyonra...@googlegroups.com
if you are asking about how to restrict access to a file -in your case MP3 file-
i just made a solution couple monthes ago with PHP that you may re implement 
create a table with fields ( path , token)
for every request of a file create a token for it with that path in that table
create a controller that get the token as parameter and download the file
when user access the url through that controller you'll delete the record 
make the controller show 404 if the token wasn't in the table
you can add a column for user id in case the file is available for multiple user

that's how you'll get a single use URL :)
please till me if there is any hole in my logic 
----
Emad Elsaid  a.k.a. (Blaze Boy)
Websites Designer/Developer
Mobile: +20 14 13 7 16 75
http://blazeeboy.blogspot.com



Garrett

unread,
Oct 1, 2011, 11:54:16 AM10/1/11
to rubyonra...@googlegroups.com
Hey!

Thanks for the response, I actually have the restricted access part working.  My problem is getting Send_file to use the content range header so that when someone seeks to a location in the song that causes a new request it doesn't just restart the song from the beginning, but from the actual location what was 'seeked' to.
Reply all
Reply to author
Forward
0 new messages