diff --git a/.editorconfig b/.editorconfig new file mode 100644 index 000000000..0f099897b --- /dev/null +++ b/.editorconfig @@ -0,0 +1,10 @@ +# editorconfig.org +root = true + +[*] +indent_style = space +indent_size = 2 +end_of_line = lf +charset = utf-8 +trim_trailing_whitespace = true +insert_final_newline = true diff --git a/.gitignore b/.gitignore new file mode 100644 index 000000000..b18c4df45 --- /dev/null +++ b/.gitignore @@ -0,0 +1,118 @@ +################################################ +############### .gitignore ################## +################################################ +# +# This file is only relevant if you are using git. +# +# Files which match the splat patterns below will +# be ignored by git. This keeps random crap and +# sensitive credentials from being uploaded to +# your repository. It allows you to configure your +# app for your machine without accidentally +# committing settings which will smash the local +# settings of other developers on your team. +# +# Some reasonable defaults are included below, +# but, of course, you should modify/extend/prune +# to fit your needs! +################################################ + + + + +################################################ +# Local Configuration +# +# Explicitly ignore files which contain: +# +# 1. Sensitive information you'd rather not push to +# your git repository. +# e.g., your personal API keys or passwords. +# +# 2. Environment-specific configuration +# Basically, anything that would be annoying +# to have to change every time you do a +# `git pull` +# e.g., your local development database, or +# the S3 bucket you're using for file uploads +# development. +# +################################################ + +config/local.js + + + + + +################################################ +# Dependencies +# +# When releasing a production app, you may +# consider including your node_modules and +# bower_components directory in your git repo, +# but during development, its best to exclude it, +# since different developers may be working on +# different kernels, where dependencies would +# need to be recompiled anyway. +# +# More on that here about node_modules dir: +# http://www.futurealoof.com/posts/nodemodules-in-git.html +# (credit Mikeal Rogers, @mikeal) +# +# About bower_components dir, you can see this: +# http://addyosmani.com/blog/checking-in-front-end-dependencies/ +# (credit Addy Osmani, @addyosmani) +# +################################################ + +node_modules +bower_components + + + + +################################################ +# Sails.js / Waterline / Grunt +# +# Files generated by Sails and Grunt, or related +# tasks and adapters. +################################################ +#.tmp +dump.rdb + + + + + +################################################ +# Node.js / NPM +# +# Common files generated by Node, NPM, and the +# related ecosystem. +################################################ +lib-cov +*.seed +*.log +*.out +*.pid +npm-debug.log + + + + + +################################################ +# Miscellaneous +# +# Common files generated by text editors, +# operating systems, file systems, etc. +################################################ + +*~ +*# +.DS_STORE +.netbeans +nbproject +.idea +.node_history diff --git a/.sailsrc b/.sailsrc new file mode 100644 index 000000000..0d2c3d9ad --- /dev/null +++ b/.sailsrc @@ -0,0 +1,8 @@ +{ + "generators": { + "modules": {} + }, + "hooks": { + "grunt": false + } +} \ No newline at end of file diff --git a/.tmp/localDiskDb.db b/.tmp/localDiskDb.db new file mode 100644 index 000000000..8b01fbee2 --- /dev/null +++ b/.tmp/localDiskDb.db @@ -0,0 +1,473 @@ +{ + "data": { + "restaurants": [ + { + "name": "Mission Chinese Food", + "neighborhood": "Manhattan", + "photograph": "1", + "address": "171 E Broadway, New York, NY 10002", + "latlng": { + "lat": 40.713829, + "lng": -73.989667 + }, + "cuisine_type": "Asian", + "operating_hours": { + "Monday": "5:30 pm - 11:00 pm", + "Tuesday": "5:30 pm - 11:00 pm", + "Wednesday": "5:30 pm - 11:00 pm", + "Thursday": "5:30 pm - 11:00 pm", + "Friday": "5:30 pm - 11:00 pm", + "Saturday": "12:00 pm - 4:00 pm, 5:30 pm - 12:00 am", + "Sunday": "12:00 pm - 4:00 pm, 5:30 pm - 11:00 pm" + }, + "reviews": [ + { + "name": "Steve", + "date": "October 26, 2016", + "rating": 4, + "comments": "Mission Chinese Food has grown up from its scrappy Orchard Street days into a big, two story restaurant equipped with a pizza oven, a prime rib cart, and a much broader menu. Yes, it still has all the hits — the kung pao pastrami, the thrice cooked bacon —but chef/proprietor Danny Bowien and executive chef Angela Dimayuga have also added a raw bar, two generous family-style set menus, and showstoppers like duck baked in clay. And you can still get a lot of food without breaking the bank." + }, + { + "name": "Morgan", + "date": "October 26, 2016", + "rating": "4", + "comments": "This place is a blast. Must orders: GREEN TEA NOODS, sounds gross (to me at least) but these were incredible!, Kung pao pastrami (but you already knew that), beef tartare was a fun appetizer that we decided to try, the spicy ma po tofu SUPER spicy but delicous, egg rolls and scallion pancake i could have passed on... I wish we would have gone with a larger group, so much more I would have liked to try!" + }, + { + "name": "Jason", + "date": "October 26, 2016", + "rating": "3", + "comments": "I was VERY excited to come here after seeing and hearing so many good things about this place. Having read much, I knew going into it that it was not going to be authentic Chinese. The place was edgy, had a punk rock throwback attitude, and generally delivered the desired atmosphere. Things went downhill from there though. The food was okay at best and the best qualities were easily overshadowed by what I believe to be poor decisions by the kitchen staff." + } + ], + "createdAt": "2017-07-25T02:26:54.985Z", + "updatedAt": "2017-07-25T02:26:54.985Z", + "id": 1 + }, + { + "name": "Emily", + "neighborhood": "Brooklyn", + "photograph": "2", + "address": "919 Fulton St, Brooklyn, NY 11238", + "latlng": { + "lat": 40.683555, + "lng": -73.966393 + }, + "cuisine_type": "Pizza", + "operating_hours": { + "Monday": "5:30 pm - 11:00 pm", + "Tuesday": "5:30 pm - 11:00 pm", + "Wednesday": "5:30 pm - 11:00 pm", + "Thursday": "5:30 pm - 11:00 pm", + "Friday": "5:30 pm - 11:00 pm", + "Saturday": "5:00 pm - 11:30 pm", + "Sunday": "12:00 pm - 3:00 pm, 5:00 pm - 11:00 pm" + }, + "reviews": [ + { + "name": "Steph", + "date": "October 26, 2016", + "rating": 4, + "comments": "Five star food, two star atmosphere. I would definitely get takeout from this place - but dont think I have the energy to deal with the hipster ridiculousness again. By the time we left the wait was two hours long." + }, + { + "name": "Steve", + "date": "October 26, 2016", + "rating": 4, + "comments": "This cozy Clinton Hill restaurant excels at both straightforward and unusual wood-fired pizzas. If you want a taste of the latter, consider ordering the Emily, which is topped with mozzarella, pistachios, truffled sottocenere cheese, and honey. The menu includes salads and a handful of starters, as well as a burger that some meat connoisseurs consider to be among the best in the city." + }, + { + "name": "Sam", + "date": "October 26, 2016", + "rating": 5, + "comments": "5 star atmosphere as it is very cozy with great staff. 5 star food as their Emmy burger is outrageously good. and its on a pretzel bun.. Too juicy for its own good and downright addicting. Also try the Colony pizza. Many others looked like worth competitors, but the Colony really found its way to my heart. when you start with a great crust, top it with top notch cheese and sauce, you've got a winner. But, if you go a step further and add the salty from the pepperoni, the sweet from the honey, and the spicy from the chili oil.... your mouth is confused and happy at the same time." + } + ], + "createdAt": "2017-07-25T02:29:58.323Z", + "updatedAt": "2017-07-25T02:29:58.323Z", + "id": 2 + }, + { + "name": "Kang Ho Dong Baekjeong", + "neighborhood": "Manhattan", + "photograph": "3", + "address": "1 E 32nd St, New York, NY 10016", + "latlng": { + "lat": 40.747143, + "lng": -73.985414 + }, + "cuisine_type": "Asian", + "operating_hours": { + "Monday": "11:30 am - 2:00 am", + "Tuesday": "11:30 am - 2:00 am", + "Wednesday": "11:30 am - 2:00 am", + "Thursday": "11:30 am - 2:00 am", + "Friday": "11:30 am - 6:00 am", + "Saturday": "11:30 am - 6:00 am", + "Sunday": "11:30 am - 2:00 am" + }, + "reviews": [ + { + "name": "Steve", + "date": "October 26, 2016", + "rating": 4, + "comments": "The tables at this 32nd Street favorite are outfitted with grills for cooking short ribs, brisket, beef tongue, rib eye, and pork jowl. The banchan plates are uniformly good, and Deuki Hong’s menu also includes winning dishes like stir-fried squid noodles, kimchi stew, and seafood pancakes. If it’s available, make sure to order the kimchi and rice “lunchbox.” Baekjeong is a great place for large groups and birthday parties." + }, + { + "name": "ZS", + "date": "October 26, 2016", + "rating": 5, + "comments": "I've been to Korea before and many other Korean BBQ places. We had the regular pork belly and a beef (forgot which cut) and a seafood tofu soup. Two meat and a soup was just prefect for the two of us. We could have done one meat and one soup. The portions of the meat are great! The beef was juicy, tender and so good. The sides were excellent. " + }, + { + "name": "Emily", + "date": "October 26, 2016", + "rating": 2, + "comments": "MEH. I've tried their Jersey location as well but Kang Ho Dong meat quality is severely decreasing. A Korean bbq place with whatever meat? I think NOT!" + } + ], + "createdAt": "2017-07-25T02:30:26.391Z", + "updatedAt": "2017-07-25T02:30:26.391Z", + "id": 3 + }, + { + "name": "Katz's Delicatessen", + "neighborhood": "Manhattan", + "photograph": "4", + "address": "205 E Houston St, New York, NY 10002", + "latlng": { + "lat": 40.722216, + "lng": -73.987501 + }, + "cuisine_type": "American", + "operating_hours": { + "Monday": "8:00 am - 10:30 pm", + "Tuesday": "8:00 am - 10:30 pm", + "Wednesday": "8:00 am - 10:30 pm", + "Thursday": "8:00 am - 2:30 am", + "Friday": "8:00 am - Sat", + "Saturday": "Open 24 hours", + "Sunday": "Sat - 10:30 pm" + }, + "reviews": [ + { + "name": "Steve", + "date": "October 26, 2016", + "rating": 5, + "comments": "In 127 years, little has changed at Katz's. It remains one of New York's — and the country's — essential Jewish delicatessens. Every inch of the massive Lower East Side space smells intensely of pastrami and rye loaves. The sandwiches are massive, so they are best when shared. Order at the counter, and don't forget to tip your slicer — your sandwich will be better for it." + }, + { + "name": "Allen", + "date": "October 26, 2016", + "rating": 5, + "comments": "If I lived in NY and got diabetes from eating here every single time I ate, I would do it over and over and over again just for that first bite. These guys know how to make a sandwich. The heart attack comes free of charge! Came by while I was visiting NYC. First pit-stop when I come back :)!" + }, + { + "name": "David", + "date": "October 26, 2016", + "rating": 2, + "comments": "Ok so four of us came. One more later who didn't order becauase it's so expensive and simple. Seriously, a bunch of meat albeit you can sample beforehand on rye/white/wheat bread. Cheese extra. Pickles free, you can just ask them at the pickup counter. But seriously 20 bucks gone for an non-flavored half sandwich. And a line that is long, especially if you want seating. I'm down to just take a quick look where Sally and Harry sat and leave to the other delis all around NYC. Oh and they accept Samsung pay." + } + ], + "createdAt": "2017-07-25T02:30:43.429Z", + "updatedAt": "2017-07-25T02:30:43.429Z", + "id": 4 + }, + { + "name": "Roberta's Pizza", + "neighborhood": "Brooklyn", + "photograph": "5", + "address": "261 Moore St, Brooklyn, NY 11206", + "latlng": { + "lat": 40.705089, + "lng": -73.933585 + }, + "cuisine_type": "Pizza", + "operating_hours": { + "Monday": "11:00 am - 12:00 am", + "Tuesday": "11:00 am - 12:00 am", + "Wednesday": "11:00 am - 12:00 am", + "Thursday": "11:00 am - 12:00 am", + "Friday": "11:00 am - 12:00 am", + "Saturday": "10:00 am - 12:00 am", + "Sunday": "10:00 am - 12:00 am" + }, + "reviews": [ + { + "name": "Steve", + "date": "October 26, 2016", + "rating": 4, + "comments": "Roberta's is the epicenter of the modern Brooklyn food scene.The pizzas are fantastic, but the restaurant really flexes its muscles with the vegetable dishes. In addition to the pies, consider ordering the radishes, the romaine salad, the roasted beets, and some of the charcuterie." + }, + { + "name": "Raymond", + "date": "October 26, 2016", + "rating": 4, + "comments": "Roberta's, one of the better pizzas I have had in my life. Very trendy and hipsterish spot. Came here for lunch on a random weekday afternoon and when we arrived, there was a line forming already. The space is a bit cramped. You'll get to know your neighbors soon enough. The pizza is just delightful and delicious. It's a ncie plus that you get to see them firing up the pizzas in the corner. The major issue with Roberta's is the trek out to the Williamsburg/Bushwick." + }, + { + "name": "Laurel", + "date": "October 26, 2016", + "rating": 4, + "comments": "The pizza is fantastic, not THE best I've ever had, but would definitely go back since it has great food and great ambiance. Definitely worth going to. It has A LOT of hype in the New York food scene, and I question if it deserves all of it, but it's still a totally great spot to hit up when in the area!!" + } + ], + "createdAt": "2017-07-25T02:30:55.347Z", + "updatedAt": "2017-07-25T02:30:55.347Z", + "id": 5 + }, + { + "name": "Hometown BBQ", + "neighborhood": "Brooklyn", + "photograph": "6", + "address": "454 Van Brunt St, Brooklyn, NY 11231", + "latlng": { + "lat": 40.674925, + "lng": -74.016162 + }, + "cuisine_type": "American", + "operating_hours": { + "Monday": "Closed", + "Tuesday": "12:00 pm - 10:00 pm", + "Wednesday": "12:00 pm - 10:00 pm", + "Thursday": "12:00 pm - 10:00 pm", + "Friday": "12:00 pm - 11:00 pm", + "Saturday": "12:00 pm - 11:00 pm", + "Sunday": "12:00 pm - 9:00 pm" + }, + "reviews": [ + { + "name": "Steve", + "date": "October 26, 2016", + "rating": 4, + "comments": "Barbecue aficionados agree that Billy Durney is cooking up some of the best Texas-style barbecue in the city. Straightforward classics like smoked brisket and baby back ribs are always a strong choice, but there are also options like pork belly tacos and a lamb belly banh mi. The space is sprawling in a way that feels like the real deal, and Durney himself can usually be found working the room, and keeping a watchful eye on the smoking meats. It's counter service only, and there's often a line, but for the scene and certainly for the meat, it's easily worth the trip to Red Hook." + }, + { + "name": "Michelle", + "date": "October 26, 2016", + "rating": 4, + "comments": "Overall, a great try of New York BBQ. The restaurant décor is rustic with a good amount of seats to sit and enjoy the meal. I definitely would love to come back and try that monster of a beef rib!" + }, + { + "name": "Ai-Mei", + "date": "October 26, 2016", + "rating": 4, + "comments": "mmmmmm, what a gem this is in bklyn! I loveeee how soft their brisket is here. Their pork tacos are to die for, their different variety of ribs and lastly, their lamb is not gamey at all. Super wallet friendly for the amount they give you. I highly recommend this spot- after eating here, you can definitely walk over for Steve's key lime pies." + } + ], + "createdAt": "2017-07-25T02:31:07.169Z", + "updatedAt": "2017-07-25T02:31:07.169Z", + "id": 6 + }, + { + "name": "Superiority Burger", + "neighborhood": "Manhattan", + "photograph": "7", + "address": "430 E 9th St, New York, NY 10009", + "latlng": { + "lat": 40.727397, + "lng": -73.983645 + }, + "cuisine_type": "American", + "operating_hours": { + "Monday": "11:30 am - 10:00 pm", + "Tuesday": "Closed", + "Wednesday": "11:30 am - 10:00 pm", + "Thursday": "11:30 am - 10:00 pm", + "Friday": "11:30 am - 10:00 pm", + "Saturday": "11:30 am - 10:00 pm", + "Sunday": "11:30 am - 10:00 pm" + }, + "reviews": [ + { + "name": "Steve", + "date": "October 26, 2016", + "rating": 4, + "comments": "Brooks Headley’s tiny East Village cafe is so much more than a veggie burger spot — it's one of the best bang-for-your-buck restaurants in Lower Manhattan. Headley and his crew turn seasonal vegetables into delectable salads and riffs on American comfort food favorites. The specials menu changes daily, and the rest of the menu is constantly evolving. You can get a lot of food to eat here for under $15 per person." + }, + { + "name": "Gabriel", + "date": "October 26, 2016", + "rating": 5, + "comments": "I was turned on to this place following the glowing NYT review. Its near my area of the city so I figured why not go? Man they weren't kidding, Superiority Burger is probably the best vegetarian experience I've ever had!" + }, + { + "name": "Shivi", + "date": "October 26, 2016", + "rating": 4, + "comments": "Great flavors and very satisfying. Craving a sandwich, I stopped by on a Friday night with a vegetarian friend. Super small location with just a few seats inside. Ambiance is a bit industrial, good is definitely much more sophisticated than the look of the place! Ordered the superiority burger anda side of potato salad. The potato salad was very light and tasted clean ( less mayo, lots of dill and some cucumber) -- refreshing for a humid summer night! Sandwich was surprisingly delicious - it is very small ( funny allusion to a White Castle burger) but it packs a punch! Not only are there layers of flavors ( amazing sauces) but the party itself had a great texture Ahmed flavor-- well done and so wonderful! Will definitely stop by again for an overall amazing burger/sandwich. Staff was super nice and accommodating but not out of the way friendly." + } + ], + "createdAt": "2017-07-25T02:31:19.824Z", + "updatedAt": "2017-07-25T02:31:19.824Z", + "id": 7 + }, + { + "name": "The Dutch", + "neighborhood": "Manhattan", + "photograph": "8", + "address": "131 Sullivan St, New York, NY 10012", + "latlng": { + "lat": 40.726584, + "lng": -74.002082 + }, + "cuisine_type": "American", + "operating_hours": { + "Monday": "11:30 am - 3:00 pm, 5:30 pm - 11:00 pm", + "Tuesday": "11:30 am - 3:00 pm, 5:30 pm - 11:00 pm", + "Wednesday": "11:30 am - 3:00 pm, 5:30 pm - 11:00 pm", + "Thursday": "11:30 am - 3:00 pm, 5:30 pm - 11:00 pm", + "Friday": "11:30 am - 3:00 pm, 5:30 pm - 11:30 pm", + "Saturday": "10:00 am - 3:00 pm, 5:30 pm - 11:30 pm", + "Sunday": "10:00 am - 3:00 pm, 5:30 pm - 11:00 pm" + }, + "reviews": [ + { + "name": "Steve", + "date": "October 26, 2016", + "rating": 4, + "comments": "Over the last five years, The Dutch has turned into the quintessential American restaurant that chef Andrew Carmellini and partners Josh Pickard and Luke Ostrom sought to evoke when it first opened. It’s a great choice when you’re craving a steak, a burger, or oysters, and the menu always includes plentiful seafood options as well as pastas. The Dutch is now an indelible part of the Soho landscape." + }, + { + "name": "Loren", + "date": "October 26, 2016", + "rating": 4, + "comments": "I randomly came here on a Saturday night. I was pleasantly surprised with the food and the service. We had the calamari and the ceviche with avocado, and then the catfish. Oh! Then we had the banana soufflé for dessert with ice cream. It was all delicious and well put together. Would love to eat here again." + }, + { + "name": "Lori", + "date": "October 26, 2016", + "rating": 4, + "comments": "Aside from the slightly claustrophobic dining area and the fact that you may have difficulty hearing your dining companion, I'd return to The Dutch without hesitation. The food is surprisingly well-executed and conceived, and our dinner service flowed smoothly without a hitch. Just make sure to get a reservation in advance, as I'm sure more than just a few other people will have the same idea." + } + ], + "createdAt": "2017-07-25T02:31:33.052Z", + "updatedAt": "2017-07-25T02:31:33.052Z", + "id": 8 + }, + { + "name": "Mu Ramen", + "neighborhood": "Queens", + "photograph": "9", + "address": "1209 Jackson Ave, Queens, NY 11101", + "latlng": { + "lat": 40.743797, + "lng": -73.950652 + }, + "cuisine_type": "Asian", + "operating_hours": { + "Monday": "5:00 pm - 10:00 pm", + "Tuesday": "5:00 pm - 10:00 pm", + "Wednesday": "5:00 pm - 10:00 pm", + "Thursday": "5:00 pm - 10:00 pm", + "Friday": "5:00 pm - 11:00 pm", + "Saturday": "5:00 pm - 11:00 pm", + "Sunday": "5:00 pm - 10:00 pm" + }, + "reviews": [ + { + "name": "Steve", + "date": "October 26, 2016", + "rating": 4, + "comments": "Joshua Smookler’s two-year-old ramen shop serves one of the best tonkotsu broths around. Beyond ramen, Mu also offers some high minded plates, like foie gras-stuffed chicken wings, as well as dry-aged Japanese Wagyu beef specials. Mu is just 10 short minutes away from Midtown via the 7-train." + }, + { + "name": "Brittany", + "date": "October 26, 2016", + "rating": 4, + "comments": "Overall, I would definitely recommend this place if you enjoy thick curly noodles with a thick, intense broth. If you don't there are still other options but I can't vouch for those." + }, + { + "name": "Sally", + "date": "October 26, 2016", + "rating": 4, + "comments": "One of the tastiest and most unique ramen places I've been to in NYC, but also the priciest. I think overall its worth the try. Not an everyday casual ramen shop though." + } + ], + "createdAt": "2017-07-25T02:31:43.330Z", + "updatedAt": "2017-07-25T02:31:43.330Z", + "id": 9 + }, + { + "name": "Casa Enrique", + "neighborhood": "Queens", + "address": "5-48 49th Ave, Queens, NY 11101", + "latlng": { + "lat": 40.743394, + "lng": -73.954235 + }, + "cuisine_type": "Mexican", + "operating_hours": { + "Monday": "5:00 pm - 12:00 am", + "Tuesday": "5:00 pm - 12:00 am", + "Wednesday": "5:00 pm - 12:00 am", + "Thursday": "5:00 pm - 12:00 am", + "Friday": "5:00 pm - 12:00 am", + "Saturday": "11:00 am - 12:00 am", + "Sunday": "11:00 am - 12:00 am" + }, + "reviews": [ + { + "name": "Steve", + "date": "October 26, 2016", + "rating": 5, + "comments": "Head to this laid-back Long Island City restaurant for beef tongue tacos, chicken smothered in a heady mole sauce, and a monster crab tostada. New York's only Michelin-starred Mexican restaurant is an especially cool choice for lunch during the week or drinks after work. Eater critic Ryan Sutton awarded this restaurant two stars." + }, + { + "name": "Rob", + "date": "October 26, 2016", + "rating": 5, + "comments": "The hype was real. Please go. Get the ceviche. And the tres leches. You're welcome" + }, + { + "name": "Jason", + "date": "October 26, 2016", + "rating": 4, + "comments": "For a Michelin star restaurant, it's fairly priced and the food is fairly good. Started with a strawberry margarita which was good in flavor but not much alcohol. Had the chicken enchiladas with salsa verde and it was really good. Great balance in flavor and a good portion. Extra tasty with their hot sauces. My wife had the lamb but it was a bit too salty for our taste. Although, it was cooked very well and fell off the bone. The highlight of the night was the tres leches cake. Probably the best I've ever had to be honest. Not too sweet and very milky. Overall, one of my top 3 favorite Mexican in NY." + } + ], + "createdAt": "2017-07-25T02:31:59.276Z", + "updatedAt": "2017-07-25T02:31:59.276Z", + "id": 10 + } + ] + }, + "schema": { + "restaurant": { + "id": { + "type": "integer", + "autoIncrement": true, + "primaryKey": true, + "unique": true + }, + "createdAt": { + "type": "datetime" + }, + "updatedAt": { + "type": "datetime" + } + }, + "restaurants": { + "id": { + "type": "integer", + "autoIncrement": true, + "primaryKey": true, + "unique": true + }, + "createdAt": { + "type": "datetime" + }, + "updatedAt": { + "type": "datetime" + } + } + }, + "counters": { + "restaurant": { + "id": 10 + }, + "restaurants": { + "id": 10 + } + } +} diff --git a/README.md b/README.md index 209424b8b..6ee728b57 100644 --- a/README.md +++ b/README.md @@ -1,18 +1,18 @@ # Mobile Web Specialist Certification Course --- -## _Three Stage Course Material Project - Restaurant Reviews: Stage 1_ +## _Three Stage Course Material Project - Restaurant Reviews: Stage 2_ -Developed by [Melanie Archer](https://github.com/mejarc/mws-restaurant-stage-1.git). +Developed by [Melanie Archer](https://github.com/mejarc/mws-restaurant-stage-2.git). A static webpage incrementally converted to a mobile-ready web application. -In **Stage One**, the enhancements include: +In **Stage One**, the enhancements included: * conversion to responsive design * screen reader accessibility features * adding a service worker to cache content. ### Installation and usage -* [Clone or download the code repo](https://github.com/mejarc/mws-restaurant-stage-1.git) +* [Clone or download the code repo](https://github.com/mejarc/mws-restaurant-stage-2.git) * In a console or terminal, go to the folder on your local computer where you have placed the code, and type ```shell diff --git a/api/controllers/.gitkeep b/api/controllers/.gitkeep new file mode 100644 index 000000000..e69de29bb diff --git a/api/controllers/RestaurantsController.js b/api/controllers/RestaurantsController.js new file mode 100644 index 000000000..1831d7ed6 --- /dev/null +++ b/api/controllers/RestaurantsController.js @@ -0,0 +1,11 @@ +/** + * RestaurantsController + * + * @description :: Server-side logic for managing restaurants + * @help :: See http://sailsjs.org/#!/documentation/concepts/Controllers + */ + +module.exports = { + +}; + diff --git a/api/models/.gitkeep b/api/models/.gitkeep new file mode 100644 index 000000000..e69de29bb diff --git a/api/models/Restaurants.js b/api/models/Restaurants.js new file mode 100644 index 000000000..68c54e22d --- /dev/null +++ b/api/models/Restaurants.js @@ -0,0 +1,14 @@ +/** + * Restaurants.js + * + * @description :: TODO: You might write a short summary of how this model works and what it represents here. + * @docs :: http://sailsjs.org/documentation/concepts/models-and-orm/models + */ + +module.exports = { + + attributes: { + + } +}; + diff --git a/api/policies/sessionAuth.js b/api/policies/sessionAuth.js new file mode 100644 index 000000000..8f9a26473 --- /dev/null +++ b/api/policies/sessionAuth.js @@ -0,0 +1,21 @@ +/** + * sessionAuth + * + * @module :: Policy + * @description :: Simple policy to allow any authenticated user + * Assumes that your login action in one of your controllers sets `req.session.authenticated = true;` + * @docs :: http://sailsjs.org/#!/documentation/concepts/Policies + * + */ +module.exports = function(req, res, next) { + + // User is allowed, proceed to the next policy, + // or if this is the last policy, the controller + if (req.session.authenticated) { + return next(); + } + + // User is not allowed + // (default res.forbidden() behavior can be overridden in `config/403.js`) + return res.forbidden('You are not permitted to perform this action.'); +}; diff --git a/api/responses/badRequest.js b/api/responses/badRequest.js new file mode 100644 index 000000000..0d37825c1 --- /dev/null +++ b/api/responses/badRequest.js @@ -0,0 +1,76 @@ +/** + * 400 (Bad Request) Handler + * + * Usage: + * return res.badRequest(); + * return res.badRequest(data); + * return res.badRequest(data, 'some/specific/badRequest/view'); + * + * e.g.: + * ``` + * return res.badRequest( + * 'Please choose a valid `password` (6-12 characters)', + * 'trial/signup' + * ); + * ``` + */ + +module.exports = function badRequest(data, options) { + + // Get access to `req`, `res`, & `sails` + var req = this.req; + var res = this.res; + var sails = req._sails; + + // Set status code + res.status(400); + + // Log error to console + if (data !== undefined) { + sails.log.verbose('Sending 400 ("Bad Request") response: \n',data); + } + else sails.log.verbose('Sending 400 ("Bad Request") response'); + + // Only include errors in response if application environment + // is not set to 'production'. In production, we shouldn't + // send back any identifying information about errors. + if (sails.config.environment === 'production' && sails.config.keepResponseErrors !== true) { + data = undefined; + } + + // If the user-agent wants JSON, always respond with JSON + // If views are disabled, revert to json + if (req.wantsJSON || sails.config.hooks.views === false) { + return res.jsonx(data); + } + + // If second argument is a string, we take that to mean it refers to a view. + // If it was omitted, use an empty object (`{}`) + options = (typeof options === 'string') ? { view: options } : options || {}; + + // Attempt to prettify data for views, if it's a non-error object + var viewData = data; + if (!(viewData instanceof Error) && 'object' == typeof viewData) { + try { + viewData = require('util').inspect(data, {depth: null}); + } + catch(e) { + viewData = undefined; + } + } + + // If a view was provided in options, serve it. + // Otherwise try to guess an appropriate view, or if that doesn't + // work, just send JSON. + if (options.view) { + return res.view(options.view, { data: viewData, title: 'Bad Request' }); + } + + // If no second argument provided, try to serve the implied view, + // but fall back to sending JSON(P) if no view can be inferred. + else return res.guessView({ data: viewData, title: 'Bad Request' }, function couldNotGuessView () { + return res.jsonx(data); + }); + +}; + diff --git a/api/responses/created.js b/api/responses/created.js new file mode 100644 index 000000000..314011345 --- /dev/null +++ b/api/responses/created.js @@ -0,0 +1,60 @@ +/** + * 201 (CREATED) Response + * + * Usage: + * return res.created(); + * return res.created(data); + * return res.created(data, 'auth/login'); + * + * @param {Object} data + * @param {String|Object} options + * - pass string to render specified view + */ + +module.exports = function created (data, options) { + + // Get access to `req`, `res`, & `sails` + var req = this.req; + var res = this.res; + var sails = req._sails; + + sails.log.silly('res.created() :: Sending 201 ("CREATED") response'); + + // Set status code + res.status(201); + + // If appropriate, serve data as JSON(P) + // If views are disabled, revert to json + if (req.wantsJSON || sails.config.hooks.views === false) { + return res.jsonx(data); + } + + // If second argument is a string, we take that to mean it refers to a view. + // If it was omitted, use an empty object (`{}`) + options = (typeof options === 'string') ? { view: options } : options || {}; + + // Attempt to prettify data for views, if it's a non-error object + var viewData = data; + if (!(viewData instanceof Error) && 'object' == typeof viewData) { + try { + viewData = require('util').inspect(data, {depth: null}); + } + catch(e) { + viewData = undefined; + } + } + + // If a view was provided in options, serve it. + // Otherwise try to guess an appropriate view, or if that doesn't + // work, just send JSON. + if (options.view) { + return res.view(options.view, { data: viewData, title: 'Created' }); + } + + // If no second argument provided, try to serve the implied view, + // but fall back to sending JSON(P) if no view can be inferred. + else return res.guessView({ data: viewData, title: 'Created' }, function couldNotGuessView () { + return res.jsonx(data); + }); + +}; diff --git a/api/responses/forbidden.js b/api/responses/forbidden.js new file mode 100644 index 000000000..ca94852fe --- /dev/null +++ b/api/responses/forbidden.js @@ -0,0 +1,89 @@ +/** + * 403 (Forbidden) Handler + * + * Usage: + * return res.forbidden(); + * return res.forbidden(err); + * return res.forbidden(err, 'some/specific/forbidden/view'); + * + * e.g.: + * ``` + * return res.forbidden('Access denied.'); + * ``` + */ + +module.exports = function forbidden (data, options) { + + // Get access to `req`, `res`, & `sails` + var req = this.req; + var res = this.res; + var sails = req._sails; + + // Set status code + res.status(403); + + // Log error to console + if (data !== undefined) { + sails.log.verbose('Sending 403 ("Forbidden") response: \n',data); + } + else sails.log.verbose('Sending 403 ("Forbidden") response'); + + // Only include errors in response if application environment + // is not set to 'production'. In production, we shouldn't + // send back any identifying information about errors. + if (sails.config.environment === 'production' && sails.config.keepResponseErrors !== true) { + data = undefined; + } + + // If the user-agent wants JSON, always respond with JSON + // If views are disabled, revert to json + if (req.wantsJSON || sails.config.hooks.views === false) { + return res.jsonx(data); + } + + // If second argument is a string, we take that to mean it refers to a view. + // If it was omitted, use an empty object (`{}`) + options = (typeof options === 'string') ? { view: options } : options || {}; + + // Attempt to prettify data for views, if it's a non-error object + var viewData = data; + if (!(viewData instanceof Error) && 'object' == typeof viewData) { + try { + viewData = require('util').inspect(data, {depth: null}); + } + catch(e) { + viewData = undefined; + } + } + + // If a view was provided in options, serve it. + // Otherwise try to guess an appropriate view, or if that doesn't + // work, just send JSON. + if (options.view) { + return res.view(options.view, { data: viewData, title: 'Forbidden' }); + } + + // If no second argument provided, try to serve the default view, + // but fall back to sending JSON(P) if any errors occur. + else return res.view('403', { data: viewData, title: 'Forbidden' }, function (err, html) { + + // If a view error occured, fall back to JSON(P). + if (err) { + // + // Additionally: + // • If the view was missing, ignore the error but provide a verbose log. + if (err.code === 'E_VIEW_FAILED') { + sails.log.verbose('res.forbidden() :: Could not locate view for error page (sending JSON instead). Details: ',err); + } + // Otherwise, if this was a more serious error, log to the console with the details. + else { + sails.log.warn('res.forbidden() :: When attempting to render error page view, an error occured (sending JSON instead). Details: ', err); + } + return res.jsonx(data); + } + + return res.send(html); + }); + +}; + diff --git a/api/responses/notFound.js b/api/responses/notFound.js new file mode 100644 index 000000000..8f0cd035a --- /dev/null +++ b/api/responses/notFound.js @@ -0,0 +1,94 @@ +/** + * 404 (Not Found) Handler + * + * Usage: + * return res.notFound(); + * return res.notFound(err); + * return res.notFound(err, 'some/specific/notfound/view'); + * + * e.g.: + * ``` + * return res.notFound(); + * ``` + * + * NOTE: + * If a request doesn't match any explicit routes (i.e. `config/routes.js`) + * or route blueprints (i.e. "shadow routes", Sails will call `res.notFound()` + * automatically. + */ + +module.exports = function notFound (data, options) { + + // Get access to `req`, `res`, & `sails` + var req = this.req; + var res = this.res; + var sails = req._sails; + + // Set status code + res.status(404); + + // Log error to console + if (data !== undefined) { + sails.log.verbose('Sending 404 ("Not Found") response: \n',data); + } + else sails.log.verbose('Sending 404 ("Not Found") response'); + + // Only include errors in response if application environment + // is not set to 'production'. In production, we shouldn't + // send back any identifying information about errors. + if (sails.config.environment === 'production' && sails.config.keepResponseErrors !== true) { + data = undefined; + } + + // If the user-agent wants JSON, always respond with JSON + // If views are disabled, revert to json + if (req.wantsJSON || sails.config.hooks.views === false) { + return res.jsonx(data); + } + + // If second argument is a string, we take that to mean it refers to a view. + // If it was omitted, use an empty object (`{}`) + options = (typeof options === 'string') ? { view: options } : options || {}; + + // Attempt to prettify data for views, if it's a non-error object + var viewData = data; + if (!(viewData instanceof Error) && 'object' == typeof viewData) { + try { + viewData = require('util').inspect(data, {depth: null}); + } + catch(e) { + viewData = undefined; + } + } + + // If a view was provided in options, serve it. + // Otherwise try to guess an appropriate view, or if that doesn't + // work, just send JSON. + if (options.view) { + return res.view(options.view, { data: viewData, title: 'Not Found' }); + } + + // If no second argument provided, try to serve the default view, + // but fall back to sending JSON(P) if any errors occur. + else return res.view('404', { data: viewData, title: 'Not Found' }, function (err, html) { + + // If a view error occured, fall back to JSON(P). + if (err) { + // + // Additionally: + // • If the view was missing, ignore the error but provide a verbose log. + if (err.code === 'E_VIEW_FAILED') { + sails.log.verbose('res.notFound() :: Could not locate view for error page (sending JSON instead). Details: ',err); + } + // Otherwise, if this was a more serious error, log to the console with the details. + else { + sails.log.warn('res.notFound() :: When attempting to render error page view, an error occured (sending JSON instead). Details: ', err); + } + return res.jsonx(data); + } + + return res.send(html); + }); + +}; + diff --git a/api/responses/ok.js b/api/responses/ok.js new file mode 100644 index 000000000..eb701447e --- /dev/null +++ b/api/responses/ok.js @@ -0,0 +1,60 @@ +/** + * 200 (OK) Response + * + * Usage: + * return res.ok(); + * return res.ok(data); + * return res.ok(data, 'auth/login'); + * + * @param {Object} data + * @param {String|Object} options + * - pass string to render specified view + */ + +module.exports = function sendOK (data, options) { + + // Get access to `req`, `res`, & `sails` + var req = this.req; + var res = this.res; + var sails = req._sails; + + sails.log.silly('res.ok() :: Sending 200 ("OK") response'); + + // Set status code + res.status(200); + + // If appropriate, serve data as JSON(P) + // If views are disabled, revert to json + if (req.wantsJSON || sails.config.hooks.views === false) { + return res.jsonx(data); + } + + // If second argument is a string, we take that to mean it refers to a view. + // If it was omitted, use an empty object (`{}`) + options = (typeof options === 'string') ? { view: options } : options || {}; + + // Attempt to prettify data for views, if it's a non-error object + var viewData = data; + if (!(viewData instanceof Error) && 'object' == typeof viewData) { + try { + viewData = require('util').inspect(data, {depth: null}); + } + catch(e) { + viewData = undefined; + } + } + + // If a view was provided in options, serve it. + // Otherwise try to guess an appropriate view, or if that doesn't + // work, just send JSON. + if (options.view) { + return res.view(options.view, { data: viewData, title: 'OK' }); + } + + // If no second argument provided, try to serve the implied view, + // but fall back to sending JSON(P) if no view can be inferred. + else return res.guessView({ data: viewData, title: 'OK' }, function couldNotGuessView () { + return res.jsonx(data); + }); + +}; diff --git a/api/responses/serverError.js b/api/responses/serverError.js new file mode 100644 index 000000000..537c248cc --- /dev/null +++ b/api/responses/serverError.js @@ -0,0 +1,89 @@ +/** + * 500 (Server Error) Response + * + * Usage: + * return res.serverError(); + * return res.serverError(err); + * return res.serverError(err, 'some/specific/error/view'); + * + * NOTE: + * If something throws in a policy or controller, or an internal + * error is encountered, Sails will call `res.serverError()` + * automatically. + */ + +module.exports = function serverError (data, options) { + + // Get access to `req`, `res`, & `sails` + var req = this.req; + var res = this.res; + var sails = req._sails; + + // Set status code + res.status(500); + + // Log error to console + if (data !== undefined) { + sails.log.error('Sending 500 ("Server Error") response: \n',data); + } + else sails.log.error('Sending empty 500 ("Server Error") response'); + + // Only include errors in response if application environment + // is not set to 'production'. In production, we shouldn't + // send back any identifying information about errors. + if (sails.config.environment === 'production' && sails.config.keepResponseErrors !== true) { + data = undefined; + } + + // If the user-agent wants JSON, always respond with JSON + // If views are disabled, revert to json + if (req.wantsJSON || sails.config.hooks.views === false) { + return res.jsonx(data); + } + + // If second argument is a string, we take that to mean it refers to a view. + // If it was omitted, use an empty object (`{}`) + options = (typeof options === 'string') ? { view: options } : options || {}; + + // Attempt to prettify data for views, if it's a non-error object + var viewData = data; + if (!(viewData instanceof Error) && 'object' == typeof viewData) { + try { + viewData = require('util').inspect(data, {depth: null}); + } + catch(e) { + viewData = undefined; + } + } + + // If a view was provided in options, serve it. + // Otherwise try to guess an appropriate view, or if that doesn't + // work, just send JSON. + if (options.view) { + return res.view(options.view, { data: viewData, title: 'Server Error' }); + } + + // If no second argument provided, try to serve the default view, + // but fall back to sending JSON(P) if any errors occur. + else return res.view('500', { data: viewData, title: 'Server Error' }, function (err, html) { + + // If a view error occured, fall back to JSON(P). + if (err) { + // + // Additionally: + // • If the view was missing, ignore the error but provide a verbose log. + if (err.code === 'E_VIEW_FAILED') { + sails.log.verbose('res.serverError() :: Could not locate view for error page (sending JSON instead). Details: ',err); + } + // Otherwise, if this was a more serious error, log to the console with the details. + else { + sails.log.warn('res.serverError() :: When attempting to render error page view, an error occured (sending JSON instead). Details: ', err); + } + return res.jsonx(data); + } + + return res.send(html); + }); + +}; + diff --git a/api/services/.gitkeep b/api/services/.gitkeep new file mode 100644 index 000000000..e69de29bb diff --git a/app.js b/app.js new file mode 100644 index 000000000..6b7517088 --- /dev/null +++ b/app.js @@ -0,0 +1,60 @@ +/** + * app.js + * + * Use `app.js` to run your app without `sails lift`. + * To start the server, run: `node app.js`. + * + * This is handy in situations where the sails CLI is not relevant or useful. + * + * For example: + * => `node app.js` + * => `forever start app.js` + * => `node debug app.js` + * => `modulus deploy` + * => `heroku scale` + * + * + * The same command-line arguments are supported, e.g.: + * `node app.js --silent --port=80 --prod` + */ + + +// Ensure we're in the project directory, so cwd-relative paths work as expected +// no matter where we actually lift from. +// > Note: This is not required in order to lift, but it is a convenient default. +process.chdir(__dirname); + +// Attempt to import `sails`. +var sails; +try { + sails = require('sails'); +} catch (e) { + console.error('To run an app using `node app.js`, you usually need to have a version of `sails` installed in the same directory as your app.'); + console.error('To do that, run `npm install sails`'); + console.error(''); + console.error('Alternatively, if you have sails installed globally (i.e. you did `npm install -g sails`), you can use `sails lift`.'); + console.error('When you run `sails lift`, your app will still use a local `./node_modules/sails` dependency if it exists,'); + console.error('but if it doesn\'t, the app will run with the global sails instead!'); + return; +} + +// --• +// Try to get `rc` dependency (for loading `.sailsrc` files). +var rc; +try { + rc = require('rc'); +} catch (e0) { + try { + rc = require('sails/node_modules/rc'); + } catch (e1) { + console.error('Could not find dependency: `rc`.'); + console.error('Your `.sailsrc` file(s) will be ignored.'); + console.error('To resolve this, run:'); + console.error('npm install rc --save'); + rc = function () { return {}; }; + } +} + + +// Start server +sails.lift(rc('sails')); diff --git a/config/blueprints.js b/config/blueprints.js new file mode 100644 index 000000000..0bee92f5a --- /dev/null +++ b/config/blueprints.js @@ -0,0 +1,162 @@ +/** + * Blueprint API Configuration + * (sails.config.blueprints) + * + * These settings are for the global configuration of blueprint routes and + * request options (which impact the behavior of blueprint actions). + * + * You may also override any of these settings on a per-controller basis + * by defining a '_config' key in your controller definition, and assigning it + * a configuration object with overrides for the settings in this file. + * A lot of the configuration options below affect so-called "CRUD methods", + * or your controllers' `find`, `create`, `update`, and `destroy` actions. + * + * It's important to realize that, even if you haven't defined these yourself, as long as + * a model exists with the same name as the controller, Sails will respond with built-in CRUD + * logic in the form of a JSON API, including support for sort, pagination, and filtering. + * + * For more information on the blueprint API, check out: + * http://sailsjs.org/#!/documentation/reference/blueprint-api + * + * For more information on the settings in this file, see: + * http://sailsjs.org/#!/documentation/reference/sails.config/sails.config.blueprints.html + * + */ + +module.exports.blueprints = { + + /*************************************************************************** + * * + * Action routes speed up the backend development workflow by * + * eliminating the need to manually bind routes. When enabled, GET, POST, * + * PUT, and DELETE routes will be generated for every one of a controller's * + * actions. * + * * + * If an `index` action exists, additional naked routes will be created for * + * it. Finally, all `actions` blueprints support an optional path * + * parameter, `id`, for convenience. * + * * + * `actions` are enabled by default, and can be OK for production-- * + * however, if you'd like to continue to use controller/action autorouting * + * in a production deployment, you must take great care not to * + * inadvertently expose unsafe/unintentional controller logic to GET * + * requests. * + * * + ***************************************************************************/ + + // actions: true, + + /*************************************************************************** + * * + * RESTful routes (`sails.config.blueprints.rest`) * + * * + * REST blueprints are the automatically generated routes Sails uses to * + * expose a conventional REST API on top of a controller's `find`, * + * `create`, `update`, and `destroy` actions. * + * * + * For example, a BoatController with `rest` enabled generates the * + * following routes: * + * ::::::::::::::::::::::::::::::::::::::::::::::::::::::: * + * GET /boat -> BoatController.find * + * GET /boat/:id -> BoatController.findOne * + * POST /boat -> BoatController.create * + * PUT /boat/:id -> BoatController.update * + * DELETE /boat/:id -> BoatController.destroy * + * * + * `rest` blueprint routes are enabled by default, and are suitable for use * + * in a production scenario, as long you take standard security precautions * + * (combine w/ policies, etc.) * + * * + ***************************************************************************/ + + // rest: true, + + /*************************************************************************** + * * + * Shortcut routes are simple helpers to provide access to a * + * controller's CRUD methods from your browser's URL bar. When enabled, * + * GET, POST, PUT, and DELETE routes will be generated for the * + * controller's`find`, `create`, `update`, and `destroy` actions. * + * * + * `shortcuts` are enabled by default, but should be disabled in * + * production. * + * * + ***************************************************************************/ + + // shortcuts: true, + + /*************************************************************************** + * * + * An optional mount path for all blueprint routes on a controller, * + * including `rest`, `actions`, and `shortcuts`. This allows you to take * + * advantage of blueprint routing, even if you need to namespace your API * + * methods. * + * * + * (NOTE: This only applies to blueprint autoroutes, not manual routes from * + * `sails.config.routes`) * + * * + ***************************************************************************/ + + // prefix: '', + + /*************************************************************************** + * * + * An optional mount path for all REST blueprint routes on a controller. * + * And it do not include `actions` and `shortcuts` routes. * + * This allows you to take advantage of REST blueprint routing, * + * even if you need to namespace your RESTful API methods * + * * + ***************************************************************************/ + + // restPrefix: '', + + /*************************************************************************** + * * + * Whether to pluralize controller names in blueprint routes. * + * * + * (NOTE: This only applies to blueprint autoroutes, not manual routes from * + * `sails.config.routes`) * + * * + * For example, REST blueprints for `FooController` with `pluralize` * + * enabled: * + * GET /foos/:id? * + * POST /foos * + * PUT /foos/:id? * + * DELETE /foos/:id? * + * * + ***************************************************************************/ + + // pluralize: false, + + /*************************************************************************** + * * + * Whether the blueprint controllers should populate model fetches with * + * data from other models which are linked by associations * + * * + * If you have a lot of data in one-to-many associations, leaving this on * + * may result in very heavy api calls * + * * + ***************************************************************************/ + + // populate: true, + + /**************************************************************************** + * * + * Whether to run Model.watch() in the find and findOne blueprint actions. * + * Can be overridden on a per-model basis. * + * * + ****************************************************************************/ + + // autoWatch: true, + + /**************************************************************************** + * * + * The default number of records to show in the response from a "find" * + * action. Doubles as the default size of populated arrays if populate is * + * true. * + * * + ****************************************************************************/ + + // defaultLimit: 30 + +}; diff --git a/config/bootstrap.js b/config/bootstrap.js new file mode 100644 index 000000000..96208a186 --- /dev/null +++ b/config/bootstrap.js @@ -0,0 +1,17 @@ +/** + * Bootstrap + * (sails.config.bootstrap) + * + * An asynchronous bootstrap function that runs before your Sails app gets lifted. + * This gives you an opportunity to set up your data model, run jobs, or perform some special logic. + * + * For more information on bootstrapping your app, check out: + * http://sailsjs.org/#!/documentation/reference/sails.config/sails.config.bootstrap.html + */ + +module.exports.bootstrap = function(cb) { + + // It's very important to trigger this callback method when you are finished + // with the bootstrap! (otherwise your server will never lift, since it's waiting on the bootstrap) + cb(); +}; diff --git a/config/connections.js b/config/connections.js new file mode 100644 index 000000000..3eebb7c66 --- /dev/null +++ b/config/connections.js @@ -0,0 +1,92 @@ +/** + * Connections + * (sails.config.connections) + * + * `Connections` are like "saved settings" for your adapters. What's the difference between + * a connection and an adapter, you might ask? An adapter (e.g. `sails-mysql`) is generic-- + * it needs some additional information to work (e.g. your database host, password, user, etc.) + * A `connection` is that additional information. + * + * Each model must have a `connection` property (a string) which is references the name of one + * of these connections. If it doesn't, the default `connection` configured in `config/models.js` + * will be applied. Of course, a connection can (and usually is) shared by multiple models. + * . + * Note: If you're using version control, you should put your passwords/api keys + * in `config/local.js`, environment variables, or use another strategy. + * (this is to prevent you inadvertently sensitive credentials up to your repository.) + * + * For more information on configuration, check out: + * http://sailsjs.org/#!/documentation/reference/sails.config/sails.config.connections.html + */ + +module.exports.connections = { + + /*************************************************************************** + * * + * Local disk storage for DEVELOPMENT ONLY * + * * + * Installed by default. * + * * + ***************************************************************************/ + localDiskDb: { + adapter: 'sails-disk' + }, + + /*************************************************************************** + * * + * MySQL is the world's most popular relational database. * + * http://en.wikipedia.org/wiki/MySQL * + * * + * Run: npm install sails-mysql * + * * + ***************************************************************************/ + // someMysqlServer: { + // adapter: 'sails-mysql', + // host: 'YOUR_MYSQL_SERVER_HOSTNAME_OR_IP_ADDRESS', + // user: 'YOUR_MYSQL_USER', //optional + // password: 'YOUR_MYSQL_PASSWORD', //optional + // database: 'YOUR_MYSQL_DB' //optional + // }, + + /*************************************************************************** + * * + * MongoDB is the leading NoSQL database. * + * http://en.wikipedia.org/wiki/MongoDB * + * * + * Run: npm install sails-mongo * + * * + ***************************************************************************/ + // someMongodbServer: { + // adapter: 'sails-mongo', + // host: 'localhost', + // port: 27017, + // user: 'username', //optional + // password: 'password', //optional + // database: 'your_mongo_db_name_here' //optional + // }, + + /*************************************************************************** + * * + * PostgreSQL is another officially supported relational database. * + * http://en.wikipedia.org/wiki/PostgreSQL * + * * + * Run: npm install sails-postgresql * + * * + * * + ***************************************************************************/ + // somePostgresqlServer: { + // adapter: 'sails-postgresql', + // host: 'YOUR_POSTGRES_SERVER_HOSTNAME_OR_IP_ADDRESS', + // user: 'YOUR_POSTGRES_USER', // optional + // password: 'YOUR_POSTGRES_PASSWORD', // optional + // database: 'YOUR_POSTGRES_DB' //optional + // } + + + /*************************************************************************** + * * + * More adapters: https://github.com/balderdashy/sails * + * * + ***************************************************************************/ + +}; diff --git a/config/cors.js b/config/cors.js new file mode 100644 index 000000000..4a3c85a5f --- /dev/null +++ b/config/cors.js @@ -0,0 +1,78 @@ +/** + * Cross-Origin Resource Sharing (CORS) Settings + * (sails.config.cors) + * + * CORS is like a more modern version of JSONP-- it allows your server/API + * to successfully respond to requests from client-side JavaScript code + * running on some other domain (e.g. google.com) + * Unlike JSONP, it works with POST, PUT, and DELETE requests + * + * For more information on CORS, check out: + * http://en.wikipedia.org/wiki/Cross-origin_resource_sharing + * + * Note that any of these settings (besides 'allRoutes') can be changed on a per-route basis + * by adding a "cors" object to the route configuration: + * + * '/get foo': { + * controller: 'foo', + * action: 'bar', + * cors: { + * origin: 'http://foobar.com,https://owlhoot.com' + * } + * } + * + * For more information on this configuration file, see: + * http://sailsjs.org/#!/documentation/reference/sails.config/sails.config.cors.html + * + */ + +module.exports.cors = { + + /*************************************************************************** + * * + * Allow CORS on all routes by default? If not, you must enable CORS on a * + * per-route basis by either adding a "cors" configuration object to the * + * route config, or setting "cors:true" in the route config to use the * + * default settings below. * + * * + ***************************************************************************/ + + allRoutes: true, + + /*************************************************************************** + * * + * Which domains which are allowed CORS access? This can be a * + * comma-delimited list of hosts (beginning with http:// or https://) or * + * "*" to allow all domains CORS access. * + * * + ***************************************************************************/ + + origin: '*', + + /*************************************************************************** + * * + * Allow cookies to be shared for CORS requests? * + * * + ***************************************************************************/ + + // credentials: true, + + /*************************************************************************** + * * + * Which methods should be allowed for CORS requests? This is only used in * + * response to preflight requests (see article linked above for more info) * + * * + ***************************************************************************/ + + // methods: 'GET, POST, PUT, DELETE, OPTIONS, HEAD', + + /*************************************************************************** + * * + * Which headers should be allowed for CORS requests? This is only used in * + * response to preflight requests. * + * * + ***************************************************************************/ + + // headers: 'content-type' + +}; diff --git a/config/csrf.js b/config/csrf.js new file mode 100644 index 000000000..50bdbc2ba --- /dev/null +++ b/config/csrf.js @@ -0,0 +1,64 @@ +/** + * Cross-Site Request Forgery Protection Settings + * (sails.config.csrf) + * + * CSRF tokens are like a tracking chip. While a session tells the server that a user + * "is who they say they are", a csrf token tells the server "you are where you say you are". + * + * When enabled, all non-GET requests to the Sails server must be accompanied by + * a special token, identified as the '_csrf' parameter. + * + * This option protects your Sails app against cross-site request forgery (or CSRF) attacks. + * A would-be attacker needs not only a user's session cookie, but also this timestamped, + * secret CSRF token, which is refreshed/granted when the user visits a URL on your app's domain. + * + * This allows us to have certainty that our users' requests haven't been hijacked, + * and that the requests they're making are intentional and legitimate. + * + * This token has a short-lived expiration timeline, and must be acquired by either: + * + * (a) For traditional view-driven web apps: + * Fetching it from one of your views, where it may be accessed as + * a local variable, e.g.: + *
+ * + *
+ * + * or (b) For AJAX/Socket-heavy and/or single-page apps: + * Sending a GET request to the `/csrfToken` route, where it will be returned + * as JSON, e.g.: + * { _csrf: 'ajg4JD(JGdajhLJALHDa' } + * + * + * Enabling this option requires managing the token in your front-end app. + * For traditional web apps, it's as easy as passing the data from a view into a form action. + * In AJAX/Socket-heavy apps, just send a GET request to the /csrfToken route to get a valid token. + * + * For more information on CSRF, check out: + * http://en.wikipedia.org/wiki/Cross-site_request_forgery + * + * For more information on this configuration file, including info on CSRF + CORS, see: + * http://sailsjs.org/#!/documentation/reference/sails.config/sails.config.csrf.html + * + */ + +/**************************************************************************** +* * +* Enabled CSRF protection for your site? * +* * +****************************************************************************/ + +// module.exports.csrf = false; + +/**************************************************************************** +* * +* You may also specify more fine-grained settings for CSRF, including the * +* domains which are allowed to request the CSRF token via AJAX. These * +* settings override the general CORS settings in your config/cors.js file. * +* * +****************************************************************************/ + +// module.exports.csrf = { +// grantTokenViaAjax: true, +// origin: '' +// } diff --git a/config/env/development.js b/config/env/development.js new file mode 100644 index 000000000..371691626 --- /dev/null +++ b/config/env/development.js @@ -0,0 +1,24 @@ +/** + * Development environment settings + * + * This file can include shared settings for a development team, + * such as API keys or remote database passwords. If you're using + * a version control solution for your Sails app, this file will + * be committed to your repository unless you add it to your .gitignore + * file. If your repository will be publicly viewable, don't add + * any private information to this file! + * + */ + +module.exports = { + + /*************************************************************************** + * Set the default database connection for models in the development * + * environment (see config/connections.js and config/models.js ) * + ***************************************************************************/ + + // models: { + // connection: 'someMongodbServer' + // } + +}; diff --git a/config/env/production.js b/config/env/production.js new file mode 100644 index 000000000..60dd60286 --- /dev/null +++ b/config/env/production.js @@ -0,0 +1,38 @@ +/** + * Production environment settings + * + * This file can include shared settings for a production environment, + * such as API keys or remote database passwords. If you're using + * a version control solution for your Sails app, this file will + * be committed to your repository unless you add it to your .gitignore + * file. If your repository will be publicly viewable, don't add + * any private information to this file! + * + */ + +module.exports = { + + /*************************************************************************** + * Set the default database connection for models in the production * + * environment (see config/connections.js and config/models.js ) * + ***************************************************************************/ + + // models: { + // connection: 'someMysqlServer' + // }, + + /*************************************************************************** + * Set the port in the production environment to 80 * + ***************************************************************************/ + + // port: 80, + + /*************************************************************************** + * Set the log level in production environment to "silent" * + ***************************************************************************/ + + // log: { + // level: "silent" + // } + +}; diff --git a/config/globals.js b/config/globals.js new file mode 100644 index 000000000..c1819c334 --- /dev/null +++ b/config/globals.js @@ -0,0 +1,63 @@ +/** + * Global Variable Configuration + * (sails.config.globals) + * + * Configure which global variables which will be exposed + * automatically by Sails. + * + * For more information on configuration, check out: + * http://sailsjs.org/#!/documentation/reference/sails.config/sails.config.globals.html + */ +module.exports.globals = { + + /**************************************************************************** + * * + * Expose the lodash installed in Sails core as a global variable. If this * + * is disabled, like any other node module you can always run npm install * + * lodash --save, then var _ = require('lodash') at the top of any file. * + * * + ****************************************************************************/ + + // _: true, + + /**************************************************************************** + * * + * Expose the async installed in Sails core as a global variable. If this is * + * disabled, like any other node module you can always run npm install async * + * --save, then var async = require('async') at the top of any file. * + * * + ****************************************************************************/ + + // async: true, + + /**************************************************************************** + * * + * Expose the sails instance representing your app. If this is disabled, you * + * can still get access via req._sails. * + * * + ****************************************************************************/ + + // sails: true, + + /**************************************************************************** + * * + * Expose each of your app's services as global variables (using their * + * "globalId"). E.g. a service defined in api/models/NaturalLanguage.js * + * would have a globalId of NaturalLanguage by default. If this is disabled, * + * you can still access your services via sails.services.* * + * * + ****************************************************************************/ + + // services: true, + + /**************************************************************************** + * * + * Expose each of your app's models as global variables (using their * + * "globalId"). E.g. a model defined in api/models/User.js would have a * + * globalId of User by default. If this is disabled, you can still access * + * your models via sails.models.*. * + * * + ****************************************************************************/ + + // models: true +}; diff --git a/config/http.js b/config/http.js new file mode 100644 index 000000000..1d096f498 --- /dev/null +++ b/config/http.js @@ -0,0 +1,93 @@ +/** + * HTTP Server Settings + * (sails.config.http) + * + * Configuration for the underlying HTTP server in Sails. + * Only applies to HTTP requests (not WebSockets) + * + * For more information on configuration, check out: + * http://sailsjs.org/#!/documentation/reference/sails.config/sails.config.http.html + */ + +module.exports.http = { + + /**************************************************************************** + * * + * Express middleware to use for every Sails request. To add custom * + * middleware to the mix, add a function to the middleware config object and * + * add its key to the "order" array. The $custom key is reserved for * + * backwards-compatibility with Sails v0.9.x apps that use the * + * `customMiddleware` config option. * + * * + ****************************************************************************/ + + middleware: { + + /*************************************************************************** + * * + * The order in which middleware should be run for HTTP request. (the Sails * + * router is invoked by the "router" middleware below.) * + * * + ***************************************************************************/ + + // order: [ + // 'startRequestTimer', + // 'cookieParser', + // 'session', + // 'myRequestLogger', + // 'bodyParser', + // 'handleBodyParserError', + // 'compress', + // 'methodOverride', + // 'poweredBy', + // '$custom', + // 'router', + // 'www', + // 'favicon', + // '404', + // '500' + // ], + + /**************************************************************************** + * * + * Example custom middleware; logs each request to the console. * + * * + ****************************************************************************/ + + // myRequestLogger: function (req, res, next) { + // console.log("Requested :: ", req.method, req.url); + // return next(); + // } + + + /*************************************************************************** + * * + * The body parser that will handle incoming multipart HTTP requests. By * + * default as of v0.10, Sails uses * + * [skipper](http://github.com/balderdashy/skipper). See * + * http://www.senchalabs.org/connect/multipart.html for other options. * + * * + * Note that Sails uses an internal instance of Skipper by default; to * + * override it and specify more options, make sure to "npm install skipper" * + * in your project first. You can also specify a different body parser or * + * a custom function with req, res and next parameters (just like any other * + * middleware function). * + * * + ***************************************************************************/ + + // bodyParser: require('skipper')({strict: true}) + + }, + + /*************************************************************************** + * * + * The number of seconds to cache flat files on disk being served by * + * Express static middleware (by default, these files are in `.tmp/public`) * + * * + * The HTTP static cache is only active in a 'production' environment, * + * since that's the only time Express will cache flat-files. * + * * + ***************************************************************************/ + + // cache: 31557600000 +}; diff --git a/config/i18n.js b/config/i18n.js new file mode 100644 index 000000000..facf7070f --- /dev/null +++ b/config/i18n.js @@ -0,0 +1,57 @@ +/** + * Internationalization / Localization Settings + * (sails.config.i18n) + * + * If your app will touch people from all over the world, i18n (or internationalization) + * may be an important part of your international strategy. + * + * + * For more informationom i18n in Sails, check out: + * http://sailsjs.org/#!/documentation/concepts/Internationalization + * + * For a complete list of i18n options, see: + * https://github.com/mashpie/i18n-node#list-of-configuration-options + * + * + */ + +module.exports.i18n = { + + /*************************************************************************** + * * + * Which locales are supported? * + * * + ***************************************************************************/ + + // locales: ['en', 'es', 'fr', 'de'], + + /**************************************************************************** + * * + * What is the default locale for the site? Note that this setting will be * + * overridden for any request that sends an "Accept-Language" header (i.e. * + * most browsers), but it's still useful if you need to localize the * + * response for requests made by non-browser clients (e.g. cURL). * + * * + ****************************************************************************/ + + // defaultLocale: 'en', + + /**************************************************************************** + * * + * Automatically add new keys to locale (translation) files when they are * + * encountered during a request? * + * * + ****************************************************************************/ + + // updateFiles: false, + + /**************************************************************************** + * * + * Path (relative to app root) of directory to store locale (translation) * + * files in. * + * * + ****************************************************************************/ + + // localesDirectory: '/config/locales' + +}; diff --git a/config/locales/_README.md b/config/locales/_README.md new file mode 100644 index 000000000..5f89b1548 --- /dev/null +++ b/config/locales/_README.md @@ -0,0 +1,28 @@ +# Internationalization / Localization Settings + +> Also see the official docs on internationalization/localization: +> http://links.sailsjs.org/docs/config/locales + +## Locales +All locale files live under `config/locales`. Here is where you can add translations +as JSON key-value pairs. The name of the file should match the language that you are supporting, which allows for automatic language detection based on request headers. + +Here is an example locale stringfile for the Spanish language (`config/locales/es.json`): +```json +{ + "Hello!": "Hola!", + "Hello %s, how are you today?": "¿Hola %s, como estas?", +} +``` +## Usage +Locales can be accessed in controllers/policies through `res.i18n()`, or in views through the `__(key)` or `i18n(key)` functions. +Remember that the keys are case sensitive and require exact key matches, e.g. + +```ejs +

<%= __('Welcome to PencilPals!') %>

+

<%= i18n('Hello %s, how are you today?', 'Pencil Maven') %>

+

<%= i18n('That\'s right-- you can use either i18n() or __()') %>

+``` + +## Configuration +Localization/internationalization config can be found in `config/i18n.js`, from where you can set your supported locales. diff --git a/config/locales/de.json b/config/locales/de.json new file mode 100644 index 000000000..12f23d94a --- /dev/null +++ b/config/locales/de.json @@ -0,0 +1,4 @@ +{ + "Welcome": "Willkommen", + "A brand new app.": "Eine neue App." +} diff --git a/config/locales/en.json b/config/locales/en.json new file mode 100644 index 000000000..6eeeb33c6 --- /dev/null +++ b/config/locales/en.json @@ -0,0 +1,4 @@ +{ + "Welcome": "Welcome", + "A brand new app.": "A brand new app." +} diff --git a/config/locales/es.json b/config/locales/es.json new file mode 100644 index 000000000..bc6edbf03 --- /dev/null +++ b/config/locales/es.json @@ -0,0 +1,4 @@ +{ + "Welcome": "Bienvenido", + "A brand new app.": "Una nueva aplicación." +} diff --git a/config/locales/fr.json b/config/locales/fr.json new file mode 100644 index 000000000..972935c6a --- /dev/null +++ b/config/locales/fr.json @@ -0,0 +1,4 @@ +{ + "Welcome": "Bienvenue", + "A brand new app.": "Une toute nouvelle application." +} diff --git a/config/log.js b/config/log.js new file mode 100644 index 000000000..e3d1a97bd --- /dev/null +++ b/config/log.js @@ -0,0 +1,30 @@ +/** + * Built-in Log Configuration + * (sails.config.log) + * + * Configure the log level for your app, as well as the transport + * (Underneath the covers, Sails uses Winston for logging, which + * allows for some pretty neat custom transports/adapters for log messages) + * + * For more information on the Sails logger, check out: + * http://sailsjs.org/#!/documentation/concepts/Logging + */ + +module.exports.log = { + + /*************************************************************************** + * * + * Valid `level` configs: i.e. the minimum log level to capture with * + * sails.log.*() * + * * + * The order of precedence for log levels from lowest to highest is: * + * silly, verbose, info, debug, warn, error * + * * + * You may also set the level to "silent" to suppress all logs. * + * * + ***************************************************************************/ + + // level: 'debug', + // noShip: true + +}; diff --git a/config/models.js b/config/models.js new file mode 100644 index 000000000..b1f08ba60 --- /dev/null +++ b/config/models.js @@ -0,0 +1,32 @@ +/** + * Default model configuration + * (sails.config.models) + * + * Unless you override them, the following properties will be included + * in each of your models. + * + * For more info on Sails models, see: + * http://sailsjs.org/#!/documentation/concepts/ORM + */ + +module.exports.models = { + + /*************************************************************************** + * * + * Your app's default connection. i.e. the name of one of your app's * + * connections (see `config/connections.js`) * + * * + ***************************************************************************/ + // connection: 'localDiskDb', + + /*************************************************************************** + * * + * How and whether Sails will attempt to automatically rebuild the * + * tables/collections/etc. in your schema. * + * * + * See http://sailsjs.org/#!/documentation/concepts/ORM/model-settings.html * + * * + ***************************************************************************/ + migrate: 'alter' + +}; diff --git a/config/policies.js b/config/policies.js new file mode 100644 index 000000000..1785d4ef6 --- /dev/null +++ b/config/policies.js @@ -0,0 +1,51 @@ +/** + * Policy Mappings + * (sails.config.policies) + * + * Policies are simple functions which run **before** your controllers. + * You can apply one or more policies to a given controller, or protect + * its actions individually. + * + * Any policy file (e.g. `api/policies/authenticated.js`) can be accessed + * below by its filename, minus the extension, (e.g. "authenticated") + * + * For more information on how policies work, see: + * http://sailsjs.org/#!/documentation/concepts/Policies + * + * For more information on configuring policies, check out: + * http://sailsjs.org/#!/documentation/reference/sails.config/sails.config.policies.html + */ + + +module.exports.policies = { + + /*************************************************************************** + * * + * Default policy for all controllers and actions (`true` allows public * + * access) * + * * + ***************************************************************************/ + + // '*': true, + + /*************************************************************************** + * * + * Here's an example of mapping some policies to run before a controller * + * and its actions * + * * + ***************************************************************************/ + // RabbitController: { + + // Apply the `false` policy as the default for all of RabbitController's actions + // (`false` prevents all access, which ensures that nothing bad happens to our rabbits) + // '*': false, + + // For the action `nurture`, apply the 'isRabbitMother' policy + // (this overrides `false` above) + // nurture : 'isRabbitMother', + + // Apply the `isNiceToAnimals` AND `hasRabbitFood` policies + // before letting any users feed our rabbits + // feed : ['isNiceToAnimals', 'hasRabbitFood'] + // } +}; diff --git a/config/routes.js b/config/routes.js new file mode 100644 index 000000000..cf1f7eddd --- /dev/null +++ b/config/routes.js @@ -0,0 +1,49 @@ +/** + * Route Mappings + * (sails.config.routes) + * + * Your routes map URLs to views and controllers. + * + * If Sails receives a URL that doesn't match any of the routes below, + * it will check for matching files (images, scripts, stylesheets, etc.) + * in your assets directory. e.g. `http://localhost:1337/images/foo.jpg` + * might match an image file: `/assets/images/foo.jpg` + * + * Finally, if those don't match either, the default 404 handler is triggered. + * See `api/responses/notFound.js` to adjust your app's 404 logic. + * + * Note: Sails doesn't ACTUALLY serve stuff from `assets`-- the default Gruntfile in Sails copies + * flat files from `assets` to `.tmp/public`. This allows you to do things like compile LESS or + * CoffeeScript for the front-end. + * + * For more information on configuring custom routes, check out: + * http://sailsjs.org/#!/documentation/concepts/Routes/RouteTargetSyntax.html + */ + +module.exports.routes = { + + /*************************************************************************** + * * + * Make the view located at `views/homepage.ejs` (or `views/homepage.jade`, * + * etc. depending on your default view engine) your home page. * + * * + * (Alternatively, remove this and add an `index.html` file in your * + * `assets` directory) * + * * + ***************************************************************************/ + + // '/': { + // view: 'homepage' + // } + + /*************************************************************************** + * * + * Custom routes here... * + * * + * If a request to a URL doesn't match any of the custom routes above, it * + * is matched against Sails route blueprints. See `config/blueprints.js` * + * for configuration options and examples. * + * * + ***************************************************************************/ + +}; diff --git a/config/session.js b/config/session.js new file mode 100644 index 000000000..30e2e47f6 --- /dev/null +++ b/config/session.js @@ -0,0 +1,100 @@ +/** + * Session Configuration + * (sails.config.session) + * + * Sails session integration leans heavily on the great work already done by + * Express, but also unifies Socket.io with the Connect session store. It uses + * Connect's cookie parser to normalize configuration differences between Express + * and Socket.io and hooks into Sails' middleware interpreter to allow you to access + * and auto-save to `req.session` with Socket.io the same way you would with Express. + * + * For more information on configuring the session, check out: + * http://sailsjs.org/#!/documentation/reference/sails.config/sails.config.session.html + */ + +module.exports.session = { + + /*************************************************************************** + * * + * Session secret is automatically generated when your new app is created * + * Replace at your own risk in production-- you will invalidate the cookies * + * of your users, forcing them to log in again. * + * * + ***************************************************************************/ + secret: '183ba7cc72e6ca9c8d1c5cc7b6f60af5', + + + /*************************************************************************** + * * + * Set the session cookie expire time The maxAge is set by milliseconds, * + * the example below is for 24 hours * + * * + ***************************************************************************/ + + // cookie: { + // maxAge: 24 * 60 * 60 * 1000 + // }, + + /*************************************************************************** + * * + * Uncomment the following lines to set up a Redis session store that can * + * be shared across multiple Sails.js servers. * + * * + * Requires connect-redis (https://www.npmjs.com/package/connect-redis) * + * * + ***************************************************************************/ + + // adapter: 'redis', + + /*************************************************************************** + * * + * The following values are optional, if no options are set a redis * + * instance running on localhost is expected. Read more about options at: * + * * + * https://github.com/visionmedia/connect-redis * + * * + ***************************************************************************/ + + // host: 'localhost', + // port: 6379, + // ttl: , + // db: 0, + // pass: , + // prefix: 'sess:', + + + /*************************************************************************** + * * + * Uncomment the following lines to set up a MongoDB session store that can * + * be shared across multiple Sails.js servers. * + * * + * Requires connect-mongo (https://www.npmjs.com/package/connect-mongo) * + * Use version 0.8.2 with Node version <= 0.12 * + * Use the latest version with Node >= 4.0 * + * * + ***************************************************************************/ + + // adapter: 'mongo', + // url: 'mongodb://user:password@localhost:27017/dbname', // user, password and port optional + + /*************************************************************************** + * * + * Optional Values: * + * * + * See https://github.com/kcbanner/connect-mongo for more * + * information about connect-mongo options. * + * * + * See http://bit.ly/mongooptions for more information about options * + * available in `mongoOptions` * + * * + ***************************************************************************/ + + // collection: 'sessions', + // stringify: true, + // mongoOptions: { + // server: { + // ssl: true + // } + // } + +}; diff --git a/config/sockets.js b/config/sockets.js new file mode 100644 index 000000000..5c2e3775e --- /dev/null +++ b/config/sockets.js @@ -0,0 +1,141 @@ +/** + * WebSocket Server Settings + * (sails.config.sockets) + * + * These settings provide transparent access to the options for Sails' + * encapsulated WebSocket server, as well as some additional Sails-specific + * configuration layered on top. + * + * For more information on sockets configuration, including advanced config options, see: + * http://sailsjs.org/#!/documentation/reference/sails.config/sails.config.sockets.html + */ + +module.exports.sockets = { + + + /*************************************************************************** + * * + * Node.js (and consequently Sails.js) apps scale horizontally. It's a * + * powerful, efficient approach, but it involves a tiny bit of planning. At * + * scale, you'll want to be able to copy your app onto multiple Sails.js * + * servers and throw them behind a load balancer. * + * * + * One of the big challenges of scaling an application is that these sorts * + * of clustered deployments cannot share memory, since they are on * + * physically different machines. On top of that, there is no guarantee * + * that a user will "stick" with the same server between requests (whether * + * HTTP or sockets), since the load balancer will route each request to the * + * Sails server with the most available resources. However that means that * + * all room/pubsub/socket processing and shared memory has to be offloaded * + * to a shared, remote messaging queue (usually Redis) * + * * + * Luckily, Socket.io (and consequently Sails.js) apps support Redis for * + * sockets by default. To enable a remote redis pubsub server, uncomment * + * the config below. * + * * + * Worth mentioning is that, if `adapter` config is `redis`, but host/port * + * is left unset, Sails will try to connect to redis running on localhost * + * via port 6379 * + * * + ***************************************************************************/ + // adapter: 'memory', + + // + // -OR- + // + + // adapter: 'socket.io-redis', + // host: '127.0.0.1', + // port: 6379, + // db: 0, + // pass: '', + + + + /*************************************************************************** + * * + * Whether to expose a 'get /__getcookie' route with CORS support that sets * + * a cookie (this is used by the sails.io.js socket client to get access to * + * a 3rd party cookie and to enable sessions). * + * * + * Warning: Currently in this scenario, CORS settings apply to interpreted * + * requests sent via a socket.io connection that used this cookie to * + * connect, even for non-browser clients! (e.g. iOS apps, toasters, node.js * + * unit tests) * + * * + ***************************************************************************/ + + // grant3rdPartyCookie: true, + + + + /*************************************************************************** + * * + * `beforeConnect` * + * * + * This custom beforeConnect function will be run each time BEFORE a new * + * socket is allowed to connect, when the initial socket.io handshake is * + * performed with the server. * + * * + * By default, when a socket tries to connect, Sails allows it, every time. * + * (much in the same way any HTTP request is allowed to reach your routes. * + * If no valid cookie was sent, a temporary session will be created for the * + * connecting socket. * + * * + * If the cookie sent as part of the connection request doesn't match any * + * known user session, a new user session is created for it. * + * * + * In most cases, the user would already have a cookie since they loaded * + * the socket.io client and the initial HTML page you're building. * + * * + * However, in the case of cross-domain requests, it is possible to receive * + * a connection upgrade request WITHOUT A COOKIE (for certain transports) * + * In this case, there is no way to keep track of the requesting user * + * between requests, since there is no identifying information to link * + * him/her with a session. The sails.io.js client solves this by connecting * + * to a CORS/jsonp endpoint first to get a 3rd party cookie(fortunately this* + * works, even in Safari), then opening the connection. * + * * + * You can also pass along a ?cookie query parameter to the upgrade url, * + * which Sails will use in the absence of a proper cookie e.g. (when * + * connecting from the client): * + * io.sails.connect('http://localhost:1337?cookie=smokeybear') * + * * + * Finally note that the user's cookie is NOT (and will never be) accessible* + * from client-side javascript. Using HTTP-only cookies is crucial for your * + * app's security. * + * * + ***************************************************************************/ + // beforeConnect: function(handshake, cb) { + // // `true` allows the connection + // return cb(null, true); + // + // // (`false` would reject the connection) + // }, + + + /*************************************************************************** + * * + * `afterDisconnect` * + * * + * This custom afterDisconnect function will be run each time a socket * + * disconnects * + * * + ***************************************************************************/ + // afterDisconnect: function(session, socket, cb) { + // // By default: do nothing. + // return cb(); + // }, + + /*************************************************************************** + * * + * `transports` * + * * + * A array of allowed transport methods which the clients will try to use. * + * On server environments that don't support sticky sessions, the "polling" * + * transport should be disabled. * + * * + ***************************************************************************/ + // transports: ["polling", "websocket"] + +}; diff --git a/config/views.js b/config/views.js new file mode 100644 index 000000000..a5addf70c --- /dev/null +++ b/config/views.js @@ -0,0 +1,95 @@ +/** + * View Engine Configuration + * (sails.config.views) + * + * Server-sent views are a classic and effective way to get your app up + * and running. Views are normally served from controllers. Below, you can + * configure your templating language/framework of choice and configure + * Sails' layout support. + * + * For more information on views and layouts, check out: + * http://sailsjs.org/#!/documentation/concepts/Views + */ + +module.exports.views = { + + /**************************************************************************** + * * + * View engine (aka template language) to use for your app's *server-side* * + * views * + * * + * Sails+Express supports all view engines which implement TJ Holowaychuk's * + * `consolidate.js`, including, but not limited to: * + * * + * ejs, jade, handlebars, mustache underscore, hogan, haml, haml-coffee, * + * dust atpl, eco, ect, jazz, jqtpl, JUST, liquor, QEJS, swig, templayed, * + * toffee, walrus, & whiskers * + * * + * For more options, check out the docs: * + * https://github.com/balderdashy/sails-wiki/blob/0.9/config.views.md#engine * + * * + ****************************************************************************/ + + // engine: 'ejs', + + + /**************************************************************************** + * * + * Layouts are simply top-level HTML templates you can use as wrappers for * + * your server-side views. If you're using ejs or jade, you can take * + * advantage of Sails' built-in `layout` support. * + * * + * When using a layout, when one of your views is served, it is injected * + * into the `body` partial defined in the layout. This lets you reuse header * + * and footer logic between views. * + * * + * NOTE: Layout support is only implemented for the `ejs` view engine! * + * For most other engines, it is not necessary, since they implement * + * partials/layouts themselves. In those cases, this config will be * + * silently ignored. * + * * + * The `layout` setting may be set to one of the following: * + * * + * If `false`, layouts will be disabled. Otherwise, if a string is * + * specified, it will be interpreted as the relative path to your layout * + * file from `views/` folder. (the file extension, ".ejs", should be * + * omitted) * + * * + ****************************************************************************/ + + /**************************************************************************** + * * + * Using Multiple Layouts * + * * + * If you're using the default `ejs` or `handlebars` Sails supports the use * + * of multiple `layout` files. To take advantage of this, before rendering a * + * view, override the `layout` local in your controller by setting * + * `res.locals.layout`. (this is handy if you parts of your app's UI look * + * completely different from each other) * + * * + * e.g. your default might be * + * layout: 'layouts/public' * + * * + * But you might override that in some of your controllers with: * + * layout: 'layouts/internal' * + * * + ****************************************************************************/ + + //layout: 'layout', + + /**************************************************************************** + * * + * Partials are simply top-level snippets you can leverage to reuse template * + * for your server-side views. If you're using handlebars, you can take * + * advantage of Sails' built-in `partials` support. * + * * + * If `false` or empty partials will be located in the same folder as views. * + * Otherwise, if a string is specified, it will be interpreted as the * + * relative path to your partial files from `views/` folder. * + * * + ****************************************************************************/ + + // partials: false + + +}; diff --git a/package.json b/package.json index ddc784ddd..fdd436673 100644 --- a/package.json +++ b/package.json @@ -1,9 +1,9 @@ { - "name": "mws-restaurant-stage-1", + "name": "mws-restaurant-stage-2", "version": "0.1.0", "repository": { "type": "git", - "url": "https://github.com/mejarc/mws-restaurant-stage-1.git" + "url": "https://github.com/mejarc/mws-restaurant-stage-2.git" }, "devDependencies": { "grunt": "~0.4.5", diff --git a/server.js b/server.js new file mode 100644 index 000000000..6b7517088 --- /dev/null +++ b/server.js @@ -0,0 +1,60 @@ +/** + * app.js + * + * Use `app.js` to run your app without `sails lift`. + * To start the server, run: `node app.js`. + * + * This is handy in situations where the sails CLI is not relevant or useful. + * + * For example: + * => `node app.js` + * => `forever start app.js` + * => `node debug app.js` + * => `modulus deploy` + * => `heroku scale` + * + * + * The same command-line arguments are supported, e.g.: + * `node app.js --silent --port=80 --prod` + */ + + +// Ensure we're in the project directory, so cwd-relative paths work as expected +// no matter where we actually lift from. +// > Note: This is not required in order to lift, but it is a convenient default. +process.chdir(__dirname); + +// Attempt to import `sails`. +var sails; +try { + sails = require('sails'); +} catch (e) { + console.error('To run an app using `node app.js`, you usually need to have a version of `sails` installed in the same directory as your app.'); + console.error('To do that, run `npm install sails`'); + console.error(''); + console.error('Alternatively, if you have sails installed globally (i.e. you did `npm install -g sails`), you can use `sails lift`.'); + console.error('When you run `sails lift`, your app will still use a local `./node_modules/sails` dependency if it exists,'); + console.error('but if it doesn\'t, the app will run with the global sails instead!'); + return; +} + +// --• +// Try to get `rc` dependency (for loading `.sailsrc` files). +var rc; +try { + rc = require('rc'); +} catch (e0) { + try { + rc = require('sails/node_modules/rc'); + } catch (e1) { + console.error('Could not find dependency: `rc`.'); + console.error('Your `.sailsrc` file(s) will be ignored.'); + console.error('To resolve this, run:'); + console.error('npm install rc --save'); + rc = function () { return {}; }; + } +} + + +// Start server +sails.lift(rc('sails'));