Skip to content

Commit

Permalink
Merge branch '40-redux'
Browse files Browse the repository at this point in the history
  • Loading branch information
chmac committed Oct 16, 2024
2 parents 4acdc5b + 0f5cec7 commit c261b22
Show file tree
Hide file tree
Showing 15 changed files with 481 additions and 111 deletions.
91 changes: 91 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,95 @@ There are 3 parts to this code.

Before starting development of either `nr-app` or `nr-server`, first setup the `nr-common` code as per `nr-common/README.md`.



## Roadmap

Goal: 70% of active Trustroots users are on Nostroots by middle of 2026
- active trustroots users: around 5K active within last month, 70% is around 3.5K ([Trustroots statistics](https://www.trustroots.org/statistics))
- "are on Nostroots": Have had a Nostroots experience means have some feature use that went well and is associated with Nostroots. The users don't need to recognize Nostr as the protocol, just that something is possible that wasn't before. This could be logging into a different site, transporting some of their network, or interacting with content from a different platform.

First step: Trial this in Berlin. Largest userbase, close to some of the developers.

200 yes/maybe hosts in Berlin. Around 1000 users. Estimate 5-10 people requesting hosting every week.


The technical side of things are manageable as long as we just care about Trustroots functionality. There are two big challenges for migrating our users.
- telling the story
- finding partners in the ecosystem.

### Telling the story
Trustroots [circles](https://www.trustroots.org/circles) around hippie, alternative, vanguard, experimental, left, gifting. The 2024 Nostr userbase is generally cryptocurrency and privacy focused.

As far as our users are concerned, Trustroots is fine and nothing is broken. So a degradation of their experience will likely only lead to frustration. At best, we can justify inconvenience through appealing to the values of the community. The community also won't care that much about the admins' wish to make Trustroots more maintainable.

Trustroots users interact with the app when they're looking for something in a new city. That is the moment they're engaged and ready to be excited and we should find a story that works for them. The core elements of this story should be:
- Trustroots was never meant to be just for hosting. It's meant to enable gifting and sharing based on trust and shared values.
- In a world of companies owning your identity online, Trustroots wants to empower you to own your own identity.
- There's more cool stuff like Trustroots in the world.

A lot of coordination around events and groups occurs on telegram, whatsapp and facebook, we think a nostr geo tool can do better.


### Partners in the ecosystem
We need platforms and communities that work in Berlin, are not money-focused, are valuable to travellers, and encourage personal connection and sharing. There are no good partner organisations in the current Nostr ecosystem. Our best bet will be supportive interested other groups that we build the tech for. So we need to build a good DX for adding logging in.

Possible groups and communities:
- [Bike Surf Berlin](bikesurf.org)
- Geocaching?
- [Couchers](couchers.org) and other hospex platforms
- related to [circles](https://www.trustroots.org/circles):
- Semi-legal rave groups [circle](https://www.trustroots.org/circles/ravers)
- [burners circle](https://www.trustroots.org/circles/burners)
- foodsharing.de, [circle](https://www.trustroots.org/circles/foodsharing)
- [acroyoga circle](https://www.trustroots.org/circles/acroyoga)
- [lindyhop circle](https://www.trustroots.org/circles/lindyhoppers)


### Timeline
**Q4 2024:**
- Add functionality on main trustroots site to display and link recommended organisations in Berlin
- at most 3, possibly rotating
- maybe also based on Circles?
- track what gets clicked on
- solicit experience reports and recs for other groups to display
- Build out Trustroots app
- full notes functionality
- ["login-with-trustroots" functionality](https://nips.nostr.com/46)
- putting more profile data onto Nostr with opt-in, starting with Circles

**Q1 2025:**
- Add login-with functionality to most promising one partner org
- Add more recommended orgs
- Solicit for some Berlin community management role?
- feed more data into the map and filter by Circles

**Q2 2025:**
- add nip-46 login to Trustroots app and begin encouraging users to store their nsec outside of the Trustroots app
- add login-with functionality to another partner org


### Log in with Nostr/Trustroots
#### User flow
- People search for something in Berlin
- A little sidebar informs them of other services in Berlin they might be interested in
- it includes a mention of the app and ease of using them via the app.
- User downloads app.
- They're onboarded onto Nostr
- private key generated and saved
- public key NIP-5 verified
- profile information published on the Nostr ecosystem (do we need extra consent here?).
- In the app, they can click on a link to an app and get taken straight to the service onto the "edit account" page to fill in missing information.

#### Technical Flow:
- partner site embeds javascript we provide on their website
- partner site adds `login-with-nostr` endpoint to their API
- user clicks a "login with trustroots" button on
- nip-46 flow is initiated
- browser sends a signed-by-user event with all user data to `login-with-nostr` api endpoint
- endpoint verifies event is properly signed and logs in the user and updates their user fields, creating their account first if necessary


---

## Background
Expand Down Expand Up @@ -88,3 +177,5 @@ It would be great to at some point connect with BW and Couchers over Nostr.
### tokens, dao, blockchain, other scams?

If you see "nostr token", run away, it is a scam. There's no nostr token. There was no nostr ICO, nostr is not a DAO, there is no blockchain. Nostr makes it easy to integrate bitcoin lightning, which may at some point be helpful to for example keep out spammers. But this is not something we are interested in for the foreseeable future.


14 changes: 10 additions & 4 deletions nr-app/app/(tabs)/explore.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ import { Button, StyleSheet, Text, View } from "react-native";
import ParallaxScrollView from "@/components/ParallaxScrollView";
import { ThemedText } from "@/components/ThemedText";
import { ThemedView } from "@/components/ThemedView";
import { addEvent, eventsSelectors } from "@/redux/eventsSlice";
import { addEvent, eventsSelectors } from "@/redux/slices/events.slice";
import { useAppDispatch, useAppSelector } from "@/redux/hooks";

import { Relay, finalizeEvent, verifyEvent } from "nostr-tools";
Expand Down Expand Up @@ -53,7 +53,13 @@ export default function TabTwoScreen() {
const relay = new Relay("wss://nos.lol");
await relay.connect();
const sub = relay.subscribe([{ kinds: [0], limit: 10 }], {
onevent: (event) => void dispatch(addEvent(event)),
onevent: (event) =>
void dispatch(
addEvent({
event,
fromRelay: "wss://nos.lol",
}),
),
oneose: () => {
sub.close();
},
Expand All @@ -64,8 +70,8 @@ export default function TabTwoScreen() {
<View>
<Text>We have a total of {events.length} events.</Text>
{events.map((event) => (
<View key={event.id}>
<Text>{event.id}</Text>
<View key={event.event.id}>
<Text>{event.event.id}</Text>
<Text>{JSON.stringify(event)}</Text>
</View>
))}
Expand Down
1 change: 1 addition & 0 deletions nr-app/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@
"fast-text-encoding": "^1.0.6",
"nip06": "^1.2.0",
"nostr-tools": "^2.7.2",
"open-location-code-typescript": "^1.5.0",
"react": "18.2.0",
"react-dom": "18.2.0",
"react-native": "0.74.5",
Expand Down
8 changes: 8 additions & 0 deletions nr-app/pnpm-lock.yaml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

28 changes: 27 additions & 1 deletion nr-app/src/components/Map.tsx
Original file line number Diff line number Diff line change
@@ -1,11 +1,37 @@
import {
allPlusCodesForRegion,
coordinatesToPlusCode,
} from "@/utils/map.utils";
import { StyleSheet, View } from "react-native";

import MapView, { Marker } from "react-native-maps";

export default function Map() {
return (
<View style={styles.mapContainer}>
<MapView style={styles.map}>
<MapView
style={styles.map}
rotateEnabled={false}
pitchEnabled={false}
onRegionChangeComplete={(region, details) => {
console.log("#rIMmxg Map move completed", region, details);
const topRightCoordinates = {
latitude: region.latitude + region.latitudeDelta,
longitude: region.longitude + region.longitudeDelta,
};
const bottomLeftCoordinates = {
latitude: region.latitude - region.latitudeDelta,
longitude: region.longitude - region.longitudeDelta,
};
const topRightCode = coordinatesToPlusCode(topRightCoordinates);
const bottomLeftCode = coordinatesToPlusCode(bottomLeftCoordinates);
console.log(
`#bu2PoU Bottom left is ${bottomLeftCode}, top right is ${topRightCode}`,
);
const parts = allPlusCodesForRegion(region);
console.log("#fWrvAt Got parts", parts);
}}
>
<Marker coordinate={{ latitude: 52, longitude: 13 }} title="A marker" />
</MapView>
</View>
Expand Down
3 changes: 3 additions & 0 deletions nr-app/src/constants.ts
Original file line number Diff line number Diff line change
@@ -1 +1,4 @@
export const ID_SEPARATOR = ":" as const;

export const DEFAULT_PLUS_CODE_LENGTH = 4;
export const NOSTR_EVENT_INDEX_MAXIMUM_PLUS_CODE_LENGTH = 6;
5 changes: 5 additions & 0 deletions nr-app/src/redux/actions/map.actions.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
import { createAction } from "@reduxjs/toolkit";

export const setVisiblePlusCodes = createAction<string[]>(
"map/setVisiblePlusCodes",
);
72 changes: 0 additions & 72 deletions nr-app/src/redux/eventsSlice.ts

This file was deleted.

30 changes: 0 additions & 30 deletions nr-app/src/redux/mapSlice.ts

This file was deleted.

9 changes: 7 additions & 2 deletions nr-app/src/redux/sagas/map.saga.ts
Original file line number Diff line number Diff line change
@@ -1,21 +1,26 @@
import { PayloadAction } from "@reduxjs/toolkit";
import { put, takeEvery } from "redux-saga/effects";
import {
setMapSubscriptionIsUpdating,
setVisiblePlusCodes,
} from "../slices/map.slice";

function* updateDataForMap(action: PayloadAction<string[]>) {
try {
// Setup a subscription
const visiblePlusCodes = action.payload;
console.log("#tJ7hyp Got visible plus codes", visiblePlusCodes);
// Write the state to redux
put({ type: "success" });
put(setMapSubscriptionIsUpdating(true));
// Call a subscription
} catch (error) {
const message = error instanceof Error ? error.message : "";
yield put({ type: "fail", action: message });
}
}

export function* mapSaga() {
yield takeEvery("some_action", updateDataForMap);
yield takeEvery(setVisiblePlusCodes, updateDataForMap);
}

/**
Expand Down
Loading

0 comments on commit c261b22

Please sign in to comment.