├── .gitignore ├── .rspec ├── .ruby-gemset ├── .ruby-version ├── Gemfile ├── Gemfile.lock ├── README.md ├── Rakefile ├── app ├── assets │ ├── images │ │ └── rails.png │ ├── javascripts │ │ ├── application.js │ │ ├── cors.js │ │ ├── products.js │ │ └── scaffold.js │ └── stylesheets │ │ ├── application.css.scss │ │ └── layout.css.scss ├── controllers │ ├── application_controller.rb │ ├── aws_controller.rb │ └── products_controller.rb ├── mailers │ └── .gitkeep ├── models │ ├── .gitkeep │ └── product.rb └── views │ ├── application │ └── _flash_messages.html.erb │ ├── layouts │ └── application.html.erb │ └── products │ └── new.html.erb ├── config.ru ├── config ├── application.rb ├── boot.rb ├── database.yml ├── environment.rb ├── environments │ ├── development.rb │ ├── production.rb │ └── test.rb ├── initializers │ ├── aws.rb │ ├── backtrace_silencers.rb │ ├── inflections.rb │ ├── mime_types.rb │ ├── session_store.rb │ ├── simple_form.rb │ ├── simple_form_bootstrap.rb │ └── wrap_parameters.rb ├── locales │ ├── en.yml │ └── simple_form.en.yml ├── routes.rb └── secrets.yml ├── db ├── migrate │ └── 20130523082752_create_products.rb ├── schema.rb └── seeds.rb ├── doc └── README_FOR_APP ├── log └── .gitkeep ├── public ├── 404.html ├── 422.html ├── 500.html ├── favicon.ico ├── img │ ├── glyphicons-halflings-white.png │ └── glyphicons-halflings.png └── robots.txt ├── script └── rails ├── spec ├── controllers │ ├── aws_controller_spec.rb │ └── products_controller_spec.rb ├── helpers │ └── products_helper_spec.rb └── spec_helper.rb ├── test ├── fixtures │ ├── .gitkeep │ └── products.yml ├── functional │ └── .gitkeep ├── integration │ └── .gitkeep ├── performance │ └── browsing_test.rb ├── test_helper.rb └── unit │ ├── .gitkeep │ └── product_test.rb └── vendor ├── assets ├── javascripts │ ├── .gitkeep │ ├── bootstrap-filestyle.js │ └── bootstrap.js └── stylesheets │ ├── .gitkeep │ ├── bootstrap-responsive.css │ └── bootstrap.css └── plugins └── .gitkeep /.gitignore: -------------------------------------------------------------------------------- 1 | # See http://help.github.com/ignore-files/ for more about ignoring files. 2 | # 3 | # If you find yourself ignoring temporary files generated by your text editor 4 | # or operating system, you probably want to add a global ignore instead: 5 | # git config --global core.excludesfile ~/.gitignore_global 6 | 7 | # Ignore bundler config 8 | /.bundle 9 | 10 | # Ignore the default SQLite database. 11 | /db/*.sqlite3 12 | 13 | # Ignore all logfiles and tempfiles. 14 | /log/*.log 15 | /tmp 16 | export.txt 17 | .env 18 | Procfile 19 | -------------------------------------------------------------------------------- /.rspec: -------------------------------------------------------------------------------- 1 | --color 2 | -------------------------------------------------------------------------------- /.ruby-gemset: -------------------------------------------------------------------------------- 1 | s3cors 2 | -------------------------------------------------------------------------------- /.ruby-version: -------------------------------------------------------------------------------- 1 | ruby-2.2.2 2 | -------------------------------------------------------------------------------- /Gemfile: -------------------------------------------------------------------------------- 1 | source "https://rubygems.org" 2 | 3 | gem "rails", "4.2.3" 4 | gem "sass-rails" 5 | gem 'uglifier', '>= 1.3.0' 6 | gem "jquery-rails" 7 | gem 'turbolinks' 8 | gem 'jbuilder', '~> 2.0' 9 | gem 'sdoc', '~> 0.4.0', group: :doc 10 | 11 | gem 'bootstrap-sass', '~> 3.3.1' 12 | gem 'autoprefixer-rails' 13 | 14 | gem "coffee-rails" 15 | gem "bcrypt" 16 | gem "simple_form" 17 | gem "cognac" 18 | 19 | group :development, :test do 20 | gem "sqlite3" 21 | gem "rspec-rails" 22 | gem 'quiet_assets' 23 | end 24 | 25 | group :test do 26 | # Pretty printed test output 27 | gem "turn", :require => false 28 | gem "factory_girl_rails" 29 | end -------------------------------------------------------------------------------- /Gemfile.lock: -------------------------------------------------------------------------------- 1 | GEM 2 | remote: https://rubygems.org/ 3 | specs: 4 | actionmailer (4.2.3) 5 | actionpack (= 4.2.3) 6 | actionview (= 4.2.3) 7 | activejob (= 4.2.3) 8 | mail (~> 2.5, >= 2.5.4) 9 | rails-dom-testing (~> 1.0, >= 1.0.5) 10 | actionpack (4.2.3) 11 | actionview (= 4.2.3) 12 | activesupport (= 4.2.3) 13 | rack (~> 1.6) 14 | rack-test (~> 0.6.2) 15 | rails-dom-testing (~> 1.0, >= 1.0.5) 16 | rails-html-sanitizer (~> 1.0, >= 1.0.2) 17 | actionview (4.2.3) 18 | activesupport (= 4.2.3) 19 | builder (~> 3.1) 20 | erubis (~> 2.7.0) 21 | rails-dom-testing (~> 1.0, >= 1.0.5) 22 | rails-html-sanitizer (~> 1.0, >= 1.0.2) 23 | activejob (4.2.3) 24 | activesupport (= 4.2.3) 25 | globalid (>= 0.3.0) 26 | activemodel (4.2.3) 27 | activesupport (= 4.2.3) 28 | builder (~> 3.1) 29 | activerecord (4.2.3) 30 | activemodel (= 4.2.3) 31 | activesupport (= 4.2.3) 32 | arel (~> 6.0) 33 | activesupport (4.2.3) 34 | i18n (~> 0.7) 35 | json (~> 1.7, >= 1.7.7) 36 | minitest (~> 5.1) 37 | thread_safe (~> 0.3, >= 0.3.4) 38 | tzinfo (~> 1.1) 39 | ansi (1.5.0) 40 | arel (6.0.0) 41 | autoprefixer-rails (5.2.1) 42 | execjs 43 | json 44 | bcrypt (3.1.10) 45 | bootstrap-sass (3.3.5.1) 46 | autoprefixer-rails (>= 5.0.0.1) 47 | sass (>= 3.3.0) 48 | builder (3.2.2) 49 | coffee-rails (4.1.0) 50 | coffee-script (>= 2.2.0) 51 | railties (>= 4.0.0, < 5.0) 52 | coffee-script (2.4.1) 53 | coffee-script-source 54 | execjs 55 | coffee-script-source (1.9.1.1) 56 | cognac (0.1.1) 57 | activesupport (~> 4.2) 58 | diff-lcs (1.2.5) 59 | erubis (2.7.0) 60 | execjs (2.5.2) 61 | factory_girl (4.5.0) 62 | activesupport (>= 3.0.0) 63 | factory_girl_rails (4.5.0) 64 | factory_girl (~> 4.5.0) 65 | railties (>= 3.0.0) 66 | globalid (0.3.5) 67 | activesupport (>= 4.1.0) 68 | i18n (0.7.0) 69 | jbuilder (2.3.0) 70 | activesupport (>= 3.0.0, < 5) 71 | multi_json (~> 1.2) 72 | jquery-rails (4.0.4) 73 | rails-dom-testing (~> 1.0) 74 | railties (>= 4.2.0) 75 | thor (>= 0.14, < 2.0) 76 | json (1.8.3) 77 | loofah (2.0.2) 78 | nokogiri (>= 1.5.9) 79 | mail (2.6.3) 80 | mime-types (>= 1.16, < 3) 81 | mime-types (2.6.1) 82 | mini_portile (0.6.2) 83 | minitest (5.7.0) 84 | multi_json (1.11.1) 85 | nokogiri (1.6.6.2) 86 | mini_portile (~> 0.6.0) 87 | quiet_assets (1.1.0) 88 | railties (>= 3.1, < 5.0) 89 | rack (1.6.4) 90 | rack-test (0.6.3) 91 | rack (>= 1.0) 92 | rails (4.2.3) 93 | actionmailer (= 4.2.3) 94 | actionpack (= 4.2.3) 95 | actionview (= 4.2.3) 96 | activejob (= 4.2.3) 97 | activemodel (= 4.2.3) 98 | activerecord (= 4.2.3) 99 | activesupport (= 4.2.3) 100 | bundler (>= 1.3.0, < 2.0) 101 | railties (= 4.2.3) 102 | sprockets-rails 103 | rails-deprecated_sanitizer (1.0.3) 104 | activesupport (>= 4.2.0.alpha) 105 | rails-dom-testing (1.0.6) 106 | activesupport (>= 4.2.0.beta, < 5.0) 107 | nokogiri (~> 1.6.0) 108 | rails-deprecated_sanitizer (>= 1.0.1) 109 | rails-html-sanitizer (1.0.2) 110 | loofah (~> 2.0) 111 | railties (4.2.3) 112 | actionpack (= 4.2.3) 113 | activesupport (= 4.2.3) 114 | rake (>= 0.8.7) 115 | thor (>= 0.18.1, < 2.0) 116 | rake (10.4.2) 117 | rdoc (4.2.0) 118 | rspec-core (3.3.1) 119 | rspec-support (~> 3.3.0) 120 | rspec-expectations (3.3.0) 121 | diff-lcs (>= 1.2.0, < 2.0) 122 | rspec-support (~> 3.3.0) 123 | rspec-mocks (3.3.1) 124 | diff-lcs (>= 1.2.0, < 2.0) 125 | rspec-support (~> 3.3.0) 126 | rspec-rails (3.3.2) 127 | actionpack (>= 3.0, < 4.3) 128 | activesupport (>= 3.0, < 4.3) 129 | railties (>= 3.0, < 4.3) 130 | rspec-core (~> 3.3.0) 131 | rspec-expectations (~> 3.3.0) 132 | rspec-mocks (~> 3.3.0) 133 | rspec-support (~> 3.3.0) 134 | rspec-support (3.3.0) 135 | sass (3.4.15) 136 | sass-rails (5.0.3) 137 | railties (>= 4.0.0, < 5.0) 138 | sass (~> 3.1) 139 | sprockets (>= 2.8, < 4.0) 140 | sprockets-rails (>= 2.0, < 4.0) 141 | tilt (~> 1.1) 142 | sdoc (0.4.1) 143 | json (~> 1.7, >= 1.7.7) 144 | rdoc (~> 4.0) 145 | simple_form (3.1.0) 146 | actionpack (~> 4.0) 147 | activemodel (~> 4.0) 148 | sprockets (3.2.0) 149 | rack (~> 1.0) 150 | sprockets-rails (2.3.2) 151 | actionpack (>= 3.0) 152 | activesupport (>= 3.0) 153 | sprockets (>= 2.8, < 4.0) 154 | sqlite3 (1.3.10) 155 | thor (0.19.1) 156 | thread_safe (0.3.5) 157 | tilt (1.4.1) 158 | turbolinks (2.5.3) 159 | coffee-rails 160 | turn (0.9.6) 161 | ansi 162 | tzinfo (1.2.2) 163 | thread_safe (~> 0.1) 164 | uglifier (2.7.1) 165 | execjs (>= 0.3.0) 166 | json (>= 1.8.0) 167 | 168 | PLATFORMS 169 | ruby 170 | 171 | DEPENDENCIES 172 | autoprefixer-rails 173 | bcrypt 174 | bootstrap-sass (~> 3.3.1) 175 | coffee-rails 176 | cognac 177 | factory_girl_rails 178 | jbuilder (~> 2.0) 179 | jquery-rails 180 | quiet_assets 181 | rails (= 4.2.3) 182 | rspec-rails 183 | sass-rails 184 | sdoc (~> 0.4.0) 185 | simple_form 186 | sqlite3 187 | turbolinks 188 | turn 189 | uglifier (>= 1.3.0) 190 | 191 | BUNDLED WITH 192 | 1.10.3 193 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | ## Step 1 - Configure S3 for CORS 2 | 3 | 1. Create an Amazon S3 Bucket 4 | 2. Click on Properties -> Permissions -> Edit CORS 5 | 3. Enter a CORS configuration, such as the following. For security purposes you should restrict the origin further. 6 | 7 | ```xml 8 | 9 | 10 | 11 | * 12 | PUT 13 | 3000 14 | * 15 | 16 | 17 | ``` 18 | 19 | ## Step 2 - Set your AWS Keys and bucket 20 | 21 | The application can be configured via the following environment variables. 22 | 23 | - `export AWS_ACCESS_KEY_ID='your aws acccess key id'` 24 | - `export AWS_SECRET_ACCESS_KEY='your aws secret access key'` 25 | - `export AWS_S3_BUCKET='example.com'` 26 | 27 | 28 | ## General Flow 29 | 30 | This project uses client-side JavaScript and server-side Ruby. In general, the completed file-upload process follows these steps: 31 | 32 | 1. A file is selected for upload by the user in their web browser; 33 | 2. JavaScript is then responsible for making a request to your web application, which produces a temporary signature with which to sign the upload request; 34 | 3. The temporary signed request is returned to the browser in JSON format; 35 | 4. JavaScript then uploads the file directly to Amazon S3 using the signed request supplied by your Rails application. 36 | 37 | ## Code overview 38 | 39 | The application flow is as follows: 40 | 41 | 1. [Client] When the user chooses a file, a callback (defined in `app/assets/javascripts/products.js`) contacts the server to request a signed URL that can be used for uploading to S3. 42 | 2. [Server] AWSController generates a signed URL according to http://docs.aws.amazon.com/AmazonS3/latest/dev/RESTAuthentication.html and returns it using a JSON hash. 43 | 3. [Client] The `uploadToS3` function (defined in `app/assets/javascripts/cors.js`), takes the signed URL and the file, and PUTs the file onto S3. 44 | 4. [Client] Once the upload is finished, the URL of the uploaded file is appended to the form as a hidden field. 45 | 5. [Client] When the user submits the form, the file field is removed (to avoid uploading the file again). Name, price and the file URL are POSTed to the server. 46 | 6. [Server] A new product record is created. 47 | 48 | The most important files are are: 49 | 50 | - `app/controllers/aws_controller` - Given a filename and content type, returns a JSON of the form: 51 | 52 | ```JSON 53 | { 54 | "put_url": "The *signed* URL that must be used to make the request from the client side]", 55 | "file_url": "The url the file will be uploaded to. A random number is appended to the file name to avoid name collisions." 56 | } 57 | ``` 58 | - `app/assets/javascripts/cors.js` - Performs the actual upload using an XMLHttpRequest. Required the signed URL, and the file to perform the request. 59 | - `app/assets/javascripts/products.js` - Handles UI callbacks, such as updating the progress bar. Also handles the file-upload callback. When the user chooses a file, the server is contacted to generate a signed URL, which is then passed to the CORS script described above to complete the upload. -------------------------------------------------------------------------------- /Rakefile: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env rake 2 | # Add your own tasks in files placed in lib/tasks ending in .rake, 3 | # for example lib/tasks/capistrano.rake, and they will automatically be available to Rake. 4 | 5 | require File.expand_path('../config/application', __FILE__) 6 | 7 | Rails.application.load_tasks -------------------------------------------------------------------------------- /app/assets/images/rails.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/app/assets/images/rails.png -------------------------------------------------------------------------------- /app/assets/javascripts/application.js: -------------------------------------------------------------------------------- 1 | // This is a manifest file that'll be compiled into application.js, which will include all the files 2 | // listed below. 3 | // 4 | // Any JavaScript/Coffee file within this directory, lib/assets/javascripts, vendor/assets/javascripts, 5 | // or vendor/assets/javascripts of plugins, if any, can be referenced here using a relative path. 6 | // 7 | // It's not advisable to add code directly here, but if you do, it'll appear at the bottom of the 8 | // the compiled file. 9 | // 10 | // WARNING: THE FIRST BLANK LINE MARKS THE END OF WHAT'S TO BE PROCESSED, ANY BLANK LINE SHOULD 11 | // GO AFTER THE REQUIRES BELOW. 12 | // 13 | //= require jquery 14 | //= require jquery_ujs 15 | //= require bootstrap 16 | //= require bootstrap-filestyle 17 | // 18 | //= require scaffold 19 | //= require products 20 | //= require cors 21 | -------------------------------------------------------------------------------- /app/assets/javascripts/cors.js: -------------------------------------------------------------------------------- 1 | function createCORSRequest(method, url) 2 | { 3 | var xhr = new XMLHttpRequest(); 4 | if ("withCredentials" in xhr) 5 | { 6 | xhr.open(method, url, true); 7 | } 8 | else if (typeof XDomainRequest != "undefined") 9 | { 10 | xhr = new XDomainRequest(); 11 | xhr.open(method, url); 12 | } 13 | else 14 | { 15 | xhr = null; 16 | } 17 | return xhr; 18 | } 19 | 20 | /** 21 | * Use a CORS call to upload the given file to S3. Assumes the url 22 | * parameter has been signed and is accessible for upload. 23 | */ 24 | function uploadToS3(file, url) 25 | { 26 | var xhr = createCORSRequest('PUT', url); 27 | if (!xhr) 28 | { 29 | setProgress(0, 'CORS not supported'); 30 | } 31 | else 32 | { 33 | xhr.onload = function() 34 | { 35 | if(xhr.status == 200) 36 | { 37 | setProgress(100, 'Upload completed.'); 38 | } 39 | else 40 | { 41 | setProgress(0, 'Upload error: ' + xhr.status); 42 | } 43 | }; 44 | 45 | xhr.onerror = function() 46 | { 47 | setProgress(0, 'XHR error.'); 48 | }; 49 | 50 | xhr.upload.onprogress = function(e) 51 | { 52 | if (e.lengthComputable) 53 | { 54 | var percentLoaded = Math.round((e.loaded / e.total) * 100); 55 | setProgress(percentLoaded, percentLoaded == 100 ? 'Finalizing.' : 'Uploading.'); 56 | } 57 | }; 58 | 59 | xhr.setRequestHeader('Content-Type', file.type); 60 | //xhr.setRequestHeader('x-amz-acl', 'public-read'); 61 | xhr.send(file); 62 | } 63 | } 64 | -------------------------------------------------------------------------------- /app/assets/javascripts/products.js: -------------------------------------------------------------------------------- 1 | window.uploadFile = function(file) { 2 | return $.getJSON("/generate_signed_s3_url?filename=" + file.name.replace(/ /g,"_") + "&content_type=" + file.type, function(data) { 3 | uploadToS3(file, data.put_url); 4 | return $("#product_url").val(data.file_url); 5 | }); 6 | }; 7 | 8 | window.setProgress = function(progress, str) { 9 | $(".progress .bar").css("width", progress + "%"); 10 | $("#product-upload-progress .text").text(str); 11 | if (progress === 100) { 12 | return $("#new_product_submit").show(); 13 | } 14 | }; 15 | 16 | $(function() { 17 | $("#product-upload-progress").hide(); 18 | $("#new_product_submit").hide(); 19 | $("#new_product_submit").click(function() { 20 | $("#product_file").remove(); 21 | return true; 22 | }); 23 | return $("#product_file").change(function(obj) { 24 | var file; 25 | 26 | file = obj.target.files[0]; 27 | if (file.size > 500 * 1024 * 1024) { 28 | alert("File cannot be larger than 500MB."); 29 | return false; 30 | } 31 | $("#product-upload-progress").show(); 32 | return uploadFile(file); 33 | }); 34 | }); -------------------------------------------------------------------------------- /app/assets/javascripts/scaffold.js: -------------------------------------------------------------------------------- 1 | $(function() { 2 | $(":file").filestyle({ 3 | icon: false 4 | }); 5 | }); -------------------------------------------------------------------------------- /app/assets/stylesheets/application.css.scss: -------------------------------------------------------------------------------- 1 | /* 2 | * This is a manifest file that'll be compiled into application.css, which will include all the files 3 | * listed below. 4 | * 5 | * Any CSS and SCSS file within this directory, lib/assets/stylesheets, vendor/assets/stylesheets, 6 | * or vendor/assets/stylesheets of plugins, if any, can be referenced here using a relative path. 7 | * 8 | * You're free to add application-wide styles to this file and they'll appear at the top of the 9 | * compiled file, but it's generally better to create a new file per style scope. 10 | * 11 | *= require_self 12 | */ 13 | 14 | @import "bootstrap-sprockets"; 15 | @import "bootstrap"; 16 | @import "bootstrap-responsive"; 17 | 18 | @import "layout"; -------------------------------------------------------------------------------- /app/assets/stylesheets/layout.css.scss: -------------------------------------------------------------------------------- 1 | body { 2 | padding: 10px; 3 | } -------------------------------------------------------------------------------- /app/controllers/application_controller.rb: -------------------------------------------------------------------------------- 1 | class ApplicationController < ActionController::Base 2 | # Prevent CSRF attacks by raising an exception. 3 | # For APIs, you may want to use :null_session instead. 4 | protect_from_forgery with: :exception 5 | end 6 | -------------------------------------------------------------------------------- /app/controllers/aws_controller.rb: -------------------------------------------------------------------------------- 1 | class AwsController < ApplicationController 2 | 3 | def generate_signed_s3_url 4 | # To avoid file collision, we prepend string to the file_name 5 | file_name = Cognac::CloudFile.generate(params[:filename]) 6 | resource_end_point = Cognac::CloudFile.resource_end_point(ENV["AWS_S3_BUCKET"], file_name) 7 | 8 | options = Cognac::Signature.generate_options_for_build_s3_upload_url(ENV["AWS_S3_BUCKET"], file_name, params[:content_type]) 9 | url = Cognac::Signature.build_s3_upload_url(resource_end_point, ENV["AWS_ACCESS_KEY_ID"], ENV["AWS_SECRET_ACCESS_KEY"], options) 10 | 11 | render :json => {:put_url => url, :file_url => resource_end_point} 12 | end 13 | end 14 | -------------------------------------------------------------------------------- /app/controllers/products_controller.rb: -------------------------------------------------------------------------------- 1 | class ProductsController < ApplicationController 2 | 3 | def new 4 | @product = Product.new 5 | end 6 | 7 | def create 8 | @product = Product.new(permitted_params) 9 | 10 | if @product.save 11 | redirect_to root_path, :notice => "Upload was successful." 12 | else 13 | render :new 14 | end 15 | end 16 | 17 | private 18 | 19 | def permitted_params 20 | params.require(:product).permit(:name, :price, :url) 21 | end 22 | end 23 | -------------------------------------------------------------------------------- /app/mailers/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/app/mailers/.gitkeep -------------------------------------------------------------------------------- /app/models/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/app/models/.gitkeep -------------------------------------------------------------------------------- /app/models/product.rb: -------------------------------------------------------------------------------- 1 | class Product < ActiveRecord::Base 2 | 3 | validates :name, :presence => true 4 | validates :price, :presence => true 5 | validates :url, :presence => true 6 | 7 | end 8 | -------------------------------------------------------------------------------- /app/views/application/_flash_messages.html.erb: -------------------------------------------------------------------------------- 1 | <% if flash[:notice] %> 2 |
3 | <%= link_to "#", :class => "close", :data => {:dismiss => "alert"} do %> 4 | × 5 | <% end %> 6 | <%= flash[:notice] %> 7 |
8 | <% end %> 9 | 10 | <% if flash[:error] %> 11 |
12 | <%= link_to "#", :class => "close", :data => {:dismiss => "alert"} do %> 13 | × 14 | <% end %> 15 | <%= flash[:error] %> 16 |
17 | <% end %> -------------------------------------------------------------------------------- /app/views/layouts/application.html.erb: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | S3CorsUploadRails 5 | <%= stylesheet_link_tag "application", :media => "all" %> 6 | <%= javascript_include_tag "application" %> 7 | <%= csrf_meta_tags %> 8 | 9 | 10 |
11 | <%= render "flash_messages" %> 12 | <%= yield %> 13 |
14 | 15 | -------------------------------------------------------------------------------- /app/views/products/new.html.erb: -------------------------------------------------------------------------------- 1 | <%= simple_form_for @product, :html => {:class => "form-horizontal"} do |f| %> 2 | 3 | <%= f.input :name, :required => true, :input_html => {:required => "required"} %> 4 | <%= f.input :price, :required => true, :input_html => {:required => "required"} %> 5 | <%= f.hidden_field :url, :required => true %> 6 | <%= f.input :file do %> 7 | <%= f.file_field :files, :id => "product_file", :accept => ".pdf,.mobi,.epub" %> 8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 | <% end %> 20 |
21 | <%= f.button :submit, :id => "new_product_submit", :class => "btn btn-primary" %> 22 |
23 | <% end %> -------------------------------------------------------------------------------- /config.ru: -------------------------------------------------------------------------------- 1 | # This file is used by Rack-based servers to start the application. 2 | 3 | require ::File.expand_path('../config/environment', __FILE__) 4 | run Rails.application 5 | -------------------------------------------------------------------------------- /config/application.rb: -------------------------------------------------------------------------------- 1 | require File.expand_path('../boot', __FILE__) 2 | 3 | require 'rails/all' 4 | 5 | # Require the gems listed in Gemfile, including any gems 6 | # you've limited to :test, :development, or :production. 7 | Bundler.require(*Rails.groups) 8 | 9 | module Blog 10 | class Application < Rails::Application 11 | # Settings in config/environments/* take precedence over those specified here. 12 | # Application configuration should go into files in config/initializers 13 | # -- all .rb files in that directory are automatically loaded. 14 | 15 | # Set Time.zone default to the specified zone and make Active Record auto-convert to this zone. 16 | # Run "rake -D time" for a list of tasks for finding time zone names. Default is UTC. 17 | # config.time_zone = 'Central Time (US & Canada)' 18 | 19 | # The default locale is :en and all translations from config/locales/*.rb,yml are auto loaded. 20 | # config.i18n.load_path += Dir[Rails.root.join('my', 'locales', '*.{rb,yml}').to_s] 21 | # config.i18n.default_locale = :de 22 | 23 | # Do not swallow errors in after_commit/after_rollback callbacks. 24 | config.active_record.raise_in_transactional_callbacks = true 25 | end 26 | end 27 | -------------------------------------------------------------------------------- /config/boot.rb: -------------------------------------------------------------------------------- 1 | ENV['BUNDLE_GEMFILE'] ||= File.expand_path('../../Gemfile', __FILE__) 2 | 3 | require 'bundler/setup' # Set up gems listed in the Gemfile. 4 | -------------------------------------------------------------------------------- /config/database.yml: -------------------------------------------------------------------------------- 1 | # SQLite version 3.x 2 | # gem install sqlite3 3 | # 4 | # Ensure the SQLite 3 gem is defined in your Gemfile 5 | # gem 'sqlite3' 6 | development: 7 | adapter: sqlite3 8 | database: db/development.sqlite3 9 | pool: 5 10 | timeout: 5000 11 | 12 | # Warning: The database defined as "test" will be erased and 13 | # re-generated from your development database when you run "rake". 14 | # Do not set this db to the same as development or production. 15 | test: 16 | adapter: sqlite3 17 | database: db/test.sqlite3 18 | pool: 5 19 | timeout: 5000 20 | 21 | production: 22 | adapter: sqlite3 23 | database: db/production.sqlite3 24 | pool: 5 25 | timeout: 5000 26 | -------------------------------------------------------------------------------- /config/environment.rb: -------------------------------------------------------------------------------- 1 | # Load the Rails application. 2 | require File.expand_path('../application', __FILE__) 3 | 4 | # Initialize the Rails application. 5 | Rails.application.initialize! 6 | -------------------------------------------------------------------------------- /config/environments/development.rb: -------------------------------------------------------------------------------- 1 | Rails.application.configure do 2 | # Settings specified here will take precedence over those in config/application.rb. 3 | 4 | # In the development environment your application's code is reloaded on 5 | # every request. This slows down response time but is perfect for development 6 | # since you don't have to restart the web server when you make code changes. 7 | config.cache_classes = false 8 | 9 | # Do not eager load code on boot. 10 | config.eager_load = false 11 | 12 | # Show full error reports and disable caching. 13 | config.consider_all_requests_local = true 14 | config.action_controller.perform_caching = false 15 | 16 | # Don't care if the mailer can't send. 17 | config.action_mailer.raise_delivery_errors = false 18 | 19 | # Print deprecation notices to the Rails logger. 20 | config.active_support.deprecation = :log 21 | 22 | # Raise an error on page load if there are pending migrations. 23 | config.active_record.migration_error = :page_load 24 | 25 | # Debug mode disables concatenation and preprocessing of assets. 26 | # This option may cause significant delays in view rendering with a large 27 | # number of complex assets. 28 | config.assets.debug = true 29 | 30 | # Asset digests allow you to set far-future HTTP expiration dates on all assets, 31 | # yet still be able to expire them through the digest params. 32 | config.assets.digest = true 33 | 34 | # Adds additional error checking when serving assets at runtime. 35 | # Checks for improperly declared sprockets dependencies. 36 | # Raises helpful error messages. 37 | config.assets.raise_runtime_errors = true 38 | 39 | # Raises error for missing translations 40 | # config.action_view.raise_on_missing_translations = true 41 | end 42 | -------------------------------------------------------------------------------- /config/environments/production.rb: -------------------------------------------------------------------------------- 1 | Rails.application.configure do 2 | # Settings specified here will take precedence over those in config/application.rb. 3 | 4 | # Code is not reloaded between requests. 5 | config.cache_classes = true 6 | 7 | # Eager load code on boot. This eager loads most of Rails and 8 | # your application in memory, allowing both threaded web servers 9 | # and those relying on copy on write to perform better. 10 | # Rake tasks automatically ignore this option for performance. 11 | config.eager_load = true 12 | 13 | # Full error reports are disabled and caching is turned on. 14 | config.consider_all_requests_local = false 15 | config.action_controller.perform_caching = true 16 | 17 | # Enable Rack::Cache to put a simple HTTP cache in front of your application 18 | # Add `rack-cache` to your Gemfile before enabling this. 19 | # For large-scale production use, consider using a caching reverse proxy like 20 | # NGINX, varnish or squid. 21 | # config.action_dispatch.rack_cache = true 22 | 23 | # Disable serving static files from the `/public` folder by default since 24 | # Apache or NGINX already handles this. 25 | config.serve_static_files = ENV['RAILS_SERVE_STATIC_FILES'].present? 26 | 27 | # Compress JavaScripts and CSS. 28 | config.assets.js_compressor = :uglifier 29 | # config.assets.css_compressor = :sass 30 | 31 | # Do not fallback to assets pipeline if a precompiled asset is missed. 32 | config.assets.compile = false 33 | 34 | # Asset digests allow you to set far-future HTTP expiration dates on all assets, 35 | # yet still be able to expire them through the digest params. 36 | config.assets.digest = true 37 | 38 | # `config.assets.precompile` and `config.assets.version` have moved to config/initializers/assets.rb 39 | 40 | # Specifies the header that your server uses for sending files. 41 | # config.action_dispatch.x_sendfile_header = 'X-Sendfile' # for Apache 42 | # config.action_dispatch.x_sendfile_header = 'X-Accel-Redirect' # for NGINX 43 | 44 | # Force all access to the app over SSL, use Strict-Transport-Security, and use secure cookies. 45 | # config.force_ssl = true 46 | 47 | # Use the lowest log level to ensure availability of diagnostic information 48 | # when problems arise. 49 | config.log_level = :debug 50 | 51 | # Prepend all log lines with the following tags. 52 | # config.log_tags = [ :subdomain, :uuid ] 53 | 54 | # Use a different logger for distributed setups. 55 | # config.logger = ActiveSupport::TaggedLogging.new(SyslogLogger.new) 56 | 57 | # Use a different cache store in production. 58 | # config.cache_store = :mem_cache_store 59 | 60 | # Enable serving of images, stylesheets, and JavaScripts from an asset server. 61 | # config.action_controller.asset_host = 'http://assets.example.com' 62 | 63 | # Ignore bad email addresses and do not raise email delivery errors. 64 | # Set this to true and configure the email server for immediate delivery to raise delivery errors. 65 | # config.action_mailer.raise_delivery_errors = false 66 | 67 | # Enable locale fallbacks for I18n (makes lookups for any locale fall back to 68 | # the I18n.default_locale when a translation cannot be found). 69 | config.i18n.fallbacks = true 70 | 71 | # Send deprecation notices to registered listeners. 72 | config.active_support.deprecation = :notify 73 | 74 | # Use default logging formatter so that PID and timestamp are not suppressed. 75 | config.log_formatter = ::Logger::Formatter.new 76 | 77 | # Do not dump schema after migrations. 78 | config.active_record.dump_schema_after_migration = false 79 | end 80 | -------------------------------------------------------------------------------- /config/environments/test.rb: -------------------------------------------------------------------------------- 1 | Rails.application.configure do 2 | # Settings specified here will take precedence over those in config/application.rb. 3 | 4 | # The test environment is used exclusively to run your application's 5 | # test suite. You never need to work with it otherwise. Remember that 6 | # your test database is "scratch space" for the test suite and is wiped 7 | # and recreated between test runs. Don't rely on the data there! 8 | config.cache_classes = true 9 | 10 | # Do not eager load code on boot. This avoids loading your whole application 11 | # just for the purpose of running a single test. If you are using a tool that 12 | # preloads Rails for running tests, you may have to set it to true. 13 | config.eager_load = false 14 | 15 | # Configure static file server for tests with Cache-Control for performance. 16 | config.serve_static_files = true 17 | config.static_cache_control = 'public, max-age=3600' 18 | 19 | # Show full error reports and disable caching. 20 | config.consider_all_requests_local = true 21 | config.action_controller.perform_caching = false 22 | 23 | # Raise exceptions instead of rendering exception templates. 24 | config.action_dispatch.show_exceptions = false 25 | 26 | # Disable request forgery protection in test environment. 27 | config.action_controller.allow_forgery_protection = false 28 | 29 | # Tell Action Mailer not to deliver emails to the real world. 30 | # The :test delivery method accumulates sent emails in the 31 | # ActionMailer::Base.deliveries array. 32 | config.action_mailer.delivery_method = :test 33 | 34 | # Randomize the order test cases are executed. 35 | config.active_support.test_order = :random 36 | 37 | # Print deprecation notices to the stderr. 38 | config.active_support.deprecation = :stderr 39 | 40 | # Raises error for missing translations 41 | # config.action_view.raise_on_missing_translations = true 42 | end 43 | -------------------------------------------------------------------------------- /config/initializers/aws.rb: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/config/initializers/aws.rb -------------------------------------------------------------------------------- /config/initializers/backtrace_silencers.rb: -------------------------------------------------------------------------------- 1 | # Be sure to restart your server when you modify this file. 2 | 3 | # You can add backtrace silencers for libraries that you're using but don't wish to see in your backtraces. 4 | # Rails.backtrace_cleaner.add_silencer { |line| line =~ /my_noisy_library/ } 5 | 6 | # You can also remove all the silencers if you're trying to debug a problem that might stem from framework code. 7 | # Rails.backtrace_cleaner.remove_silencers! 8 | -------------------------------------------------------------------------------- /config/initializers/inflections.rb: -------------------------------------------------------------------------------- 1 | # Be sure to restart your server when you modify this file. 2 | 3 | # Add new inflection rules using the following format 4 | # (all these examples are active by default): 5 | # ActiveSupport::Inflector.inflections do |inflect| 6 | # inflect.plural /^(ox)$/i, '\1en' 7 | # inflect.singular /^(ox)en/i, '\1' 8 | # inflect.irregular 'person', 'people' 9 | # inflect.uncountable %w( fish sheep ) 10 | # end 11 | # 12 | # These inflection rules are supported but not enabled by default: 13 | # ActiveSupport::Inflector.inflections do |inflect| 14 | # inflect.acronym 'RESTful' 15 | # end 16 | -------------------------------------------------------------------------------- /config/initializers/mime_types.rb: -------------------------------------------------------------------------------- 1 | # Be sure to restart your server when you modify this file. 2 | 3 | # Add new mime types for use in respond_to blocks: 4 | # Mime::Type.register "text/richtext", :rtf 5 | # Mime::Type.register_alias "text/html", :iphone 6 | -------------------------------------------------------------------------------- /config/initializers/session_store.rb: -------------------------------------------------------------------------------- 1 | # Be sure to restart your server when you modify this file. 2 | 3 | Rails.application.config.session_store :cookie_store, key: '_s3-cors-upload-rails_session' 4 | -------------------------------------------------------------------------------- /config/initializers/simple_form.rb: -------------------------------------------------------------------------------- 1 | # Use this setup block to configure all options available in SimpleForm. 2 | SimpleForm.setup do |config| 3 | # Wrappers are used by the form builder to generate a 4 | # complete input. You can remove any component from the 5 | # wrapper, change the order or even add your own to the 6 | # stack. The options given below are used to wrap the 7 | # whole input. 8 | config.wrappers :default, :class => :input, 9 | :hint_class => :field_with_hint, :error_class => :field_with_errors do |b| 10 | ## Extensions enabled by default 11 | # Any of these extensions can be disabled for a 12 | # given input by passing: `f.input EXTENSION_NAME => false`. 13 | # You can make any of these extensions optional by 14 | # renaming `b.use` to `b.optional`. 15 | 16 | # Determines whether to use HTML5 (:email, :url, ...) 17 | # and required attributes 18 | b.use :html5 19 | 20 | # Calculates placeholders automatically from I18n 21 | # You can also pass a string as f.input :placeholder => "Placeholder" 22 | b.use :placeholder 23 | 24 | ## Optional extensions 25 | # They are disabled unless you pass `f.input EXTENSION_NAME => :lookup` 26 | # to the input. If so, they will retrieve the values from the model 27 | # if any exists. If you want to enable the lookup for any of those 28 | # extensions by default, you can change `b.optional` to `b.use`. 29 | 30 | # Calculates maxlength from length validations for string inputs 31 | b.optional :maxlength 32 | 33 | # Calculates pattern from format validations for string inputs 34 | b.optional :pattern 35 | 36 | # Calculates min and max from length validations for numeric inputs 37 | b.optional :min_max 38 | 39 | # Calculates readonly automatically from readonly attributes 40 | b.optional :readonly 41 | 42 | ## Inputs 43 | b.use :label_input 44 | b.use :hint, :wrap_with => { :tag => :span, :class => :hint } 45 | b.use :error, :wrap_with => { :tag => :span, :class => :error } 46 | end 47 | 48 | # The default wrapper to be used by the FormBuilder. 49 | config.default_wrapper = :default 50 | 51 | # Define the way to render check boxes / radio buttons with labels. 52 | # Defaults to :nested for bootstrap config. 53 | # :inline => input + label 54 | # :nested => label > input 55 | config.boolean_style = :nested 56 | 57 | # Default class for buttons 58 | config.button_class = 'btn' 59 | 60 | # Method used to tidy up errors. Specify any Rails Array method. 61 | # :first lists the first message for each field. 62 | # Use :to_sentence to list all errors for each field. 63 | # config.error_method = :first 64 | 65 | # Default tag used for error notification helper. 66 | config.error_notification_tag = :div 67 | 68 | # CSS class to add for error notification helper. 69 | config.error_notification_class = 'alert alert-error' 70 | 71 | # ID to add for error notification helper. 72 | # config.error_notification_id = nil 73 | 74 | # Series of attempts to detect a default label method for collection. 75 | # config.collection_label_methods = [ :to_label, :name, :title, :to_s ] 76 | 77 | # Series of attempts to detect a default value method for collection. 78 | # config.collection_value_methods = [ :id, :to_s ] 79 | 80 | # You can wrap a collection of radio/check boxes in a pre-defined tag, defaulting to none. 81 | # config.collection_wrapper_tag = nil 82 | 83 | # You can define the class to use on all collection wrappers. Defaulting to none. 84 | # config.collection_wrapper_class = nil 85 | 86 | # You can wrap each item in a collection of radio/check boxes with a tag, 87 | # defaulting to :span. Please note that when using :boolean_style = :nested, 88 | # SimpleForm will force this option to be a label. 89 | # config.item_wrapper_tag = :span 90 | 91 | # You can define a class to use in all item wrappers. Defaulting to none. 92 | # config.item_wrapper_class = nil 93 | 94 | # How the label text should be generated altogether with the required text. 95 | # config.label_text = lambda { |label, required| "#{required} #{label}" } 96 | 97 | # You can define the class to use on all labels. Default is nil. 98 | config.label_class = 'control-label' 99 | 100 | # You can define the class to use on all forms. Default is simple_form. 101 | # config.form_class = :simple_form 102 | 103 | # You can define which elements should obtain additional classes 104 | # config.generate_additional_classes_for = [:wrapper, :label, :input] 105 | 106 | # Whether attributes are required by default (or not). Default is true. 107 | # config.required_by_default = true 108 | 109 | # Tell browsers whether to use default HTML5 validations (novalidate option). 110 | # Default is enabled. 111 | config.browser_validations = true 112 | 113 | # Collection of methods to detect if a file type was given. 114 | # config.file_methods = [ :mounted_as, :file?, :public_filename ] 115 | 116 | # Custom mappings for input types. This should be a hash containing a regexp 117 | # to match as key, and the input type that will be used when the field name 118 | # matches the regexp as value. 119 | # config.input_mappings = { /count/ => :integer } 120 | 121 | # Custom wrappers for input types. This should be a hash containing an input 122 | # type as key and the wrapper that will be used for all inputs with specified type. 123 | # config.wrapper_mappings = { :string => :prepend } 124 | 125 | # Default priority for time_zone inputs. 126 | # config.time_zone_priority = nil 127 | 128 | # Default priority for country inputs. 129 | # config.country_priority = nil 130 | 131 | # Default size for text inputs. 132 | # config.default_input_size = 50 133 | 134 | # When false, do not use translations for labels. 135 | # config.translate_labels = true 136 | 137 | # Automatically discover new inputs in Rails' autoload path. 138 | # config.inputs_discovery = true 139 | 140 | # Cache SimpleForm inputs discovery 141 | # config.cache_discovery = !Rails.env.development? 142 | end 143 | -------------------------------------------------------------------------------- /config/initializers/simple_form_bootstrap.rb: -------------------------------------------------------------------------------- 1 | # Use this setup block to configure all options available in SimpleForm. 2 | SimpleForm.setup do |config| 3 | config.wrappers :bootstrap, :tag => 'div', :class => 'control-group', :error_class => 'error' do |b| 4 | b.use :html5 5 | b.use :placeholder 6 | b.use :label 7 | b.wrapper :tag => 'div', :class => 'controls' do |ba| 8 | ba.use :input 9 | ba.use :error, :wrap_with => { :tag => 'span', :class => 'help-inline' } 10 | ba.use :hint, :wrap_with => { :tag => 'p', :class => 'help-block' } 11 | end 12 | end 13 | 14 | config.wrappers :prepend, :tag => 'div', :class => "control-group", :error_class => 'error' do |b| 15 | b.use :html5 16 | b.use :placeholder 17 | b.use :label 18 | b.wrapper :tag => 'div', :class => 'controls' do |input| 19 | input.wrapper :tag => 'div', :class => 'input-prepend' do |prepend| 20 | prepend.use :input 21 | end 22 | input.use :hint, :wrap_with => { :tag => 'span', :class => 'help-block' } 23 | input.use :error, :wrap_with => { :tag => 'span', :class => 'help-inline' } 24 | end 25 | end 26 | 27 | config.wrappers :append, :tag => 'div', :class => "control-group", :error_class => 'error' do |b| 28 | b.use :html5 29 | b.use :placeholder 30 | b.use :label 31 | b.wrapper :tag => 'div', :class => 'controls' do |input| 32 | input.wrapper :tag => 'div', :class => 'input-append' do |append| 33 | append.use :input 34 | end 35 | input.use :hint, :wrap_with => { :tag => 'span', :class => 'help-block' } 36 | input.use :error, :wrap_with => { :tag => 'span', :class => 'help-inline' } 37 | end 38 | end 39 | 40 | # Wrappers for forms and inputs using the Twitter Bootstrap toolkit. 41 | # Check the Bootstrap docs (http://twitter.github.com/bootstrap) 42 | # to learn about the different styles for forms and inputs, 43 | # buttons and other elements. 44 | config.default_wrapper = :bootstrap 45 | end 46 | -------------------------------------------------------------------------------- /config/initializers/wrap_parameters.rb: -------------------------------------------------------------------------------- 1 | # Be sure to restart your server when you modify this file. 2 | 3 | # This file contains settings for ActionController::ParamsWrapper which 4 | # is enabled by default. 5 | 6 | # Enable parameter wrapping for JSON. You can disable this by setting :format to an empty array. 7 | ActiveSupport.on_load(:action_controller) do 8 | wrap_parameters format: [:json] if respond_to?(:wrap_parameters) 9 | end 10 | 11 | # To enable root element in JSON for ActiveRecord objects. 12 | # ActiveSupport.on_load(:active_record) do 13 | # self.include_root_in_json = true 14 | # end 15 | -------------------------------------------------------------------------------- /config/locales/en.yml: -------------------------------------------------------------------------------- 1 | # Sample localization file for English. Add more files in this directory for other locales. 2 | # See https://github.com/svenfuchs/rails-i18n/tree/master/rails%2Flocale for starting points. 3 | 4 | en: 5 | hello: "Hello world" 6 | -------------------------------------------------------------------------------- /config/locales/simple_form.en.yml: -------------------------------------------------------------------------------- 1 | en: 2 | simple_form: 3 | "yes": 'Yes' 4 | "no": 'No' 5 | required: 6 | text: 'required' 7 | mark: '*' 8 | # You can uncomment the line below if you need to overwrite the whole required html. 9 | # When using html, text and mark won't be used. 10 | # html: '*' 11 | error_notification: 12 | default_message: "Please review the problems below:" 13 | # Labels and hints examples 14 | # labels: 15 | # defaults: 16 | # password: 'Password' 17 | # user: 18 | # new: 19 | # email: 'E-mail to sign in.' 20 | # edit: 21 | # email: 'E-mail.' 22 | # hints: 23 | # defaults: 24 | # username: 'User name to sign in.' 25 | # password: 'No special characters, please.' 26 | 27 | -------------------------------------------------------------------------------- /config/routes.rb: -------------------------------------------------------------------------------- 1 | Rails.application.routes.draw do 2 | 3 | root :to => "products#new" 4 | resources :products 5 | 6 | get "generate_signed_s3_url" => "aws#generate_signed_s3_url" 7 | 8 | end 9 | -------------------------------------------------------------------------------- /config/secrets.yml: -------------------------------------------------------------------------------- 1 | # Be sure to restart your server when you modify this file. 2 | 3 | # Your secret key is used for verifying the integrity of signed cookies. 4 | # If you change this key, all old signed cookies will become invalid! 5 | 6 | # Make sure the secret is at least 30 characters and all random, 7 | # no regular words or you'll be exposed to dictionary attacks. 8 | # You can use `rake secret` to generate a secure secret key. 9 | 10 | # Make sure the secrets in this file are kept private 11 | # if you're sharing your code publicly. 12 | 13 | development: 14 | secret_key_base: aff6a0e568e022bd28f4112c3ca7bdcf6605079a78df0f2ec7410712434b509ecaddb2d40602fcce5e38304c043247d8429d2a5279330f94b915fbfe6053d453 15 | 16 | test: 17 | secret_key_base: ad3d15f0af35b0dfebebfcce513d07e3c1800e663266a56469ec909019c8d55d4b00b2fc987ac9ff8d982add98a95ac970de2566b237299b428ac38b7c494e62 18 | 19 | # Do not keep production secrets in the repository, 20 | # instead read values from the environment. 21 | production: 22 | secret_key_base: <%= ENV["SECRET_KEY_BASE"] %> 23 | -------------------------------------------------------------------------------- /db/migrate/20130523082752_create_products.rb: -------------------------------------------------------------------------------- 1 | class CreateProducts < ActiveRecord::Migration 2 | def change 3 | create_table :products do |t| 4 | t.string :name 5 | t.decimal :price 6 | t.string :url 7 | 8 | t.timestamps 9 | end 10 | end 11 | end 12 | -------------------------------------------------------------------------------- /db/schema.rb: -------------------------------------------------------------------------------- 1 | # encoding: UTF-8 2 | # This file is auto-generated from the current state of the database. Instead 3 | # of editing this file, please use the migrations feature of Active Record to 4 | # incrementally modify your database, and then regenerate this schema definition. 5 | # 6 | # Note that this schema.rb definition is the authoritative source for your 7 | # database schema. If you need to create the application database on another 8 | # system, you should be using db:schema:load, not running all the migrations 9 | # from scratch. The latter is a flawed and unsustainable approach (the more migrations 10 | # you'll amass, the slower it'll run and the greater likelihood for issues). 11 | # 12 | # It's strongly recommended that you check this file into your version control system. 13 | 14 | ActiveRecord::Schema.define(version: 20130523082752) do 15 | 16 | create_table "products", force: :cascade do |t| 17 | t.string "name" 18 | t.decimal "price" 19 | t.string "url" 20 | t.datetime "created_at" 21 | t.datetime "updated_at" 22 | end 23 | 24 | end 25 | -------------------------------------------------------------------------------- /db/seeds.rb: -------------------------------------------------------------------------------- 1 | # This file should contain all the record creation needed to seed the database with its default values. 2 | # The data can then be loaded with the rake db:seed (or created alongside the db with db:setup). 3 | # 4 | # Examples: 5 | # 6 | # cities = City.create([{ name: 'Chicago' }, { name: 'Copenhagen' }]) 7 | # Mayor.create(name: 'Emanuel', city: cities.first) 8 | -------------------------------------------------------------------------------- /doc/README_FOR_APP: -------------------------------------------------------------------------------- 1 | Use this README file to introduce your application and point to useful places in the API for learning more. 2 | Run "rake doc:app" to generate API documentation for your models, controllers, helpers, and libraries. 3 | -------------------------------------------------------------------------------- /log/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/log/.gitkeep -------------------------------------------------------------------------------- /public/404.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | The page you were looking for doesn't exist (404) 5 | 17 | 18 | 19 | 20 | 21 |
22 |

The page you were looking for doesn't exist.

23 |

You may have mistyped the address or the page may have moved.

24 |
25 | 26 | 27 | -------------------------------------------------------------------------------- /public/422.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | The change you wanted was rejected (422) 5 | 17 | 18 | 19 | 20 | 21 |
22 |

The change you wanted was rejected.

23 |

Maybe you tried to change something you didn't have access to.

24 |
25 | 26 | 27 | -------------------------------------------------------------------------------- /public/500.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | We're sorry, but something went wrong (500) 5 | 17 | 18 | 19 | 20 | 21 |
22 |

We're sorry, but something went wrong.

23 |
24 | 25 | 26 | -------------------------------------------------------------------------------- /public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/public/favicon.ico -------------------------------------------------------------------------------- /public/img/glyphicons-halflings-white.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/public/img/glyphicons-halflings-white.png -------------------------------------------------------------------------------- /public/img/glyphicons-halflings.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/public/img/glyphicons-halflings.png -------------------------------------------------------------------------------- /public/robots.txt: -------------------------------------------------------------------------------- 1 | # See http://www.robotstxt.org/wc/norobots.html for documentation on how to use the robots.txt file 2 | # 3 | # To ban all spiders from the entire site uncomment the next two lines: 4 | # User-Agent: * 5 | # Disallow: / 6 | -------------------------------------------------------------------------------- /script/rails: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env ruby 2 | # This command will automatically be run when you run "rails" with Rails 3 gems installed from the root of your application. 3 | 4 | APP_PATH = File.expand_path('../../config/application', __FILE__) 5 | require File.expand_path('../../config/boot', __FILE__) 6 | require 'rails/commands' 7 | -------------------------------------------------------------------------------- /spec/controllers/aws_controller_spec.rb: -------------------------------------------------------------------------------- 1 | require 'spec_helper' 2 | 3 | describe AwsController do 4 | 5 | end 6 | -------------------------------------------------------------------------------- /spec/controllers/products_controller_spec.rb: -------------------------------------------------------------------------------- 1 | require 'spec_helper' 2 | 3 | describe ProductsController do 4 | 5 | end 6 | -------------------------------------------------------------------------------- /spec/helpers/products_helper_spec.rb: -------------------------------------------------------------------------------- 1 | require 'spec_helper' 2 | 3 | # Specs in this file have access to a helper object that includes 4 | # the ProductsHelper. For example: 5 | # 6 | # describe ProductsHelper do 7 | # describe "string concat" do 8 | # it "concats two strings with spaces" do 9 | # helper.concat_strings("this","that").should == "this that" 10 | # end 11 | # end 12 | # end 13 | describe ProductsHelper do 14 | pending "add some examples to (or delete) #{__FILE__}" 15 | end 16 | -------------------------------------------------------------------------------- /spec/spec_helper.rb: -------------------------------------------------------------------------------- 1 | # This file is copied to spec/ when you run 'rails generate rspec:install' 2 | ENV["RAILS_ENV"] ||= 'test' 3 | require File.expand_path("../../config/environment", __FILE__) 4 | require 'rspec/rails' 5 | 6 | # Requires supporting ruby files with custom matchers and macros, etc, 7 | # in spec/support/ and its subdirectories. 8 | Dir[Rails.root.join("spec/support/**/*.rb")].each {|f| require f} 9 | 10 | RSpec.configure do |config| 11 | # ## Mock Framework 12 | # 13 | # If you prefer to use mocha, flexmock or RR, uncomment the appropriate line: 14 | # 15 | # config.mock_with :mocha 16 | # config.mock_with :flexmock 17 | # config.mock_with :rr 18 | 19 | # Remove this line if you're not using ActiveRecord or ActiveRecord fixtures 20 | config.fixture_path = "#{::Rails.root}/spec/fixtures" 21 | 22 | # If you're not using ActiveRecord, or you'd prefer not to run each of your 23 | # examples within a transaction, remove the following line or assign false 24 | # instead of true. 25 | config.use_transactional_fixtures = true 26 | 27 | # If true, the base class of anonymous controllers will be inferred 28 | # automatically. This will be the default behavior in future versions of 29 | # rspec-rails. 30 | config.infer_base_class_for_anonymous_controllers = false 31 | 32 | # Run specs in random order to surface order dependencies. If you find an 33 | # order dependency and want to debug it, you can fix the order by providing 34 | # the seed, which is printed after each run. 35 | # --seed 1234 36 | config.order = "random" 37 | end 38 | -------------------------------------------------------------------------------- /test/fixtures/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/test/fixtures/.gitkeep -------------------------------------------------------------------------------- /test/fixtures/products.yml: -------------------------------------------------------------------------------- 1 | # Read about fixtures at http://api.rubyonrails.org/classes/ActiveRecord/Fixtures.html 2 | 3 | one: 4 | name: MyString 5 | price: 9.99 6 | url: MyString 7 | 8 | two: 9 | name: MyString 10 | price: 9.99 11 | url: MyString 12 | -------------------------------------------------------------------------------- /test/functional/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/test/functional/.gitkeep -------------------------------------------------------------------------------- /test/integration/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/test/integration/.gitkeep -------------------------------------------------------------------------------- /test/performance/browsing_test.rb: -------------------------------------------------------------------------------- 1 | require 'test_helper' 2 | require 'rails/performance_test_help' 3 | 4 | class BrowsingTest < ActionDispatch::PerformanceTest 5 | # Refer to the documentation for all available options 6 | # self.profile_options = { :runs => 5, :metrics => [:wall_time, :memory] 7 | # :output => 'tmp/performance', :formats => [:flat] } 8 | 9 | def test_homepage 10 | get '/' 11 | end 12 | end 13 | -------------------------------------------------------------------------------- /test/test_helper.rb: -------------------------------------------------------------------------------- 1 | ENV["RAILS_ENV"] = "test" 2 | require File.expand_path('../../config/environment', __FILE__) 3 | require 'rails/test_help' 4 | 5 | class ActiveSupport::TestCase 6 | # Setup all fixtures in test/fixtures/*.(yml|csv) for all tests in alphabetical order. 7 | # 8 | # Note: You'll currently still have to declare fixtures explicitly in integration tests 9 | # -- they do not yet inherit this setting 10 | fixtures :all 11 | 12 | # Add more helper methods to be used by all tests here... 13 | end 14 | -------------------------------------------------------------------------------- /test/unit/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/test/unit/.gitkeep -------------------------------------------------------------------------------- /test/unit/product_test.rb: -------------------------------------------------------------------------------- 1 | require 'test_helper' 2 | 3 | class ProductTest < ActiveSupport::TestCase 4 | # test "the truth" do 5 | # assert true 6 | # end 7 | end 8 | -------------------------------------------------------------------------------- /vendor/assets/javascripts/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/bparanj/s3-cors-upload-rails/b5d5b6c1783aa2b9e9891e6184e97710db5512c2/vendor/assets/javascripts/.gitkeep -------------------------------------------------------------------------------- /vendor/assets/javascripts/bootstrap-filestyle.js: -------------------------------------------------------------------------------- 1 | /* 2 | * bootstrap-filestyle 3 | * http://dev.tudosobreweb.com.br/bootstrap-filestyle/ 4 | * 5 | * Copyright (c) 2013 Markus Vinicius da Silva Lima 6 | * Version 1.0.3 7 | * Licensed under the MIT license. 8 | */ 9 | (function ($) { 10 | "use strict"; 11 | 12 | var Filestyle = function (element, options) { 13 | this.options = options; 14 | this.$elementFilestyle = []; 15 | this.$element = $(element); 16 | }; 17 | 18 | Filestyle.prototype = { 19 | clear: function () { 20 | this.$element.val(''); 21 | this.$elementFilestyle.find(':text').val(''); 22 | }, 23 | 24 | destroy: function () { 25 | this.$element 26 | .removeAttr('style') 27 | .removeData('filestyle') 28 | .val(''); 29 | this.$elementFilestyle.remove(); 30 | }, 31 | 32 | icon: function (value) { 33 | if (value === true) { 34 | if (!this.options.icon) { 35 | this.options.icon = true; 36 | this.$elementFilestyle.find('label').prepend(this.htmlIcon()); 37 | } 38 | } else if (value === false) { 39 | if (this.options.icon) { 40 | this.options.icon = false; 41 | this.$elementFilestyle.find('i').remove(); 42 | } 43 | } else { 44 | return this.options.icon; 45 | } 46 | }, 47 | 48 | input: function (value) { 49 | if (value === true) { 50 | if (!this.options.input) { 51 | this.options.input = true; 52 | this.$elementFilestyle.prepend(this.htmlInput()); 53 | 54 | var content = '', 55 | files = []; 56 | if (this.$element[0].files === undefined) { 57 | files[0] = {'name': this.$element[0].value}; 58 | } else { 59 | files = this.$element[0].files; 60 | } 61 | 62 | for (var i = 0; i < files.length; i++) { 63 | content += files[i].name.split("\\").pop() + ', '; 64 | } 65 | if (content !== '') { 66 | this.$elementFilestyle.find(':text').val(content.replace(/\, $/g, '')); 67 | } 68 | } 69 | } else if (value === false) { 70 | if (this.options.input) { 71 | this.options.input = false; 72 | this.$elementFilestyle.find(':text').remove(); 73 | } 74 | } else { 75 | return this.options.input; 76 | } 77 | }, 78 | 79 | buttonText: function (value) { 80 | if (value !== undefined) { 81 | this.options.buttonText = value; 82 | this.$elementFilestyle.find('label span').html(this.options.buttonText); 83 | } else { 84 | return this.options.buttonText; 85 | } 86 | }, 87 | 88 | classButton: function (value) { 89 | if (value !== undefined) { 90 | this.options.classButton = value; 91 | this.$elementFilestyle.find('label').attr({'class': this.options.classButton}); 92 | if (this.options.classButton.search(/btn-inverse|btn-primary|btn-danger|btn-warning|btn-success/i) !== -1) { 93 | this.$elementFilestyle.find('label i').addClass('icon-white'); 94 | } else { 95 | this.$elementFilestyle.find('label i').removeClass('icon-white'); 96 | } 97 | } else { 98 | return this.options.classButton; 99 | } 100 | }, 101 | 102 | classIcon: function (value) { 103 | if (value !== undefined) { 104 | this.options.classIcon = value; 105 | if (this.options.classButton.search(/btn-inverse|btn-primary|btn-danger|btn-warning|btn-success/i) !== -1) { 106 | this.$elementFilestyle.find('label').find('i').attr({'class': 'icon-white '+this.options.classIcon}); 107 | } else { 108 | this.$elementFilestyle.find('label').find('i').attr({'class': this.options.classIcon}); 109 | } 110 | } else { 111 | return this.options.classIcon; 112 | } 113 | }, 114 | 115 | classInput: function (value) { 116 | if (value !== undefined) { 117 | this.options.classInput = value; 118 | this.$elementFilestyle.find(':text').addClass(this.options.classInput); 119 | } else { 120 | return this.options.classInput; 121 | } 122 | }, 123 | 124 | htmlIcon: function () { 125 | if (this.options.icon) { 126 | var colorIcon = ''; 127 | if (this.options.classButton.search(/btn-inverse|btn-primary|btn-danger|btn-warning|btn-success/i) !== -1) { 128 | colorIcon = ' icon-white '; 129 | } 130 | 131 | return ' '; 132 | } else { 133 | return ''; 134 | } 135 | }, 136 | 137 | htmlInput: function () { 138 | if (this.options.input) { 139 | return ' '; 140 | } else { 141 | return ''; 142 | } 143 | }, 144 | 145 | constructor: function () { 146 | var _self = this, 147 | html = '', 148 | id = this.$element.attr('id'), 149 | files = []; 150 | 151 | if (id === '' || !id) { 152 | id = 'filestyle-'+$('.bootstrap-filestyle').length; 153 | this.$element.attr({'id': id}); 154 | } 155 | 156 | html = this.htmlInput()+ 157 | ''; 161 | 162 | this.$elementFilestyle = $('
'+html+'
'); 163 | 164 | // hidding input file and add filestyle 165 | this.$element 166 | .css({'position':'fixed','left':'-500px'}) 167 | .after(this.$elementFilestyle); 168 | 169 | // Getting input file value 170 | this.$element.change(function () { 171 | var content = ''; 172 | if (this.files === undefined) { 173 | files[0] = {'name': this.value}; 174 | } else { 175 | files = this.files; 176 | } 177 | 178 | for (var i = 0; i < files.length; i++) { 179 | content += files[i].name.split("\\").pop() + ', '; 180 | } 181 | 182 | if (content !== '') { 183 | _self.$elementFilestyle.find(':text').val(content.replace(/\, $/g, '')); 184 | } 185 | }); 186 | 187 | // Check if browser is Firefox 188 | if (window.navigator.userAgent.search(/firefox/i) > -1) { 189 | // Simulating choose file for firefox 190 | this.$elementFilestyle.find('label').click(function () { 191 | _self.$element.click(); 192 | return false; 193 | }); 194 | } 195 | } 196 | }; 197 | 198 | var old = $.fn.filestyle; 199 | 200 | $.fn.filestyle = function (option, value) { 201 | var get = '', 202 | element = this.each(function () { 203 | if ($(this).attr('type') === 'file') { 204 | var $this = $(this), 205 | data = $this.data('filestyle'), 206 | options = $.extend({}, $.fn.filestyle.defaults, option, typeof option === 'object' && option); 207 | 208 | if (!data) { 209 | $this.data('filestyle', (data = new Filestyle(this, options))); 210 | data.constructor(); 211 | } 212 | 213 | if (typeof option === 'string') { 214 | get = data[option](value); 215 | } 216 | } 217 | }); 218 | 219 | if (typeof get !== undefined) { 220 | return get; 221 | } else { 222 | return element; 223 | } 224 | }; 225 | 226 | $.fn.filestyle.defaults = { 227 | 'buttonText': 'Choose file', 228 | 'input': true, 229 | 'icon': true, 230 | 'classButton': 'btn', 231 | 'classInput': 'input-large', 232 | 'classIcon': 'icon-folder-open' 233 | }; 234 | 235 | $.fn.filestyle.noConflict = function () { 236 | $.fn.filestyle = old; 237 | return this; 238 | }; 239 | 240 | // Data attributes register 241 | $('.filestyle').each(function () { 242 | var $this = $(this), 243 | options = { 244 | 'buttonText': $this.attr('data-buttonText'), 245 | 'input': $this.attr('data-input') === 'false' ? false : true, 246 | 'icon': $this.attr('data-icon') === 'false' ? false : true, 247 | 'classButton': $this.attr('data-classButton'), 248 | 'classInput': $this.attr('data-classInput'), 249 | 'classIcon': $this.attr('data-classIcon') 250 | }; 251 | 252 | $this.filestyle(options); 253 | }); 254 | 255 | })(window.jQuery); -------------------------------------------------------------------------------- /vendor/assets/javascripts/bootstrap.js: -------------------------------------------------------------------------------- 1 | /* =================================================== 2 | * bootstrap-transition.js v2.3.2 3 | * http://twitter.github.com/bootstrap/javascript.html#transitions 4 | * =================================================== 5 | * Copyright 2012 Twitter, Inc. 6 | * 7 | * Licensed under the Apache License, Version 2.0 (the "License"); 8 | * you may not use this file except in compliance with the License. 9 | * You may obtain a copy of the License at 10 | * 11 | * http://www.apache.org/licenses/LICENSE-2.0 12 | * 13 | * Unless required by applicable law or agreed to in writing, software 14 | * distributed under the License is distributed on an "AS IS" BASIS, 15 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 16 | * See the License for the specific language governing permissions and 17 | * limitations under the License. 18 | * ========================================================== */ 19 | 20 | 21 | !function ($) { 22 | 23 | "use strict"; // jshint ;_; 24 | 25 | 26 | /* CSS TRANSITION SUPPORT (http://www.modernizr.com/) 27 | * ======================================================= */ 28 | 29 | $(function () { 30 | 31 | $.support.transition = (function () { 32 | 33 | var transitionEnd = (function () { 34 | 35 | var el = document.createElement('bootstrap') 36 | , transEndEventNames = { 37 | 'WebkitTransition' : 'webkitTransitionEnd' 38 | , 'MozTransition' : 'transitionend' 39 | , 'OTransition' : 'oTransitionEnd otransitionend' 40 | , 'transition' : 'transitionend' 41 | } 42 | , name 43 | 44 | for (name in transEndEventNames){ 45 | if (el.style[name] !== undefined) { 46 | return transEndEventNames[name] 47 | } 48 | } 49 | 50 | }()) 51 | 52 | return transitionEnd && { 53 | end: transitionEnd 54 | } 55 | 56 | })() 57 | 58 | }) 59 | 60 | }(window.jQuery);/* ========================================================== 61 | * bootstrap-alert.js v2.3.2 62 | * http://twitter.github.com/bootstrap/javascript.html#alerts 63 | * ========================================================== 64 | * Copyright 2012 Twitter, Inc. 65 | * 66 | * Licensed under the Apache License, Version 2.0 (the "License"); 67 | * you may not use this file except in compliance with the License. 68 | * You may obtain a copy of the License at 69 | * 70 | * http://www.apache.org/licenses/LICENSE-2.0 71 | * 72 | * Unless required by applicable law or agreed to in writing, software 73 | * distributed under the License is distributed on an "AS IS" BASIS, 74 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 75 | * See the License for the specific language governing permissions and 76 | * limitations under the License. 77 | * ========================================================== */ 78 | 79 | 80 | !function ($) { 81 | 82 | "use strict"; // jshint ;_; 83 | 84 | 85 | /* ALERT CLASS DEFINITION 86 | * ====================== */ 87 | 88 | var dismiss = '[data-dismiss="alert"]' 89 | , Alert = function (el) { 90 | $(el).on('click', dismiss, this.close) 91 | } 92 | 93 | Alert.prototype.close = function (e) { 94 | var $this = $(this) 95 | , selector = $this.attr('data-target') 96 | , $parent 97 | 98 | if (!selector) { 99 | selector = $this.attr('href') 100 | selector = selector && selector.replace(/.*(?=#[^\s]*$)/, '') //strip for ie7 101 | } 102 | 103 | $parent = $(selector) 104 | 105 | e && e.preventDefault() 106 | 107 | $parent.length || ($parent = $this.hasClass('alert') ? $this : $this.parent()) 108 | 109 | $parent.trigger(e = $.Event('close')) 110 | 111 | if (e.isDefaultPrevented()) return 112 | 113 | $parent.removeClass('in') 114 | 115 | function removeElement() { 116 | $parent 117 | .trigger('closed') 118 | .remove() 119 | } 120 | 121 | $.support.transition && $parent.hasClass('fade') ? 122 | $parent.on($.support.transition.end, removeElement) : 123 | removeElement() 124 | } 125 | 126 | 127 | /* ALERT PLUGIN DEFINITION 128 | * ======================= */ 129 | 130 | var old = $.fn.alert 131 | 132 | $.fn.alert = function (option) { 133 | return this.each(function () { 134 | var $this = $(this) 135 | , data = $this.data('alert') 136 | if (!data) $this.data('alert', (data = new Alert(this))) 137 | if (typeof option == 'string') data[option].call($this) 138 | }) 139 | } 140 | 141 | $.fn.alert.Constructor = Alert 142 | 143 | 144 | /* ALERT NO CONFLICT 145 | * ================= */ 146 | 147 | $.fn.alert.noConflict = function () { 148 | $.fn.alert = old 149 | return this 150 | } 151 | 152 | 153 | /* ALERT DATA-API 154 | * ============== */ 155 | 156 | $(document).on('click.alert.data-api', dismiss, Alert.prototype.close) 157 | 158 | }(window.jQuery);/* ============================================================ 159 | * bootstrap-button.js v2.3.2 160 | * http://twitter.github.com/bootstrap/javascript.html#buttons 161 | * ============================================================ 162 | * Copyright 2012 Twitter, Inc. 163 | * 164 | * Licensed under the Apache License, Version 2.0 (the "License"); 165 | * you may not use this file except in compliance with the License. 166 | * You may obtain a copy of the License at 167 | * 168 | * http://www.apache.org/licenses/LICENSE-2.0 169 | * 170 | * Unless required by applicable law or agreed to in writing, software 171 | * distributed under the License is distributed on an "AS IS" BASIS, 172 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 173 | * See the License for the specific language governing permissions and 174 | * limitations under the License. 175 | * ============================================================ */ 176 | 177 | 178 | !function ($) { 179 | 180 | "use strict"; // jshint ;_; 181 | 182 | 183 | /* BUTTON PUBLIC CLASS DEFINITION 184 | * ============================== */ 185 | 186 | var Button = function (element, options) { 187 | this.$element = $(element) 188 | this.options = $.extend({}, $.fn.button.defaults, options) 189 | } 190 | 191 | Button.prototype.setState = function (state) { 192 | var d = 'disabled' 193 | , $el = this.$element 194 | , data = $el.data() 195 | , val = $el.is('input') ? 'val' : 'html' 196 | 197 | state = state + 'Text' 198 | data.resetText || $el.data('resetText', $el[val]()) 199 | 200 | $el[val](data[state] || this.options[state]) 201 | 202 | // push to event loop to allow forms to submit 203 | setTimeout(function () { 204 | state == 'loadingText' ? 205 | $el.addClass(d).attr(d, d) : 206 | $el.removeClass(d).removeAttr(d) 207 | }, 0) 208 | } 209 | 210 | Button.prototype.toggle = function () { 211 | var $parent = this.$element.closest('[data-toggle="buttons-radio"]') 212 | 213 | $parent && $parent 214 | .find('.active') 215 | .removeClass('active') 216 | 217 | this.$element.toggleClass('active') 218 | } 219 | 220 | 221 | /* BUTTON PLUGIN DEFINITION 222 | * ======================== */ 223 | 224 | var old = $.fn.button 225 | 226 | $.fn.button = function (option) { 227 | return this.each(function () { 228 | var $this = $(this) 229 | , data = $this.data('button') 230 | , options = typeof option == 'object' && option 231 | if (!data) $this.data('button', (data = new Button(this, options))) 232 | if (option == 'toggle') data.toggle() 233 | else if (option) data.setState(option) 234 | }) 235 | } 236 | 237 | $.fn.button.defaults = { 238 | loadingText: 'loading...' 239 | } 240 | 241 | $.fn.button.Constructor = Button 242 | 243 | 244 | /* BUTTON NO CONFLICT 245 | * ================== */ 246 | 247 | $.fn.button.noConflict = function () { 248 | $.fn.button = old 249 | return this 250 | } 251 | 252 | 253 | /* BUTTON DATA-API 254 | * =============== */ 255 | 256 | $(document).on('click.button.data-api', '[data-toggle^=button]', function (e) { 257 | var $btn = $(e.target) 258 | if (!$btn.hasClass('btn')) $btn = $btn.closest('.btn') 259 | $btn.button('toggle') 260 | }) 261 | 262 | }(window.jQuery);/* ========================================================== 263 | * bootstrap-carousel.js v2.3.2 264 | * http://twitter.github.com/bootstrap/javascript.html#carousel 265 | * ========================================================== 266 | * Copyright 2012 Twitter, Inc. 267 | * 268 | * Licensed under the Apache License, Version 2.0 (the "License"); 269 | * you may not use this file except in compliance with the License. 270 | * You may obtain a copy of the License at 271 | * 272 | * http://www.apache.org/licenses/LICENSE-2.0 273 | * 274 | * Unless required by applicable law or agreed to in writing, software 275 | * distributed under the License is distributed on an "AS IS" BASIS, 276 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 277 | * See the License for the specific language governing permissions and 278 | * limitations under the License. 279 | * ========================================================== */ 280 | 281 | 282 | !function ($) { 283 | 284 | "use strict"; // jshint ;_; 285 | 286 | 287 | /* CAROUSEL CLASS DEFINITION 288 | * ========================= */ 289 | 290 | var Carousel = function (element, options) { 291 | this.$element = $(element) 292 | this.$indicators = this.$element.find('.carousel-indicators') 293 | this.options = options 294 | this.options.pause == 'hover' && this.$element 295 | .on('mouseenter', $.proxy(this.pause, this)) 296 | .on('mouseleave', $.proxy(this.cycle, this)) 297 | } 298 | 299 | Carousel.prototype = { 300 | 301 | cycle: function (e) { 302 | if (!e) this.paused = false 303 | if (this.interval) clearInterval(this.interval); 304 | this.options.interval 305 | && !this.paused 306 | && (this.interval = setInterval($.proxy(this.next, this), this.options.interval)) 307 | return this 308 | } 309 | 310 | , getActiveIndex: function () { 311 | this.$active = this.$element.find('.item.active') 312 | this.$items = this.$active.parent().children() 313 | return this.$items.index(this.$active) 314 | } 315 | 316 | , to: function (pos) { 317 | var activeIndex = this.getActiveIndex() 318 | , that = this 319 | 320 | if (pos > (this.$items.length - 1) || pos < 0) return 321 | 322 | if (this.sliding) { 323 | return this.$element.one('slid', function () { 324 | that.to(pos) 325 | }) 326 | } 327 | 328 | if (activeIndex == pos) { 329 | return this.pause().cycle() 330 | } 331 | 332 | return this.slide(pos > activeIndex ? 'next' : 'prev', $(this.$items[pos])) 333 | } 334 | 335 | , pause: function (e) { 336 | if (!e) this.paused = true 337 | if (this.$element.find('.next, .prev').length && $.support.transition.end) { 338 | this.$element.trigger($.support.transition.end) 339 | this.cycle(true) 340 | } 341 | clearInterval(this.interval) 342 | this.interval = null 343 | return this 344 | } 345 | 346 | , next: function () { 347 | if (this.sliding) return 348 | return this.slide('next') 349 | } 350 | 351 | , prev: function () { 352 | if (this.sliding) return 353 | return this.slide('prev') 354 | } 355 | 356 | , slide: function (type, next) { 357 | var $active = this.$element.find('.item.active') 358 | , $next = next || $active[type]() 359 | , isCycling = this.interval 360 | , direction = type == 'next' ? 'left' : 'right' 361 | , fallback = type == 'next' ? 'first' : 'last' 362 | , that = this 363 | , e 364 | 365 | this.sliding = true 366 | 367 | isCycling && this.pause() 368 | 369 | $next = $next.length ? $next : this.$element.find('.item')[fallback]() 370 | 371 | e = $.Event('slide', { 372 | relatedTarget: $next[0] 373 | , direction: direction 374 | }) 375 | 376 | if ($next.hasClass('active')) return 377 | 378 | if (this.$indicators.length) { 379 | this.$indicators.find('.active').removeClass('active') 380 | this.$element.one('slid', function () { 381 | var $nextIndicator = $(that.$indicators.children()[that.getActiveIndex()]) 382 | $nextIndicator && $nextIndicator.addClass('active') 383 | }) 384 | } 385 | 386 | if ($.support.transition && this.$element.hasClass('slide')) { 387 | this.$element.trigger(e) 388 | if (e.isDefaultPrevented()) return 389 | $next.addClass(type) 390 | $next[0].offsetWidth // force reflow 391 | $active.addClass(direction) 392 | $next.addClass(direction) 393 | this.$element.one($.support.transition.end, function () { 394 | $next.removeClass([type, direction].join(' ')).addClass('active') 395 | $active.removeClass(['active', direction].join(' ')) 396 | that.sliding = false 397 | setTimeout(function () { that.$element.trigger('slid') }, 0) 398 | }) 399 | } else { 400 | this.$element.trigger(e) 401 | if (e.isDefaultPrevented()) return 402 | $active.removeClass('active') 403 | $next.addClass('active') 404 | this.sliding = false 405 | this.$element.trigger('slid') 406 | } 407 | 408 | isCycling && this.cycle() 409 | 410 | return this 411 | } 412 | 413 | } 414 | 415 | 416 | /* CAROUSEL PLUGIN DEFINITION 417 | * ========================== */ 418 | 419 | var old = $.fn.carousel 420 | 421 | $.fn.carousel = function (option) { 422 | return this.each(function () { 423 | var $this = $(this) 424 | , data = $this.data('carousel') 425 | , options = $.extend({}, $.fn.carousel.defaults, typeof option == 'object' && option) 426 | , action = typeof option == 'string' ? option : options.slide 427 | if (!data) $this.data('carousel', (data = new Carousel(this, options))) 428 | if (typeof option == 'number') data.to(option) 429 | else if (action) data[action]() 430 | else if (options.interval) data.pause().cycle() 431 | }) 432 | } 433 | 434 | $.fn.carousel.defaults = { 435 | interval: 5000 436 | , pause: 'hover' 437 | } 438 | 439 | $.fn.carousel.Constructor = Carousel 440 | 441 | 442 | /* CAROUSEL NO CONFLICT 443 | * ==================== */ 444 | 445 | $.fn.carousel.noConflict = function () { 446 | $.fn.carousel = old 447 | return this 448 | } 449 | 450 | /* CAROUSEL DATA-API 451 | * ================= */ 452 | 453 | $(document).on('click.carousel.data-api', '[data-slide], [data-slide-to]', function (e) { 454 | var $this = $(this), href 455 | , $target = $($this.attr('data-target') || (href = $this.attr('href')) && href.replace(/.*(?=#[^\s]+$)/, '')) //strip for ie7 456 | , options = $.extend({}, $target.data(), $this.data()) 457 | , slideIndex 458 | 459 | $target.carousel(options) 460 | 461 | if (slideIndex = $this.attr('data-slide-to')) { 462 | $target.data('carousel').pause().to(slideIndex).cycle() 463 | } 464 | 465 | e.preventDefault() 466 | }) 467 | 468 | }(window.jQuery);/* ============================================================= 469 | * bootstrap-collapse.js v2.3.2 470 | * http://twitter.github.com/bootstrap/javascript.html#collapse 471 | * ============================================================= 472 | * Copyright 2012 Twitter, Inc. 473 | * 474 | * Licensed under the Apache License, Version 2.0 (the "License"); 475 | * you may not use this file except in compliance with the License. 476 | * You may obtain a copy of the License at 477 | * 478 | * http://www.apache.org/licenses/LICENSE-2.0 479 | * 480 | * Unless required by applicable law or agreed to in writing, software 481 | * distributed under the License is distributed on an "AS IS" BASIS, 482 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 483 | * See the License for the specific language governing permissions and 484 | * limitations under the License. 485 | * ============================================================ */ 486 | 487 | 488 | !function ($) { 489 | 490 | "use strict"; // jshint ;_; 491 | 492 | 493 | /* COLLAPSE PUBLIC CLASS DEFINITION 494 | * ================================ */ 495 | 496 | var Collapse = function (element, options) { 497 | this.$element = $(element) 498 | this.options = $.extend({}, $.fn.collapse.defaults, options) 499 | 500 | if (this.options.parent) { 501 | this.$parent = $(this.options.parent) 502 | } 503 | 504 | this.options.toggle && this.toggle() 505 | } 506 | 507 | Collapse.prototype = { 508 | 509 | constructor: Collapse 510 | 511 | , dimension: function () { 512 | var hasWidth = this.$element.hasClass('width') 513 | return hasWidth ? 'width' : 'height' 514 | } 515 | 516 | , show: function () { 517 | var dimension 518 | , scroll 519 | , actives 520 | , hasData 521 | 522 | if (this.transitioning || this.$element.hasClass('in')) return 523 | 524 | dimension = this.dimension() 525 | scroll = $.camelCase(['scroll', dimension].join('-')) 526 | actives = this.$parent && this.$parent.find('> .accordion-group > .in') 527 | 528 | if (actives && actives.length) { 529 | hasData = actives.data('collapse') 530 | if (hasData && hasData.transitioning) return 531 | actives.collapse('hide') 532 | hasData || actives.data('collapse', null) 533 | } 534 | 535 | this.$element[dimension](0) 536 | this.transition('addClass', $.Event('show'), 'shown') 537 | $.support.transition && this.$element[dimension](this.$element[0][scroll]) 538 | } 539 | 540 | , hide: function () { 541 | var dimension 542 | if (this.transitioning || !this.$element.hasClass('in')) return 543 | dimension = this.dimension() 544 | this.reset(this.$element[dimension]()) 545 | this.transition('removeClass', $.Event('hide'), 'hidden') 546 | this.$element[dimension](0) 547 | } 548 | 549 | , reset: function (size) { 550 | var dimension = this.dimension() 551 | 552 | this.$element 553 | .removeClass('collapse') 554 | [dimension](size || 'auto') 555 | [0].offsetWidth 556 | 557 | this.$element[size !== null ? 'addClass' : 'removeClass']('collapse') 558 | 559 | return this 560 | } 561 | 562 | , transition: function (method, startEvent, completeEvent) { 563 | var that = this 564 | , complete = function () { 565 | if (startEvent.type == 'show') that.reset() 566 | that.transitioning = 0 567 | that.$element.trigger(completeEvent) 568 | } 569 | 570 | this.$element.trigger(startEvent) 571 | 572 | if (startEvent.isDefaultPrevented()) return 573 | 574 | this.transitioning = 1 575 | 576 | this.$element[method]('in') 577 | 578 | $.support.transition && this.$element.hasClass('collapse') ? 579 | this.$element.one($.support.transition.end, complete) : 580 | complete() 581 | } 582 | 583 | , toggle: function () { 584 | this[this.$element.hasClass('in') ? 'hide' : 'show']() 585 | } 586 | 587 | } 588 | 589 | 590 | /* COLLAPSE PLUGIN DEFINITION 591 | * ========================== */ 592 | 593 | var old = $.fn.collapse 594 | 595 | $.fn.collapse = function (option) { 596 | return this.each(function () { 597 | var $this = $(this) 598 | , data = $this.data('collapse') 599 | , options = $.extend({}, $.fn.collapse.defaults, $this.data(), typeof option == 'object' && option) 600 | if (!data) $this.data('collapse', (data = new Collapse(this, options))) 601 | if (typeof option == 'string') data[option]() 602 | }) 603 | } 604 | 605 | $.fn.collapse.defaults = { 606 | toggle: true 607 | } 608 | 609 | $.fn.collapse.Constructor = Collapse 610 | 611 | 612 | /* COLLAPSE NO CONFLICT 613 | * ==================== */ 614 | 615 | $.fn.collapse.noConflict = function () { 616 | $.fn.collapse = old 617 | return this 618 | } 619 | 620 | 621 | /* COLLAPSE DATA-API 622 | * ================= */ 623 | 624 | $(document).on('click.collapse.data-api', '[data-toggle=collapse]', function (e) { 625 | var $this = $(this), href 626 | , target = $this.attr('data-target') 627 | || e.preventDefault() 628 | || (href = $this.attr('href')) && href.replace(/.*(?=#[^\s]+$)/, '') //strip for ie7 629 | , option = $(target).data('collapse') ? 'toggle' : $this.data() 630 | $this[$(target).hasClass('in') ? 'addClass' : 'removeClass']('collapsed') 631 | $(target).collapse(option) 632 | }) 633 | 634 | }(window.jQuery);/* ============================================================ 635 | * bootstrap-dropdown.js v2.3.2 636 | * http://twitter.github.com/bootstrap/javascript.html#dropdowns 637 | * ============================================================ 638 | * Copyright 2012 Twitter, Inc. 639 | * 640 | * Licensed under the Apache License, Version 2.0 (the "License"); 641 | * you may not use this file except in compliance with the License. 642 | * You may obtain a copy of the License at 643 | * 644 | * http://www.apache.org/licenses/LICENSE-2.0 645 | * 646 | * Unless required by applicable law or agreed to in writing, software 647 | * distributed under the License is distributed on an "AS IS" BASIS, 648 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 649 | * See the License for the specific language governing permissions and 650 | * limitations under the License. 651 | * ============================================================ */ 652 | 653 | 654 | !function ($) { 655 | 656 | "use strict"; // jshint ;_; 657 | 658 | 659 | /* DROPDOWN CLASS DEFINITION 660 | * ========================= */ 661 | 662 | var toggle = '[data-toggle=dropdown]' 663 | , Dropdown = function (element) { 664 | var $el = $(element).on('click.dropdown.data-api', this.toggle) 665 | $('html').on('click.dropdown.data-api', function () { 666 | $el.parent().removeClass('open') 667 | }) 668 | } 669 | 670 | Dropdown.prototype = { 671 | 672 | constructor: Dropdown 673 | 674 | , toggle: function (e) { 675 | var $this = $(this) 676 | , $parent 677 | , isActive 678 | 679 | if ($this.is('.disabled, :disabled')) return 680 | 681 | $parent = getParent($this) 682 | 683 | isActive = $parent.hasClass('open') 684 | 685 | clearMenus() 686 | 687 | if (!isActive) { 688 | if ('ontouchstart' in document.documentElement) { 689 | // if mobile we we use a backdrop because click events don't delegate 690 | $('