Add attendees and suggestions (except for UI)

This commit is contained in:
Aaron Lindsay 2016-12-26 08:29:18 -05:00
parent c4f07d3b3e
commit c8232050da
11 changed files with 283 additions and 14 deletions

View File

@ -15,10 +15,19 @@ type Attendee struct {
Date time.Time `json:"-"` Date time.Time `json:"-"`
} }
type PopularAttendee struct {
Name string
Popularity int64
}
type AttendeeList struct { type AttendeeList struct {
Attendees *[]*Attendee `json:"attendees"` Attendees *[]*Attendee `json:"attendees"`
} }
type PopularAttendeeList struct {
PopularAttendees *[]*PopularAttendee `json:"popularattendees"`
}
func (a *Attendee) Write(w http.ResponseWriter) error { func (a *Attendee) Write(w http.ResponseWriter) error {
enc := json.NewEncoder(w) enc := json.NewEncoder(w)
return enc.Encode(a) return enc.Encode(a)
@ -34,6 +43,16 @@ func (al *AttendeeList) Write(w http.ResponseWriter) error {
return enc.Encode(al) return enc.Encode(al)
} }
func (pa *PopularAttendee) Write(w http.ResponseWriter) error {
enc := json.NewEncoder(w)
return enc.Encode(pa)
}
func (pal *PopularAttendeeList) Write(w http.ResponseWriter) error {
enc := json.NewEncoder(w)
return enc.Encode(pal)
}
type AttendeeExistsError struct{} type AttendeeExistsError struct{}
func (aeu AttendeeExistsError) Error() string { func (aeu AttendeeExistsError) Error() string {
@ -50,6 +69,29 @@ func GetAttendees(userid int64, date time.Time) (*[]*Attendee, error) {
return &attendees, nil return &attendees, nil
} }
func GetPopularAttendees() (*[]*PopularAttendee, error) {
var attendees []*Attendee
var attendeeMap map[string]int64
popularAttendees := make([]*PopularAttendee, 0)
_, err := DB.Select(&attendees, "SELECT * from attendees")
if err != nil {
return nil, err
}
for i := range attendees {
attendeeMap[attendees[i].Name] += 1
}
for name, count := range attendeeMap {
var popularAttendee PopularAttendee
popularAttendee.Name = name
popularAttendee.Popularity = count
popularAttendees = append(popularAttendees, &popularAttendee)
}
return &popularAttendees, nil
}
func InsertAttendee(a *Attendee) error { func InsertAttendee(a *Attendee) error {
transaction, err := DB.Begin() transaction, err := DB.Begin()
if err != nil { if err != nil {
@ -148,3 +190,28 @@ func AttendeeHandler(w http.ResponseWriter, r *http.Request) {
return return
} }
} }
func PopularAttendeeHandler(w http.ResponseWriter, r *http.Request) {
if r.Method == "GET" {
var pal PopularAttendeeList
attendees, err := GetPopularAttendees()
if err != nil {
WriteError(w, 999 /*Internal Error*/)
log.Print(err)
return
}
pal.PopularAttendees = attendees
err = (&pal).Write(w)
if err != nil {
WriteError(w, 999 /*Internal Error*/)
log.Print(err)
return
}
} else {
/* No POST, PUT, or DELETE */
WriteError(w, 3 /*Invalid Request*/)
return
}
}

View File

@ -32,6 +32,19 @@ function attendeeCreated(attendee) {
} }
} }
function fetchPopularAttendees() {
return {
type: AttendeeConstants.FETCH_POPULAR_ATTENDEES
}
}
function popularAttendeesFetched(attendees) {
return {
type: AttendeeConstants.POPULAR_ATTENDEES_FETCHED,
attendees: attendees
}
}
function fetchAll() { function fetchAll() {
return function (dispatch) { return function (dispatch) {
dispatch(fetchAttendees()); dispatch(fetchAttendees());
@ -87,7 +100,36 @@ function create(attendee) {
}; };
} }
function fetchPopular() {
return function (dispatch) {
dispatch(fetchPopularAttendees());
$.ajax({
type: "GET",
dataType: "json",
url: "popularattendees/",
success: function(data, status, jqXHR) {
var e = new Error();
e.fromJSON(data);
if (e.isError()) {
ErrorActions.serverError(e);
} else {
dispatch(popularAttendeesFetched(data.attendees.map(function(json) {
var a = new Attendee();
a.fromJSON(json);
return a;
})));
}
},
error: function(jqXHR, status, error) {
ErrorActions.ajaxError(e);
}
});
};
}
module.exports = { module.exports = {
fetchAll: fetchAll, fetchAll: fetchAll,
create: create create: create,
fetchPopular: fetchPopular
}; };

View File

@ -12,10 +12,10 @@ function fetchSuggestions() {
} }
} }
function restaurantsFetched(restaurants) { function suggestionsFetched(suggestions) {
return { return {
type: SuggestionConstants.SUGGESTIONS_FETCHED, type: SuggestionConstants.SUGGESTIONS_FETCHED,
restaurants: restaurants suggestions: suggestions
} }
} }
@ -25,10 +25,23 @@ function createSuggestion() {
} }
} }
function restaurantCreated(restaurant) { function suggestionCreated(suggestion) {
return { return {
type: SuggestionConstants.SUGGESTION_CREATED, type: SuggestionConstants.SUGGESTION_CREATED,
restaurant: restaurant suggestion: suggestion
}
}
function fetchPopularSuggestions() {
return {
type: SuggestionConstants.FETCH_POPULAR_SUGGESTIONS
}
}
function popularSuggestionsFetched(suggestions) {
return {
type: SuggestionConstants.POPULAR_SUGGESTIONS_FETCHED,
suggestions: suggestions
} }
} }
@ -39,14 +52,14 @@ function fetchAll() {
$.ajax({ $.ajax({
type: "GET", type: "GET",
dataType: "json", dataType: "json",
url: "restaurant/", url: "suggestion/",
success: function(data, status, jqXHR) { success: function(data, status, jqXHR) {
var e = new Error(); var e = new Error();
e.fromJSON(data); e.fromJSON(data);
if (e.isError()) { if (e.isError()) {
ErrorActions.serverError(e); ErrorActions.serverError(e);
} else { } else {
dispatch(restaurantsFetched(data.restaurants.map(function(json) { dispatch(suggestionsFetched(data.suggestions.map(function(json) {
var a = new Suggestion(); var a = new Suggestion();
a.fromJSON(json); a.fromJSON(json);
return a; return a;
@ -60,15 +73,15 @@ function fetchAll() {
}; };
} }
function create(restaurant) { function create(suggestion) {
return function (dispatch) { return function (dispatch) {
dispatch(createSuggestion()); dispatch(createSuggestion());
$.ajax({ $.ajax({
type: "POST", type: "POST",
dataType: "json", dataType: "json",
url: "restaurant/", url: "suggestion/",
data: {restaurant: restaurant.toJSON()}, data: {suggestion: suggestion.toJSON()},
success: function(data, status, jqXHR) { success: function(data, status, jqXHR) {
var e = new Error(); var e = new Error();
e.fromJSON(data); e.fromJSON(data);
@ -77,7 +90,35 @@ function create(restaurant) {
} else { } else {
var a = new Suggestion(); var a = new Suggestion();
a.fromJSON(data); a.fromJSON(data);
dispatch(restaurantCreated(a)); dispatch(suggestionCreated(a));
}
},
error: function(jqXHR, status, error) {
ErrorActions.ajaxError(e);
}
});
};
}
function fetchPopular() {
return function (dispatch) {
dispatch(fetchPopularSuggestions());
$.ajax({
type: "GET",
dataType: "json",
url: "popularsuggestions/",
success: function(data, status, jqXHR) {
var e = new Error();
e.fromJSON(data);
if (e.isError()) {
ErrorActions.serverError(e);
} else {
dispatch(popularSuggestionsFetched(data.suggestions.map(function(json) {
var a = new Suggestion();
a.fromJSON(json);
return a;
})));
} }
}, },
error: function(jqXHR, status, error) { error: function(jqXHR, status, error) {
@ -89,5 +130,6 @@ function create(restaurant) {
module.exports = { module.exports = {
fetchAll: fetchAll, fetchAll: fetchAll,
create: create create: create,
fetchPopular: fetchPopular
}; };

View File

@ -93,6 +93,8 @@ function initializeSession(dispatch, session) {
dispatch(fetch(session.UserId)); dispatch(fetch(session.UserId));
dispatch(AttendeeActions.fetchAll()); dispatch(AttendeeActions.fetchAll());
dispatch(SuggestionActions.fetchAll()); dispatch(SuggestionActions.fetchAll());
dispatch(AttendeeActions.fetchPopular());
dispatch(SuggestionActions.fetchPopular());
} }
function login(user) { function login(user) {

View File

@ -4,5 +4,7 @@ module.exports = keyMirror({
FETCH_ATTENDEES: null, FETCH_ATTENDEES: null,
ATTENDEES_FETCHED: null, ATTENDEES_FETCHED: null,
CREATE_ATTENDEE: null, CREATE_ATTENDEE: null,
ATTENDEE_CREATED: null ATTENDEE_CREATED: null,
FETCH_POPULAR_ATTENDEES: null,
POPULAR_ATTENDEES_FETCHED: null
}); });

View File

@ -4,5 +4,7 @@ module.exports = keyMirror({
FETCH_SUGGESTIONS: null, FETCH_SUGGESTIONS: null,
SUGGESTIONS_FETCHED: null, SUGGESTIONS_FETCHED: null,
CREATE_SUGGESTION: null, CREATE_SUGGESTION: null,
SUGGESTION_CREATED: null SUGGESTION_CREATED: null,
FETCH_POPULAR_SUGGESTIONS: null,
POPULAR_SUGGESTIONS_FETCHED: null,
}); });

View File

@ -3,13 +3,17 @@ var Redux = require('redux');
var UserReducer = require('./UserReducer'); var UserReducer = require('./UserReducer');
var SessionReducer = require('./SessionReducer'); var SessionReducer = require('./SessionReducer');
var AttendeeReducer = require('./AttendeeReducer'); var AttendeeReducer = require('./AttendeeReducer');
var PopularAttendeeReducer = require('./PopularAttendeeReducer');
var SuggestionReducer = require('./SuggestionReducer'); var SuggestionReducer = require('./SuggestionReducer');
var PopularSuggestionReducer = require('./PopularSuggestionReducer');
var ErrorReducer = require('./ErrorReducer'); var ErrorReducer = require('./ErrorReducer');
module.exports = Redux.combineReducers({ module.exports = Redux.combineReducers({
user: UserReducer, user: UserReducer,
session: SessionReducer, session: SessionReducer,
attendees: AttendeeReducer, attendees: AttendeeReducer,
popularAttendees: PopularAttendeeReducer,
suggestions: SuggestionReducer, suggestions: SuggestionReducer,
popularSuggestions: PopularSuggestionReducer,
error: ErrorReducer error: ErrorReducer
}); });

View File

@ -0,0 +1,20 @@
var assign = require('object-assign');
var AttendeeConstants = require('../constants/AttendeeConstants');
var UserConstants = require('../constants/UserConstants');
module.exports = function(state = {}, action) {
switch (action.type) {
case AttendeeConstants.POPULAR_ATTENDEES_FETCHED:
var attendees = {};
for (var i = 0; i < action.attendees.length; i++) {
var attendee = action.attendees[i];
attendees[attendee.AttendeeId] = attendee;
}
return attendees;
case UserConstants.USER_LOGGEDOUT:
return {};
default:
return state;
}
};

View File

@ -0,0 +1,20 @@
var assign = require('object-assign');
var SuggestionConstants = require('../constants/SuggestionConstants');
var UserConstants = require('../constants/UserConstants');
module.exports = function(state = {}, action) {
switch (action.type) {
case SuggestionConstants.POPULAR_SUGGESTIONS_FETCHED:
var suggestions = {};
for (var i = 0; i < action.suggestions.length; i++) {
var suggestion = action.suggestions[i];
suggestions[suggestion.SuggestionId] = suggestion;
}
return suggestions;
case UserConstants.USER_LOGGEDOUT:
return {};
default:
return state;
}
};

View File

@ -70,7 +70,9 @@ func main() {
servemux.HandleFunc("/session/", SessionHandler) servemux.HandleFunc("/session/", SessionHandler)
servemux.HandleFunc("/user/", UserHandler) servemux.HandleFunc("/user/", UserHandler)
servemux.HandleFunc("/attendee/", AttendeeHandler) servemux.HandleFunc("/attendee/", AttendeeHandler)
servemux.HandleFunc("/popularattendees/", PopularAttendeeHandler)
servemux.HandleFunc("/suggestion/", SuggestionHandler) servemux.HandleFunc("/suggestion/", SuggestionHandler)
servemux.HandleFunc("/popularsuggestions/", PopularSuggestionHandler)
listener, err := net.Listen("tcp", ":"+strconv.Itoa(port)) listener, err := net.Listen("tcp", ":"+strconv.Itoa(port))
if err != nil { if err != nil {

View File

@ -17,10 +17,19 @@ type Suggestion struct {
Date time.Time `json:"-"` Date time.Time `json:"-"`
} }
type PopularSuggestion struct {
RestaurantName string
Popularity int64
}
type SuggestionList struct { type SuggestionList struct {
Suggestions *[]*Suggestion `json:"suggestions"` Suggestions *[]*Suggestion `json:"suggestions"`
} }
type PopularSuggestionList struct {
PopularSuggestions *[]*PopularSuggestion `json:"popularsuggestions"`
}
func (s *Suggestion) Write(w http.ResponseWriter) error { func (s *Suggestion) Write(w http.ResponseWriter) error {
enc := json.NewEncoder(w) enc := json.NewEncoder(w)
return enc.Encode(s) return enc.Encode(s)
@ -42,6 +51,16 @@ func (aeu SuggestionExistsError) Error() string {
return "Suggestion exists" return "Suggestion exists"
} }
func (s *PopularSuggestion) Write(w http.ResponseWriter) error {
enc := json.NewEncoder(w)
return enc.Encode(s)
}
func (sl *PopularSuggestionList) Write(w http.ResponseWriter) error {
enc := json.NewEncoder(w)
return enc.Encode(sl)
}
func GetSuggestions(userid int64, date time.Time) (*[]*Suggestion, error) { func GetSuggestions(userid int64, date time.Time) (*[]*Suggestion, error) {
var suggestions []*Suggestion var suggestions []*Suggestion
@ -52,6 +71,28 @@ func GetSuggestions(userid int64, date time.Time) (*[]*Suggestion, error) {
return &suggestions, nil return &suggestions, nil
} }
func GetPopularSuggestions() (*[]*PopularSuggestion, error) {
var suggestions []*Suggestion
var suggestionMap map[string]int64
popularSuggestions := make([]*PopularSuggestion, 0)
_, err := DB.Select(&suggestions, "SELECT * from suggestions")
if err != nil {
return nil, err
}
for i := range suggestions {
suggestionMap[suggestions[i].RestaurantName] += 1
}
for name, count := range suggestionMap {
var popularSuggestion PopularSuggestion
popularSuggestion.RestaurantName = name
popularSuggestion.Popularity = count
popularSuggestions = append(popularSuggestions, &popularSuggestion)
}
return &popularSuggestions, nil
}
func InsertSuggestion(s *Suggestion) error { func InsertSuggestion(s *Suggestion) error {
transaction, err := DB.Begin() transaction, err := DB.Begin()
if err != nil { if err != nil {
@ -150,3 +191,28 @@ func SuggestionHandler(w http.ResponseWriter, r *http.Request) {
return return
} }
} }
func PopularSuggestionHandler(w http.ResponseWriter, r *http.Request) {
if r.Method == "GET" {
var sl PopularSuggestionList
suggestions, err := GetPopularSuggestions()
if err != nil {
WriteError(w, 999 /*Internal Error*/)
log.Print(err)
return
}
sl.PopularSuggestions = suggestions
err = (&sl).Write(w)
if err != nil {
WriteError(w, 999 /*Internal Error*/)
log.Print(err)
return
}
} else {
/* No POST, PUT, or DELETE */
WriteError(w, 3 /*Invalid Request*/)
return
}
}